gvisor/test/util/file_descriptor.h

135 lines
4.0 KiB
C++

// Copyright 2018 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.
#ifndef GVISOR_TEST_UTIL_FILE_DESCRIPTOR_H_
#define GVISOR_TEST_UTIL_FILE_DESCRIPTOR_H_
#include <fcntl.h>
#include <sys/stat.h>
#include <sys/types.h>
#include <unistd.h>
#include <algorithm>
#include <string>
#include "gmock/gmock.h"
#include "absl/strings/str_cat.h"
#include "absl/strings/str_format.h"
#include "test/util/logging.h"
#include "test/util/posix_error.h"
#include "test/util/save_util.h"
namespace gvisor {
namespace testing {
// FileDescriptor is an RAII type class which takes ownership of a file
// descriptor. It will close the FD when this object goes out of scope.
class FileDescriptor {
public:
// Constructs an empty FileDescriptor (one that does not own a file
// descriptor).
FileDescriptor() = default;
// Constructs a FileDescriptor that owns fd. If fd is negative, constructs an
// empty FileDescriptor.
explicit FileDescriptor(int fd) { set_fd(fd); }
FileDescriptor(FileDescriptor&& orig) : fd_(orig.release()) {}
FileDescriptor& operator=(FileDescriptor&& orig) {
reset(orig.release());
return *this;
}
PosixErrorOr<FileDescriptor> Dup() const {
if (fd_ < 0) {
return PosixError(EINVAL, "Attempting to Dup unset fd");
}
int fd = dup(fd_);
if (fd < 0) {
return PosixError(errno, absl::StrCat("dup ", fd_));
}
MaybeSave();
return FileDescriptor(fd);
}
FileDescriptor(FileDescriptor const& other) = delete;
FileDescriptor& operator=(FileDescriptor const& other) = delete;
~FileDescriptor() { reset(); }
// If this object is non-empty, returns the owned file descriptor. (Ownership
// is retained by the FileDescriptor.) Otherwise returns -1.
int get() const { return fd_; }
// If this object is non-empty, transfers ownership of the file descriptor to
// the caller and returns it. Otherwise returns -1.
int release() {
int const fd = fd_;
fd_ = -1;
return fd;
}
// If this object is non-empty, closes the owned file descriptor (recording a
// test failure if the close fails).
void reset() { reset(-1); }
// Like no-arg reset(), but the FileDescriptor takes ownership of fd after
// closing its existing file descriptor.
void reset(int fd) {
if (fd_ >= 0) {
TEST_PCHECK(close(fd_) == 0);
MaybeSave();
}
set_fd(fd);
}
private:
// Wrapper that coerces negative fd values other than -1 to -1 so that get()
// etc. return -1.
void set_fd(int fd) { fd_ = std::max(fd, -1); }
int fd_ = -1;
};
// Wrapper around open(2) that returns a FileDescriptor.
inline PosixErrorOr<FileDescriptor> Open(std::string const& path, int flags,
mode_t mode = 0) {
int fd = open(path.c_str(), flags, mode);
if (fd < 0) {
return PosixError(errno, absl::StrFormat("open(%s, %#x, %#o)", path.c_str(),
flags, mode));
}
MaybeSave();
return FileDescriptor(fd);
}
// Wrapper around openat(2) that returns a FileDescriptor.
inline PosixErrorOr<FileDescriptor> OpenAt(int dirfd, std::string const& path,
int flags, mode_t mode = 0) {
int fd = openat(dirfd, path.c_str(), flags, mode);
if (fd < 0) {
return PosixError(errno, absl::StrFormat("openat(%d, %s, %#x, %#o)", dirfd,
path, flags, mode));
}
MaybeSave();
return FileDescriptor(fd);
}
} // namespace testing
} // namespace gvisor
#endif // GVISOR_TEST_UTIL_FILE_DESCRIPTOR_H_