#include "victoriafalls_server.h" #include #include #include glcr::ErrorOr> VFSServer::Create( Ext2Driver& driver) { z_cap_t endpoint_cap; RET_ERR(ZEndpointCreate(&endpoint_cap)); return glcr::UniquePtr(new VFSServer(endpoint_cap, driver)); } glcr::Status VFSServer::HandleOpenFile(const OpenFileRequest& request, OpenFileResponse& response) { auto path_tokens = glcr::StrSplit(request.path(), '/'); // Require all paths to be absolute rather than relative. // If the path starts with '/' then the first token will be empty. if (path_tokens.at(0) != "") { return glcr::InvalidArgument("Open file supports only absolute paths."); } ASSIGN_OR_RETURN(auto files, driver_.ReadDirectory(2)); for (uint64_t i = 1; i < path_tokens.size() - 1; i++) { bool found_token = false; for (uint64_t j = 0; j < files.size() && !found_token; j++) { if (path_tokens.at(i) == glcr::StringView(files.at(j).name, files.at(j).name_len)) { ASSIGN_OR_RETURN(files, driver_.ReadDirectory(files.at(j).inode)); found_token = true; } } if (!found_token) { return glcr::NotFound(glcr::StrFormat("Directory '{}' not found.", glcr::String(path_tokens.at(i)))); } } uint64_t inode_num; mmth::OwnedMemoryRegion region; for (uint64_t j = 0; j < files.size(); j++) { if (path_tokens.at(path_tokens.size() - 1) == glcr::StringView(files.at(j).name, files.at(j).name_len)) { inode_num = files.at(j).inode; ASSIGN_OR_RETURN(region, driver_.ReadFile(files.at(j).inode)); break; } } if (!region) { return glcr::NotFound( glcr::StrFormat("File '{}' not found.", glcr::String(path_tokens.at(path_tokens.size() - 1)))); } response.set_path(request.path()); // FIXME: There isn't really a reason we need to map the file into memory then // duplicate the cap. In the future just get the cap from the read then pass // it to the caller directly. response.set_memory(region.DuplicateCap()); // TODO: Consider folding this up into the actual read call. ASSIGN_OR_RETURN(Inode * inode, driver_.GetInode(inode_num)); // FIXME: This technically only sets the lower 32 bits. response.set_size(inode->size); return glcr::Status::Ok(); } glcr::Status VFSServer::HandleGetDirectory(const GetDirectoryRequest& request, Directory& response) { auto path_tokens = glcr::StrSplit(request.path(), '/'); if (path_tokens.at(0) != "") { return glcr::InvalidArgument("Get Directory only supports absolute path."); } // If there is a trailing slash we can get rid of the empty string. if (path_tokens.at(path_tokens.size() - 1) == "") { path_tokens.PopBack(); } ASSIGN_OR_RETURN(auto files, driver_.ReadDirectory(2)); for (uint64_t i = 1; i < path_tokens.size(); i++) { bool found_token = false; for (uint64_t j = 0; j < files.size() && !found_token; j++) { if (path_tokens.at(i) == glcr::StringView(files.at(j).name, files.at(j).name_len)) { ASSIGN_OR_RETURN(files, driver_.ReadDirectory(files.at(j).inode)); found_token = true; } } if (!found_token) { return glcr::NotFound(glcr::StrFormat("Directory '{}' not found.", glcr::String(path_tokens.at(i)))); } } glcr::VariableStringBuilder filelist; for (const DirEntry& file : files) { filelist.PushBack(glcr::StringView(file.name, file.name_len)); filelist.PushBack(','); } // Remove trailing comma. if (filelist.size() > 0) { filelist.DeleteLast(); } response.set_filenames(filelist.ToString()); return glcr::Status::Ok(); }