Files
gvisor/test/fuse/linux/readdir_test.cc
T
gVisor bot ae59e52979 Merge pull request #3934 from avagin:feature/fuse
PiperOrigin-RevId: 332122081
2020-09-16 17:12:57 -07:00

194 lines
6.0 KiB
C++

// Copyright 2020 The gVisor Authors.
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
// http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.
#include <dirent.h>
#include <errno.h>
#include <fcntl.h>
#include <linux/fuse.h>
#include <linux/unistd.h>
#include <sys/stat.h>
#include <sys/statfs.h>
#include <sys/types.h>
#include <unistd.h>
#include <string>
#include "gtest/gtest.h"
#include "test/fuse/linux/fuse_base.h"
#include "test/util/fuse_util.h"
#include "test/util/test_util.h"
#define FUSE_NAME_OFFSET offsetof(struct fuse_dirent, name)
#define FUSE_DIRENT_ALIGN(x) \
(((x) + sizeof(uint64_t) - 1) & ~(sizeof(uint64_t) - 1))
#define FUSE_DIRENT_SIZE(d) FUSE_DIRENT_ALIGN(FUSE_NAME_OFFSET + (d)->namelen)
namespace gvisor {
namespace testing {
namespace {
class ReaddirTest : public FuseTest {
public:
void fill_fuse_dirent(char *buf, const char *name, uint64_t ino) {
size_t namelen = strlen(name);
size_t entlen = FUSE_NAME_OFFSET + namelen;
size_t entlen_padded = FUSE_DIRENT_ALIGN(entlen);
struct fuse_dirent *dirent;
dirent = reinterpret_cast<struct fuse_dirent *>(buf);
dirent->ino = ino;
dirent->namelen = namelen;
memcpy(dirent->name, name, namelen);
memset(dirent->name + namelen, 0, entlen_padded - entlen);
}
protected:
const std::string test_dir_name_ = "test_dir";
};
TEST_F(ReaddirTest, SingleEntry) {
const std::string test_dir_path =
JoinPath(mount_point_.path().c_str(), test_dir_name_);
const uint64_t ino_dir = 1024;
// We need to make sure the test dir is a directory that can be found.
mode_t expected_mode =
S_IFDIR | S_IRWXU | S_IRGRP | S_IXGRP | S_IROTH | S_IXOTH;
struct fuse_attr dir_attr = {
.ino = ino_dir,
.size = 512,
.blocks = 4,
.mode = expected_mode,
.blksize = 4096,
};
// We need to make sure the test dir is a directory that can be found.
struct fuse_out_header lookup_header = {
.len = sizeof(struct fuse_out_header) + sizeof(struct fuse_entry_out),
};
struct fuse_entry_out lookup_payload = {
.nodeid = 1,
.entry_valid = true,
.attr_valid = true,
.attr = dir_attr,
};
struct fuse_out_header open_header = {
.len = sizeof(struct fuse_out_header) + sizeof(struct fuse_open_out),
};
struct fuse_open_out open_payload = {
.fh = 1,
};
auto iov_out = FuseGenerateIovecs(lookup_header, lookup_payload);
SetServerResponse(FUSE_LOOKUP, iov_out);
iov_out = FuseGenerateIovecs(open_header, open_payload);
SetServerResponse(FUSE_OPENDIR, iov_out);
FileDescriptor fd =
ASSERT_NO_ERRNO_AND_VALUE(Open(test_dir_path.c_str(), O_RDONLY));
// The open command makes two syscalls. Lookup the dir file and open.
// We don't need to inspect those headers in this test.
SkipServerActualRequest(); // LOOKUP.
SkipServerActualRequest(); // OPENDIR.
// Readdir test code.
std::string dot = ".";
std::string dot_dot = "..";
std::string test_file = "testFile";
// Figure out how many dirents to send over and allocate them appropriately.
// Each dirent has a dynamic name and a static metadata part. The dirent size
// is aligned to being a multiple of 8.
size_t dot_file_dirent_size =
FUSE_DIRENT_ALIGN(dot.length() + FUSE_NAME_OFFSET);
size_t dot_dot_file_dirent_size =
FUSE_DIRENT_ALIGN(dot_dot.length() + FUSE_NAME_OFFSET);
size_t test_file_dirent_size =
FUSE_DIRENT_ALIGN(test_file.length() + FUSE_NAME_OFFSET);
// Create an appropriately sized payload.
size_t readdir_payload_size =
test_file_dirent_size + dot_file_dirent_size + dot_dot_file_dirent_size;
std::vector<char> readdir_payload_vec(readdir_payload_size);
char *readdir_payload = readdir_payload_vec.data();
// Use fake ino for other directories.
fill_fuse_dirent(readdir_payload, dot.c_str(), ino_dir - 2);
fill_fuse_dirent(readdir_payload + dot_file_dirent_size, dot_dot.c_str(),
ino_dir - 1);
fill_fuse_dirent(
readdir_payload + dot_file_dirent_size + dot_dot_file_dirent_size,
test_file.c_str(), ino_dir);
struct fuse_out_header readdir_header = {
.len = uint32_t(sizeof(struct fuse_out_header) + readdir_payload_size),
};
struct fuse_out_header readdir_header_break = {
.len = uint32_t(sizeof(struct fuse_out_header)),
};
iov_out = FuseGenerateIovecs(readdir_header, readdir_payload_vec);
SetServerResponse(FUSE_READDIR, iov_out);
iov_out = FuseGenerateIovecs(readdir_header_break);
SetServerResponse(FUSE_READDIR, iov_out);
std::vector<char> buf(4090, 0);
int nread, off = 0, i = 0;
EXPECT_THAT(
nread = syscall(__NR_getdents64, fd.get(), buf.data(), buf.size()),
SyscallSucceeds());
for (; off < nread;) {
struct dirent64 *ent = (struct dirent64 *)(buf.data() + off);
off += ent->d_reclen;
switch (i++) {
case 0:
EXPECT_EQ(std::string(ent->d_name), dot);
break;
case 1:
EXPECT_EQ(std::string(ent->d_name), dot_dot);
break;
case 2:
EXPECT_EQ(std::string(ent->d_name), test_file);
break;
}
}
EXPECT_THAT(
nread = syscall(__NR_getdents64, fd.get(), buf.data(), buf.size()),
SyscallSucceedsWithValue(0));
SkipServerActualRequest(); // READDIR.
SkipServerActualRequest(); // READDIR with no data.
// Clean up.
fd.reset(-1);
struct fuse_in_header in_header;
struct fuse_release_in in_payload;
auto iov_in = FuseGenerateIovecs(in_header, in_payload);
GetServerActualRequest(iov_in);
EXPECT_EQ(in_header.len, sizeof(in_header) + sizeof(in_payload));
EXPECT_EQ(in_header.opcode, FUSE_RELEASEDIR);
}
} // namespace
} // namespace testing
} // namespace gvisor