393 lines
10 KiB
C++
393 lines
10 KiB
C++
/*
|
|
Test wrappers around POSIX functions.
|
|
|
|
Copyright (c) 2012-2014, Victor Zverovich
|
|
All rights reserved.
|
|
|
|
Redistribution and use in source and binary forms, with or without
|
|
modification, are permitted provided that the following conditions are met:
|
|
|
|
1. Redistributions of source code must retain the above copyright notice, this
|
|
list of conditions and the following disclaimer.
|
|
2. Redistributions in binary form must reproduce the above copyright notice,
|
|
this list of conditions and the following disclaimer in the documentation
|
|
and/or other materials provided with the distribution.
|
|
|
|
THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND
|
|
ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
|
|
WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
|
|
DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR
|
|
ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
|
|
(INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
|
|
LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
|
|
ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
|
(INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
|
|
SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
|
*/
|
|
|
|
#include "posix-test.h"
|
|
|
|
#include <errno.h>
|
|
#include <fcntl.h>
|
|
#include <climits>
|
|
|
|
#ifdef _WIN32
|
|
# include <io.h>
|
|
#endif
|
|
|
|
#include "gtest-extra.h"
|
|
|
|
using fmt::BufferedFile;
|
|
using fmt::ErrorCode;
|
|
using fmt::File;
|
|
|
|
namespace {
|
|
int open_count;
|
|
int close_count;
|
|
int dup_count;
|
|
int dup2_count;
|
|
int fdopen_count;
|
|
int read_count;
|
|
int write_count;
|
|
int pipe_count;
|
|
int fopen_count;
|
|
int fclose_count;
|
|
int fileno_count;
|
|
std::size_t read_nbyte;
|
|
std::size_t write_nbyte;
|
|
bool increase_file_size;
|
|
}
|
|
|
|
#define EMULATE_EINTR(func, error_result) \
|
|
if (func##_count != 0) { \
|
|
if (func##_count++ != 3) { \
|
|
errno = EINTR; \
|
|
return error_result; \
|
|
} \
|
|
}
|
|
|
|
#ifndef _WIN32
|
|
int test::open(const char *path, int oflag, int mode) {
|
|
EMULATE_EINTR(open, -1);
|
|
return ::open(path, oflag, mode);
|
|
}
|
|
|
|
static off_t max_file_size() { return std::numeric_limits<off_t>::max(); }
|
|
|
|
int test::fstat(int fd, struct stat *buf) {
|
|
int result = ::fstat(fd, buf);
|
|
if (increase_file_size)
|
|
buf->st_size = max_file_size();
|
|
return result;
|
|
}
|
|
#else
|
|
errno_t test::sopen_s(
|
|
int* pfh, const char *filename, int oflag, int shflag, int pmode) {
|
|
EMULATE_EINTR(open, EINTR);
|
|
return _sopen_s(pfh, filename, oflag, shflag, pmode);
|
|
}
|
|
|
|
static LONGLONG max_file_size() {return std::numeric_limits<LONGLONG>::max(); }
|
|
|
|
BOOL test::GetFileSizeEx(HANDLE hFile, PLARGE_INTEGER lpFileSize) {
|
|
BOOL result = GetFileSizeEx(hFile, lpFileSize);
|
|
if (increase_file_size)
|
|
lpFileSize->QuadPart = max_file_size();
|
|
return result;
|
|
}
|
|
#endif
|
|
|
|
int test::close(int fildes) {
|
|
// Close the file first because close shouldn't be retried.
|
|
int result = ::FMT_POSIX(close(fildes));
|
|
EMULATE_EINTR(close, -1);
|
|
return result;
|
|
}
|
|
|
|
int test::dup(int fildes) {
|
|
EMULATE_EINTR(dup, -1);
|
|
return ::FMT_POSIX(dup(fildes));
|
|
}
|
|
|
|
int test::dup2(int fildes, int fildes2) {
|
|
EMULATE_EINTR(dup2, -1);
|
|
return ::FMT_POSIX(dup2(fildes, fildes2));
|
|
}
|
|
|
|
FILE *test::fdopen(int fildes, const char *mode) {
|
|
EMULATE_EINTR(fdopen, 0);
|
|
return ::FMT_POSIX(fdopen(fildes, mode));
|
|
}
|
|
|
|
test::ssize_t test::read(int fildes, void *buf, test::size_t nbyte) {
|
|
read_nbyte = nbyte;
|
|
EMULATE_EINTR(read, -1);
|
|
return ::FMT_POSIX(read(fildes, buf, nbyte));
|
|
}
|
|
|
|
test::ssize_t test::write(int fildes, const void *buf, test::size_t nbyte) {
|
|
write_nbyte = nbyte;
|
|
EMULATE_EINTR(write, -1);
|
|
return ::FMT_POSIX(write(fildes, buf, nbyte));
|
|
}
|
|
|
|
#ifndef _WIN32
|
|
int test::pipe(int fildes[2]) {
|
|
EMULATE_EINTR(pipe, -1);
|
|
return ::pipe(fildes);
|
|
}
|
|
#else
|
|
int test::pipe(int *pfds, unsigned psize, int textmode) {
|
|
EMULATE_EINTR(pipe, -1);
|
|
return _pipe(pfds, psize, textmode);
|
|
}
|
|
#endif
|
|
|
|
FILE *test::fopen(const char *filename, const char *mode) {
|
|
EMULATE_EINTR(fopen, 0);
|
|
return ::fopen(filename, mode);
|
|
}
|
|
|
|
int test::fclose(FILE *stream) {
|
|
EMULATE_EINTR(fclose, EOF);
|
|
return ::fclose(stream);
|
|
}
|
|
|
|
int test::fileno(FILE *stream) {
|
|
EMULATE_EINTR(fileno, -1);
|
|
return ::FMT_POSIX(fileno(stream));
|
|
}
|
|
|
|
#ifndef _WIN32
|
|
# define EXPECT_RETRY(statement, func, message) \
|
|
func##_count = 1; \
|
|
statement; \
|
|
EXPECT_EQ(4, func##_count); \
|
|
func##_count = 0;
|
|
# define EXPECT_EQ_POSIX(expected, actual) EXPECT_EQ(expected, actual)
|
|
#else
|
|
# define EXPECT_RETRY(statement, func, message) \
|
|
func##_count = 1; \
|
|
EXPECT_SYSTEM_ERROR(statement, EINTR, message); \
|
|
func##_count = 0;
|
|
# define EXPECT_EQ_POSIX(expected, actual)
|
|
#endif
|
|
|
|
void write_file(fmt::StringRef filename, fmt::StringRef content) {
|
|
fmt::BufferedFile f(filename, "w");
|
|
f.print("{}", content);
|
|
}
|
|
|
|
TEST(UtilTest, StaticAssert) {
|
|
FMT_STATIC_ASSERT(true, "success");
|
|
// Static assertion failure is tested in compile-test because it causes
|
|
// a compile-time error.
|
|
}
|
|
|
|
TEST(FileTest, OpenRetry) {
|
|
write_file("test", "there must be something here");
|
|
File *f = 0;
|
|
EXPECT_RETRY(f = new File("test", File::RDONLY),
|
|
open, "cannot open file test");
|
|
#ifndef _WIN32
|
|
char c = 0;
|
|
f->read(&c, 1);
|
|
#endif
|
|
delete f;
|
|
}
|
|
|
|
TEST(FileTest, CloseNoRetryInDtor) {
|
|
File read_end, write_end;
|
|
File::pipe(read_end, write_end);
|
|
File *f = new File(std::move(read_end));
|
|
int saved_close_count = 0;
|
|
EXPECT_WRITE(stderr, {
|
|
close_count = 1;
|
|
delete f;
|
|
saved_close_count = close_count;
|
|
close_count = 0;
|
|
}, format_system_error(EINTR, "cannot close file") + "\n");
|
|
EXPECT_EQ(2, saved_close_count);
|
|
}
|
|
|
|
TEST(FileTest, CloseNoRetry) {
|
|
File read_end, write_end;
|
|
File::pipe(read_end, write_end);
|
|
close_count = 1;
|
|
EXPECT_SYSTEM_ERROR(read_end.close(), EINTR, "cannot close file");
|
|
EXPECT_EQ(2, close_count);
|
|
close_count = 0;
|
|
}
|
|
|
|
TEST(FileTest, Size) {
|
|
std::string content = "top secret, destroy before reading";
|
|
write_file("test", content);
|
|
File f("test", File::RDONLY);
|
|
EXPECT_EQ(content.size(), f.size());
|
|
f.close();
|
|
EXPECT_SYSTEM_ERROR(f.size(), EBADF, "cannot get file attributes");
|
|
}
|
|
|
|
TEST(FileTest, MaxSize) {
|
|
write_file("test", "");
|
|
File f("test", File::RDONLY);
|
|
increase_file_size = true;
|
|
EXPECT_GE(f.size(), 0);
|
|
EXPECT_EQ(max_file_size(), f.size());
|
|
increase_file_size = false;
|
|
}
|
|
|
|
TEST(FileTest, ReadRetry) {
|
|
File read_end, write_end;
|
|
File::pipe(read_end, write_end);
|
|
enum { SIZE = 4 };
|
|
write_end.write("test", SIZE);
|
|
write_end.close();
|
|
char buffer[SIZE];
|
|
std::streamsize count = 0;
|
|
EXPECT_RETRY(count = read_end.read(buffer, SIZE),
|
|
read, "cannot read from file");
|
|
EXPECT_EQ_POSIX(static_cast<std::streamsize>(SIZE), count);
|
|
}
|
|
|
|
TEST(FileTest, WriteRetry) {
|
|
File read_end, write_end;
|
|
File::pipe(read_end, write_end);
|
|
enum { SIZE = 4 };
|
|
std::streamsize count = 0;
|
|
EXPECT_RETRY(count = write_end.write("test", SIZE),
|
|
write, "cannot write to file");
|
|
write_end.close();
|
|
#ifndef _WIN32
|
|
EXPECT_EQ(static_cast<std::streamsize>(SIZE), count);
|
|
char buffer[SIZE + 1];
|
|
read_end.read(buffer, SIZE);
|
|
buffer[SIZE] = '\0';
|
|
EXPECT_STREQ("test", buffer);
|
|
#endif
|
|
}
|
|
|
|
#ifdef _WIN32
|
|
TEST(FileTest, ConvertReadCount) {
|
|
File read_end, write_end;
|
|
File::pipe(read_end, write_end);
|
|
char c;
|
|
std::size_t size = UINT_MAX;
|
|
if (sizeof(unsigned) != sizeof(std::size_t))
|
|
++size;
|
|
read_count = 1;
|
|
read_nbyte = 0;
|
|
EXPECT_THROW(read_end.read(&c, size), fmt::SystemError);
|
|
read_count = 0;
|
|
EXPECT_EQ(UINT_MAX, read_nbyte);
|
|
}
|
|
|
|
TEST(FileTest, ConvertWriteCount) {
|
|
File read_end, write_end;
|
|
File::pipe(read_end, write_end);
|
|
char c;
|
|
std::size_t size = UINT_MAX;
|
|
if (sizeof(unsigned) != sizeof(std::size_t))
|
|
++size;
|
|
write_count = 1;
|
|
write_nbyte = 0;
|
|
EXPECT_THROW(write_end.write(&c, size), fmt::SystemError);
|
|
write_count = 0;
|
|
EXPECT_EQ(UINT_MAX, write_nbyte);
|
|
}
|
|
#endif
|
|
|
|
TEST(FileTest, DupNoRetry) {
|
|
int stdout_fd = FMT_POSIX(fileno(stdout));
|
|
dup_count = 1;
|
|
EXPECT_SYSTEM_ERROR(File::dup(stdout_fd), EINTR,
|
|
fmt::format("cannot duplicate file descriptor {}", stdout_fd));
|
|
dup_count = 0;
|
|
}
|
|
|
|
TEST(FileTest, Dup2Retry) {
|
|
int stdout_fd = FMT_POSIX(fileno(stdout));
|
|
File f1 = File::dup(stdout_fd), f2 = File::dup(stdout_fd);
|
|
EXPECT_RETRY(f1.dup2(f2.descriptor()), dup2,
|
|
fmt::format("cannot duplicate file descriptor {} to {}",
|
|
f1.descriptor(), f2.descriptor()));
|
|
}
|
|
|
|
TEST(FileTest, Dup2NoExceptRetry) {
|
|
int stdout_fd = FMT_POSIX(fileno(stdout));
|
|
File f1 = File::dup(stdout_fd), f2 = File::dup(stdout_fd);
|
|
ErrorCode ec;
|
|
dup2_count = 1;
|
|
f1.dup2(f2.descriptor(), ec);
|
|
#ifndef _WIN32
|
|
EXPECT_EQ(4, dup2_count);
|
|
#else
|
|
EXPECT_EQ(EINTR, ec.get());
|
|
#endif
|
|
dup2_count = 0;
|
|
}
|
|
|
|
TEST(FileTest, PipeNoRetry) {
|
|
File read_end, write_end;
|
|
pipe_count = 1;
|
|
EXPECT_SYSTEM_ERROR(
|
|
File::pipe(read_end, write_end), EINTR, "cannot create pipe");
|
|
pipe_count = 0;
|
|
}
|
|
|
|
TEST(FileTest, FdopenNoRetry) {
|
|
File read_end, write_end;
|
|
File::pipe(read_end, write_end);
|
|
fdopen_count = 1;
|
|
EXPECT_SYSTEM_ERROR(read_end.fdopen("r"),
|
|
EINTR, "cannot associate stream with file descriptor");
|
|
fdopen_count = 0;
|
|
}
|
|
|
|
TEST(BufferedFileTest, OpenRetry) {
|
|
write_file("test", "there must be something here");
|
|
BufferedFile *f = 0;
|
|
EXPECT_RETRY(f = new BufferedFile("test", "r"),
|
|
fopen, "cannot open file test");
|
|
#ifndef _WIN32
|
|
char c = 0;
|
|
if (fread(&c, 1, 1, f->get()) < 1)
|
|
throw fmt::SystemError(errno, "fread failed");
|
|
#endif
|
|
delete f;
|
|
}
|
|
|
|
TEST(BufferedFileTest, CloseNoRetryInDtor) {
|
|
File read_end, write_end;
|
|
File::pipe(read_end, write_end);
|
|
BufferedFile *f = new BufferedFile(read_end.fdopen("r"));
|
|
int saved_fclose_count = 0;
|
|
EXPECT_WRITE(stderr, {
|
|
fclose_count = 1;
|
|
delete f;
|
|
saved_fclose_count = fclose_count;
|
|
fclose_count = 0;
|
|
}, format_system_error(EINTR, "cannot close file") + "\n");
|
|
EXPECT_EQ(2, saved_fclose_count);
|
|
}
|
|
|
|
TEST(BufferedFileTest, CloseNoRetry) {
|
|
File read_end, write_end;
|
|
File::pipe(read_end, write_end);
|
|
BufferedFile f = read_end.fdopen("r");
|
|
fclose_count = 1;
|
|
EXPECT_SYSTEM_ERROR(f.close(), EINTR, "cannot close file");
|
|
EXPECT_EQ(2, fclose_count);
|
|
fclose_count = 0;
|
|
}
|
|
|
|
TEST(BufferedFileTest, FilenoNoRetry) {
|
|
File read_end, write_end;
|
|
File::pipe(read_end, write_end);
|
|
BufferedFile f = read_end.fdopen("r");
|
|
fileno_count = 1;
|
|
EXPECT_SYSTEM_ERROR(f.fileno(), EINTR, "cannot get file descriptor");
|
|
EXPECT_EQ(2, fileno_count);
|
|
fileno_count = 0;
|
|
}
|