1*5c90c05cSAndroid Build Coastguard Worker // Tests of the C++ interface to POSIX functions that require mocks
2*5c90c05cSAndroid Build Coastguard Worker //
3*5c90c05cSAndroid Build Coastguard Worker // Copyright (c) 2012 - present, Victor Zverovich
4*5c90c05cSAndroid Build Coastguard Worker // All rights reserved.
5*5c90c05cSAndroid Build Coastguard Worker //
6*5c90c05cSAndroid Build Coastguard Worker // For the license information refer to format.h.
7*5c90c05cSAndroid Build Coastguard Worker
8*5c90c05cSAndroid Build Coastguard Worker // Disable bogus MSVC warnings.
9*5c90c05cSAndroid Build Coastguard Worker #if !defined(_CRT_SECURE_NO_WARNINGS) && defined(_MSC_VER)
10*5c90c05cSAndroid Build Coastguard Worker # define _CRT_SECURE_NO_WARNINGS
11*5c90c05cSAndroid Build Coastguard Worker #endif
12*5c90c05cSAndroid Build Coastguard Worker
13*5c90c05cSAndroid Build Coastguard Worker #include "posix-mock.h"
14*5c90c05cSAndroid Build Coastguard Worker
15*5c90c05cSAndroid Build Coastguard Worker #include <errno.h>
16*5c90c05cSAndroid Build Coastguard Worker #include <fcntl.h>
17*5c90c05cSAndroid Build Coastguard Worker
18*5c90c05cSAndroid Build Coastguard Worker #include <climits>
19*5c90c05cSAndroid Build Coastguard Worker #include <memory>
20*5c90c05cSAndroid Build Coastguard Worker
21*5c90c05cSAndroid Build Coastguard Worker #include "../src/os.cc"
22*5c90c05cSAndroid Build Coastguard Worker
23*5c90c05cSAndroid Build Coastguard Worker #ifdef _WIN32
24*5c90c05cSAndroid Build Coastguard Worker # include <io.h>
25*5c90c05cSAndroid Build Coastguard Worker # undef max
26*5c90c05cSAndroid Build Coastguard Worker #endif
27*5c90c05cSAndroid Build Coastguard Worker
28*5c90c05cSAndroid Build Coastguard Worker #include "gmock/gmock.h"
29*5c90c05cSAndroid Build Coastguard Worker #include "gtest-extra.h"
30*5c90c05cSAndroid Build Coastguard Worker #include "util.h"
31*5c90c05cSAndroid Build Coastguard Worker
32*5c90c05cSAndroid Build Coastguard Worker using fmt::buffered_file;
33*5c90c05cSAndroid Build Coastguard Worker
34*5c90c05cSAndroid Build Coastguard Worker using testing::_;
35*5c90c05cSAndroid Build Coastguard Worker using testing::Return;
36*5c90c05cSAndroid Build Coastguard Worker using testing::StrEq;
37*5c90c05cSAndroid Build Coastguard Worker
38*5c90c05cSAndroid Build Coastguard Worker template <typename Mock> struct scoped_mock : testing::StrictMock<Mock> {
scoped_mockscoped_mock39*5c90c05cSAndroid Build Coastguard Worker scoped_mock() { Mock::instance = this; }
~scoped_mockscoped_mock40*5c90c05cSAndroid Build Coastguard Worker ~scoped_mock() { Mock::instance = nullptr; }
41*5c90c05cSAndroid Build Coastguard Worker };
42*5c90c05cSAndroid Build Coastguard Worker
43*5c90c05cSAndroid Build Coastguard Worker namespace {
44*5c90c05cSAndroid Build Coastguard Worker int open_count;
45*5c90c05cSAndroid Build Coastguard Worker int close_count;
46*5c90c05cSAndroid Build Coastguard Worker int dup_count;
47*5c90c05cSAndroid Build Coastguard Worker int dup2_count;
48*5c90c05cSAndroid Build Coastguard Worker int fdopen_count;
49*5c90c05cSAndroid Build Coastguard Worker int read_count;
50*5c90c05cSAndroid Build Coastguard Worker int write_count;
51*5c90c05cSAndroid Build Coastguard Worker int pipe_count;
52*5c90c05cSAndroid Build Coastguard Worker int fopen_count;
53*5c90c05cSAndroid Build Coastguard Worker int fclose_count;
54*5c90c05cSAndroid Build Coastguard Worker int fileno_count;
55*5c90c05cSAndroid Build Coastguard Worker size_t read_nbyte;
56*5c90c05cSAndroid Build Coastguard Worker size_t write_nbyte;
57*5c90c05cSAndroid Build Coastguard Worker bool sysconf_error;
58*5c90c05cSAndroid Build Coastguard Worker
59*5c90c05cSAndroid Build Coastguard Worker enum { none, max_size, error } fstat_sim;
60*5c90c05cSAndroid Build Coastguard Worker } // namespace
61*5c90c05cSAndroid Build Coastguard Worker
62*5c90c05cSAndroid Build Coastguard Worker #define EMULATE_EINTR(func, error_result) \
63*5c90c05cSAndroid Build Coastguard Worker if (func##_count != 0) { \
64*5c90c05cSAndroid Build Coastguard Worker if (func##_count++ != 3) { \
65*5c90c05cSAndroid Build Coastguard Worker errno = EINTR; \
66*5c90c05cSAndroid Build Coastguard Worker return error_result; \
67*5c90c05cSAndroid Build Coastguard Worker } \
68*5c90c05cSAndroid Build Coastguard Worker }
69*5c90c05cSAndroid Build Coastguard Worker
70*5c90c05cSAndroid Build Coastguard Worker #ifndef _MSC_VER
open(const char * path,int oflag,int mode)71*5c90c05cSAndroid Build Coastguard Worker int test::open(const char* path, int oflag, int mode) {
72*5c90c05cSAndroid Build Coastguard Worker EMULATE_EINTR(open, -1);
73*5c90c05cSAndroid Build Coastguard Worker return ::open(path, oflag, mode);
74*5c90c05cSAndroid Build Coastguard Worker }
75*5c90c05cSAndroid Build Coastguard Worker #endif
76*5c90c05cSAndroid Build Coastguard Worker
77*5c90c05cSAndroid Build Coastguard Worker #ifndef _WIN32
78*5c90c05cSAndroid Build Coastguard Worker
sysconf(int name)79*5c90c05cSAndroid Build Coastguard Worker long test::sysconf(int name) {
80*5c90c05cSAndroid Build Coastguard Worker long result = ::sysconf(name);
81*5c90c05cSAndroid Build Coastguard Worker if (!sysconf_error) return result;
82*5c90c05cSAndroid Build Coastguard Worker // Simulate an error.
83*5c90c05cSAndroid Build Coastguard Worker errno = EINVAL;
84*5c90c05cSAndroid Build Coastguard Worker return -1;
85*5c90c05cSAndroid Build Coastguard Worker }
86*5c90c05cSAndroid Build Coastguard Worker
max_file_size()87*5c90c05cSAndroid Build Coastguard Worker static off_t max_file_size() { return std::numeric_limits<off_t>::max(); }
88*5c90c05cSAndroid Build Coastguard Worker
fstat(int fd,struct stat * buf)89*5c90c05cSAndroid Build Coastguard Worker int test::fstat(int fd, struct stat* buf) {
90*5c90c05cSAndroid Build Coastguard Worker int result = ::fstat(fd, buf);
91*5c90c05cSAndroid Build Coastguard Worker if (fstat_sim == max_size) buf->st_size = max_file_size();
92*5c90c05cSAndroid Build Coastguard Worker return result;
93*5c90c05cSAndroid Build Coastguard Worker }
94*5c90c05cSAndroid Build Coastguard Worker
95*5c90c05cSAndroid Build Coastguard Worker #else
96*5c90c05cSAndroid Build Coastguard Worker
max_file_size()97*5c90c05cSAndroid Build Coastguard Worker static LONGLONG max_file_size() { return std::numeric_limits<LONGLONG>::max(); }
98*5c90c05cSAndroid Build Coastguard Worker
GetFileSize(HANDLE hFile,LPDWORD lpFileSizeHigh)99*5c90c05cSAndroid Build Coastguard Worker DWORD test::GetFileSize(HANDLE hFile, LPDWORD lpFileSizeHigh) {
100*5c90c05cSAndroid Build Coastguard Worker if (fstat_sim == error) {
101*5c90c05cSAndroid Build Coastguard Worker SetLastError(ERROR_ACCESS_DENIED);
102*5c90c05cSAndroid Build Coastguard Worker return INVALID_FILE_SIZE;
103*5c90c05cSAndroid Build Coastguard Worker }
104*5c90c05cSAndroid Build Coastguard Worker if (fstat_sim == max_size) {
105*5c90c05cSAndroid Build Coastguard Worker DWORD max = std::numeric_limits<DWORD>::max();
106*5c90c05cSAndroid Build Coastguard Worker *lpFileSizeHigh = max >> 1;
107*5c90c05cSAndroid Build Coastguard Worker return max;
108*5c90c05cSAndroid Build Coastguard Worker }
109*5c90c05cSAndroid Build Coastguard Worker return ::GetFileSize(hFile, lpFileSizeHigh);
110*5c90c05cSAndroid Build Coastguard Worker }
111*5c90c05cSAndroid Build Coastguard Worker
112*5c90c05cSAndroid Build Coastguard Worker #endif
113*5c90c05cSAndroid Build Coastguard Worker
close(int fildes)114*5c90c05cSAndroid Build Coastguard Worker int test::close(int fildes) {
115*5c90c05cSAndroid Build Coastguard Worker // Close the file first because close shouldn't be retried.
116*5c90c05cSAndroid Build Coastguard Worker int result = ::FMT_POSIX(close(fildes));
117*5c90c05cSAndroid Build Coastguard Worker EMULATE_EINTR(close, -1);
118*5c90c05cSAndroid Build Coastguard Worker return result;
119*5c90c05cSAndroid Build Coastguard Worker }
120*5c90c05cSAndroid Build Coastguard Worker
dup(int fildes)121*5c90c05cSAndroid Build Coastguard Worker int test::dup(int fildes) {
122*5c90c05cSAndroid Build Coastguard Worker EMULATE_EINTR(dup, -1);
123*5c90c05cSAndroid Build Coastguard Worker return ::FMT_POSIX(dup(fildes));
124*5c90c05cSAndroid Build Coastguard Worker }
125*5c90c05cSAndroid Build Coastguard Worker
dup2(int fildes,int fildes2)126*5c90c05cSAndroid Build Coastguard Worker int test::dup2(int fildes, int fildes2) {
127*5c90c05cSAndroid Build Coastguard Worker EMULATE_EINTR(dup2, -1);
128*5c90c05cSAndroid Build Coastguard Worker return ::FMT_POSIX(dup2(fildes, fildes2));
129*5c90c05cSAndroid Build Coastguard Worker }
130*5c90c05cSAndroid Build Coastguard Worker
fdopen(int fildes,const char * mode)131*5c90c05cSAndroid Build Coastguard Worker FILE* test::fdopen(int fildes, const char* mode) {
132*5c90c05cSAndroid Build Coastguard Worker EMULATE_EINTR(fdopen, nullptr);
133*5c90c05cSAndroid Build Coastguard Worker return ::FMT_POSIX(fdopen(fildes, mode));
134*5c90c05cSAndroid Build Coastguard Worker }
135*5c90c05cSAndroid Build Coastguard Worker
read(int fildes,void * buf,test::size_t nbyte)136*5c90c05cSAndroid Build Coastguard Worker test::ssize_t test::read(int fildes, void* buf, test::size_t nbyte) {
137*5c90c05cSAndroid Build Coastguard Worker read_nbyte = nbyte;
138*5c90c05cSAndroid Build Coastguard Worker EMULATE_EINTR(read, -1);
139*5c90c05cSAndroid Build Coastguard Worker return ::FMT_POSIX(read(fildes, buf, nbyte));
140*5c90c05cSAndroid Build Coastguard Worker }
141*5c90c05cSAndroid Build Coastguard Worker
write(int fildes,const void * buf,test::size_t nbyte)142*5c90c05cSAndroid Build Coastguard Worker test::ssize_t test::write(int fildes, const void* buf, test::size_t nbyte) {
143*5c90c05cSAndroid Build Coastguard Worker write_nbyte = nbyte;
144*5c90c05cSAndroid Build Coastguard Worker EMULATE_EINTR(write, -1);
145*5c90c05cSAndroid Build Coastguard Worker return ::FMT_POSIX(write(fildes, buf, nbyte));
146*5c90c05cSAndroid Build Coastguard Worker }
147*5c90c05cSAndroid Build Coastguard Worker
148*5c90c05cSAndroid Build Coastguard Worker #ifndef _WIN32
pipe(int fildes[2])149*5c90c05cSAndroid Build Coastguard Worker int test::pipe(int fildes[2]) {
150*5c90c05cSAndroid Build Coastguard Worker EMULATE_EINTR(pipe, -1);
151*5c90c05cSAndroid Build Coastguard Worker return ::pipe(fildes);
152*5c90c05cSAndroid Build Coastguard Worker }
153*5c90c05cSAndroid Build Coastguard Worker #else
pipe(int * pfds,unsigned psize,int textmode)154*5c90c05cSAndroid Build Coastguard Worker int test::pipe(int* pfds, unsigned psize, int textmode) {
155*5c90c05cSAndroid Build Coastguard Worker EMULATE_EINTR(pipe, -1);
156*5c90c05cSAndroid Build Coastguard Worker return _pipe(pfds, psize, textmode);
157*5c90c05cSAndroid Build Coastguard Worker }
158*5c90c05cSAndroid Build Coastguard Worker #endif
159*5c90c05cSAndroid Build Coastguard Worker
fopen(const char * filename,const char * mode)160*5c90c05cSAndroid Build Coastguard Worker FILE* test::fopen(const char* filename, const char* mode) {
161*5c90c05cSAndroid Build Coastguard Worker EMULATE_EINTR(fopen, nullptr);
162*5c90c05cSAndroid Build Coastguard Worker return ::fopen(filename, mode);
163*5c90c05cSAndroid Build Coastguard Worker }
164*5c90c05cSAndroid Build Coastguard Worker
fclose(FILE * stream)165*5c90c05cSAndroid Build Coastguard Worker int test::fclose(FILE* stream) {
166*5c90c05cSAndroid Build Coastguard Worker EMULATE_EINTR(fclose, EOF);
167*5c90c05cSAndroid Build Coastguard Worker return ::fclose(stream);
168*5c90c05cSAndroid Build Coastguard Worker }
169*5c90c05cSAndroid Build Coastguard Worker
170*5c90c05cSAndroid Build Coastguard Worker int(test::fileno)(FILE* stream) {
171*5c90c05cSAndroid Build Coastguard Worker EMULATE_EINTR(fileno, -1);
172*5c90c05cSAndroid Build Coastguard Worker #ifdef fileno
173*5c90c05cSAndroid Build Coastguard Worker return FMT_POSIX(fileno(stream));
174*5c90c05cSAndroid Build Coastguard Worker #else
175*5c90c05cSAndroid Build Coastguard Worker return ::FMT_POSIX(fileno(stream));
176*5c90c05cSAndroid Build Coastguard Worker #endif
177*5c90c05cSAndroid Build Coastguard Worker }
178*5c90c05cSAndroid Build Coastguard Worker
179*5c90c05cSAndroid Build Coastguard Worker #ifndef _WIN32
180*5c90c05cSAndroid Build Coastguard Worker # define EXPECT_RETRY(statement, func, message) \
181*5c90c05cSAndroid Build Coastguard Worker func##_count = 1; \
182*5c90c05cSAndroid Build Coastguard Worker statement; \
183*5c90c05cSAndroid Build Coastguard Worker EXPECT_EQ(4, func##_count); \
184*5c90c05cSAndroid Build Coastguard Worker func##_count = 0;
185*5c90c05cSAndroid Build Coastguard Worker # define EXPECT_EQ_POSIX(expected, actual) EXPECT_EQ(expected, actual)
186*5c90c05cSAndroid Build Coastguard Worker #else
187*5c90c05cSAndroid Build Coastguard Worker # define EXPECT_RETRY(statement, func, message) \
188*5c90c05cSAndroid Build Coastguard Worker func##_count = 1; \
189*5c90c05cSAndroid Build Coastguard Worker EXPECT_SYSTEM_ERROR(statement, EINTR, message); \
190*5c90c05cSAndroid Build Coastguard Worker func##_count = 0;
191*5c90c05cSAndroid Build Coastguard Worker # define EXPECT_EQ_POSIX(expected, actual)
192*5c90c05cSAndroid Build Coastguard Worker #endif
193*5c90c05cSAndroid Build Coastguard Worker
194*5c90c05cSAndroid Build Coastguard Worker #if FMT_USE_FCNTL
write_file(fmt::cstring_view filename,fmt::string_view content)195*5c90c05cSAndroid Build Coastguard Worker void write_file(fmt::cstring_view filename, fmt::string_view content) {
196*5c90c05cSAndroid Build Coastguard Worker fmt::buffered_file f(filename, "w");
197*5c90c05cSAndroid Build Coastguard Worker f.print("{}", content);
198*5c90c05cSAndroid Build Coastguard Worker }
199*5c90c05cSAndroid Build Coastguard Worker
200*5c90c05cSAndroid Build Coastguard Worker using fmt::file;
201*5c90c05cSAndroid Build Coastguard Worker
TEST(os_test,getpagesize)202*5c90c05cSAndroid Build Coastguard Worker TEST(os_test, getpagesize) {
203*5c90c05cSAndroid Build Coastguard Worker # ifdef _WIN32
204*5c90c05cSAndroid Build Coastguard Worker SYSTEM_INFO si = {};
205*5c90c05cSAndroid Build Coastguard Worker GetSystemInfo(&si);
206*5c90c05cSAndroid Build Coastguard Worker EXPECT_EQ(si.dwPageSize, fmt::getpagesize());
207*5c90c05cSAndroid Build Coastguard Worker # else
208*5c90c05cSAndroid Build Coastguard Worker EXPECT_EQ(sysconf(_SC_PAGESIZE), fmt::getpagesize());
209*5c90c05cSAndroid Build Coastguard Worker sysconf_error = true;
210*5c90c05cSAndroid Build Coastguard Worker EXPECT_SYSTEM_ERROR(fmt::getpagesize(), EINVAL,
211*5c90c05cSAndroid Build Coastguard Worker "cannot get memory page size");
212*5c90c05cSAndroid Build Coastguard Worker sysconf_error = false;
213*5c90c05cSAndroid Build Coastguard Worker # endif
214*5c90c05cSAndroid Build Coastguard Worker }
215*5c90c05cSAndroid Build Coastguard Worker
TEST(file_test,open_retry)216*5c90c05cSAndroid Build Coastguard Worker TEST(file_test, open_retry) {
217*5c90c05cSAndroid Build Coastguard Worker # ifndef _WIN32
218*5c90c05cSAndroid Build Coastguard Worker write_file("temp", "there must be something here");
219*5c90c05cSAndroid Build Coastguard Worker std::unique_ptr<file> f{nullptr};
220*5c90c05cSAndroid Build Coastguard Worker EXPECT_RETRY(f.reset(new file("temp", file::RDONLY)), open,
221*5c90c05cSAndroid Build Coastguard Worker "cannot open file temp");
222*5c90c05cSAndroid Build Coastguard Worker char c = 0;
223*5c90c05cSAndroid Build Coastguard Worker f->read(&c, 1);
224*5c90c05cSAndroid Build Coastguard Worker # endif
225*5c90c05cSAndroid Build Coastguard Worker }
226*5c90c05cSAndroid Build Coastguard Worker
TEST(file_test,close_no_retry_in_dtor)227*5c90c05cSAndroid Build Coastguard Worker TEST(file_test, close_no_retry_in_dtor) {
228*5c90c05cSAndroid Build Coastguard Worker auto pipe = fmt::pipe();
229*5c90c05cSAndroid Build Coastguard Worker std::unique_ptr<file> f(new file(std::move(pipe.read_end)));
230*5c90c05cSAndroid Build Coastguard Worker int saved_close_count = 0;
231*5c90c05cSAndroid Build Coastguard Worker EXPECT_WRITE(
232*5c90c05cSAndroid Build Coastguard Worker stderr,
233*5c90c05cSAndroid Build Coastguard Worker {
234*5c90c05cSAndroid Build Coastguard Worker close_count = 1;
235*5c90c05cSAndroid Build Coastguard Worker f.reset(nullptr);
236*5c90c05cSAndroid Build Coastguard Worker saved_close_count = close_count;
237*5c90c05cSAndroid Build Coastguard Worker close_count = 0;
238*5c90c05cSAndroid Build Coastguard Worker },
239*5c90c05cSAndroid Build Coastguard Worker system_error_message(EINTR, "cannot close file") + "\n");
240*5c90c05cSAndroid Build Coastguard Worker EXPECT_EQ(2, saved_close_count);
241*5c90c05cSAndroid Build Coastguard Worker }
242*5c90c05cSAndroid Build Coastguard Worker
TEST(file_test,close_no_retry)243*5c90c05cSAndroid Build Coastguard Worker TEST(file_test, close_no_retry) {
244*5c90c05cSAndroid Build Coastguard Worker auto pipe = fmt::pipe();
245*5c90c05cSAndroid Build Coastguard Worker close_count = 1;
246*5c90c05cSAndroid Build Coastguard Worker EXPECT_SYSTEM_ERROR(pipe.read_end.close(), EINTR, "cannot close file");
247*5c90c05cSAndroid Build Coastguard Worker EXPECT_EQ(2, close_count);
248*5c90c05cSAndroid Build Coastguard Worker close_count = 0;
249*5c90c05cSAndroid Build Coastguard Worker }
250*5c90c05cSAndroid Build Coastguard Worker
TEST(file_test,size)251*5c90c05cSAndroid Build Coastguard Worker TEST(file_test, size) {
252*5c90c05cSAndroid Build Coastguard Worker std::string content = "top secret, destroy before reading";
253*5c90c05cSAndroid Build Coastguard Worker write_file("temp", content);
254*5c90c05cSAndroid Build Coastguard Worker file f("temp", file::RDONLY);
255*5c90c05cSAndroid Build Coastguard Worker EXPECT_GE(f.size(), 0);
256*5c90c05cSAndroid Build Coastguard Worker EXPECT_EQ(content.size(), static_cast<unsigned long long>(f.size()));
257*5c90c05cSAndroid Build Coastguard Worker # ifdef _WIN32
258*5c90c05cSAndroid Build Coastguard Worker auto error_code = std::error_code();
259*5c90c05cSAndroid Build Coastguard Worker fstat_sim = error;
260*5c90c05cSAndroid Build Coastguard Worker try {
261*5c90c05cSAndroid Build Coastguard Worker f.size();
262*5c90c05cSAndroid Build Coastguard Worker } catch (const std::system_error& e) {
263*5c90c05cSAndroid Build Coastguard Worker error_code = e.code();
264*5c90c05cSAndroid Build Coastguard Worker }
265*5c90c05cSAndroid Build Coastguard Worker fstat_sim = none;
266*5c90c05cSAndroid Build Coastguard Worker EXPECT_EQ(error_code,
267*5c90c05cSAndroid Build Coastguard Worker std::error_code(ERROR_ACCESS_DENIED, fmt::system_category()));
268*5c90c05cSAndroid Build Coastguard Worker # else
269*5c90c05cSAndroid Build Coastguard Worker f.close();
270*5c90c05cSAndroid Build Coastguard Worker EXPECT_SYSTEM_ERROR(f.size(), EBADF, "cannot get file attributes");
271*5c90c05cSAndroid Build Coastguard Worker # endif
272*5c90c05cSAndroid Build Coastguard Worker }
273*5c90c05cSAndroid Build Coastguard Worker
TEST(file_test,max_size)274*5c90c05cSAndroid Build Coastguard Worker TEST(file_test, max_size) {
275*5c90c05cSAndroid Build Coastguard Worker write_file("temp", "");
276*5c90c05cSAndroid Build Coastguard Worker file f("temp", file::RDONLY);
277*5c90c05cSAndroid Build Coastguard Worker fstat_sim = max_size;
278*5c90c05cSAndroid Build Coastguard Worker EXPECT_GE(f.size(), 0);
279*5c90c05cSAndroid Build Coastguard Worker EXPECT_EQ(max_file_size(), f.size());
280*5c90c05cSAndroid Build Coastguard Worker fstat_sim = none;
281*5c90c05cSAndroid Build Coastguard Worker }
282*5c90c05cSAndroid Build Coastguard Worker
TEST(file_test,read_retry)283*5c90c05cSAndroid Build Coastguard Worker TEST(file_test, read_retry) {
284*5c90c05cSAndroid Build Coastguard Worker auto pipe = fmt::pipe();
285*5c90c05cSAndroid Build Coastguard Worker enum { SIZE = 4 };
286*5c90c05cSAndroid Build Coastguard Worker pipe.write_end.write("test", SIZE);
287*5c90c05cSAndroid Build Coastguard Worker pipe.write_end.close();
288*5c90c05cSAndroid Build Coastguard Worker char buffer[SIZE];
289*5c90c05cSAndroid Build Coastguard Worker size_t count = 0;
290*5c90c05cSAndroid Build Coastguard Worker EXPECT_RETRY(count = pipe.read_end.read(buffer, SIZE), read,
291*5c90c05cSAndroid Build Coastguard Worker "cannot read from file");
292*5c90c05cSAndroid Build Coastguard Worker EXPECT_EQ_POSIX(static_cast<std::streamsize>(SIZE), count);
293*5c90c05cSAndroid Build Coastguard Worker }
294*5c90c05cSAndroid Build Coastguard Worker
TEST(file_test,write_retry)295*5c90c05cSAndroid Build Coastguard Worker TEST(file_test, write_retry) {
296*5c90c05cSAndroid Build Coastguard Worker auto pipe = fmt::pipe();
297*5c90c05cSAndroid Build Coastguard Worker enum { SIZE = 4 };
298*5c90c05cSAndroid Build Coastguard Worker size_t count = 0;
299*5c90c05cSAndroid Build Coastguard Worker EXPECT_RETRY(count = pipe.write_end.write("test", SIZE), write,
300*5c90c05cSAndroid Build Coastguard Worker "cannot write to file");
301*5c90c05cSAndroid Build Coastguard Worker pipe.write_end.close();
302*5c90c05cSAndroid Build Coastguard Worker # ifndef _WIN32
303*5c90c05cSAndroid Build Coastguard Worker EXPECT_EQ(static_cast<std::streamsize>(SIZE), count);
304*5c90c05cSAndroid Build Coastguard Worker char buffer[SIZE + 1];
305*5c90c05cSAndroid Build Coastguard Worker pipe.read_end.read(buffer, SIZE);
306*5c90c05cSAndroid Build Coastguard Worker buffer[SIZE] = '\0';
307*5c90c05cSAndroid Build Coastguard Worker EXPECT_STREQ("test", buffer);
308*5c90c05cSAndroid Build Coastguard Worker # endif
309*5c90c05cSAndroid Build Coastguard Worker }
310*5c90c05cSAndroid Build Coastguard Worker
311*5c90c05cSAndroid Build Coastguard Worker # ifdef _WIN32
TEST(file_test,convert_read_count)312*5c90c05cSAndroid Build Coastguard Worker TEST(file_test, convert_read_count) {
313*5c90c05cSAndroid Build Coastguard Worker auto pipe = fmt::pipe();
314*5c90c05cSAndroid Build Coastguard Worker char c;
315*5c90c05cSAndroid Build Coastguard Worker size_t size = UINT_MAX;
316*5c90c05cSAndroid Build Coastguard Worker if (sizeof(unsigned) != sizeof(size_t)) ++size;
317*5c90c05cSAndroid Build Coastguard Worker read_count = 1;
318*5c90c05cSAndroid Build Coastguard Worker read_nbyte = 0;
319*5c90c05cSAndroid Build Coastguard Worker EXPECT_THROW(pipe.read_end.read(&c, size), std::system_error);
320*5c90c05cSAndroid Build Coastguard Worker read_count = 0;
321*5c90c05cSAndroid Build Coastguard Worker EXPECT_EQ(UINT_MAX, read_nbyte);
322*5c90c05cSAndroid Build Coastguard Worker }
323*5c90c05cSAndroid Build Coastguard Worker
TEST(file_test,convert_write_count)324*5c90c05cSAndroid Build Coastguard Worker TEST(file_test, convert_write_count) {
325*5c90c05cSAndroid Build Coastguard Worker auto pipe = fmt::pipe();
326*5c90c05cSAndroid Build Coastguard Worker char c;
327*5c90c05cSAndroid Build Coastguard Worker size_t size = UINT_MAX;
328*5c90c05cSAndroid Build Coastguard Worker if (sizeof(unsigned) != sizeof(size_t)) ++size;
329*5c90c05cSAndroid Build Coastguard Worker write_count = 1;
330*5c90c05cSAndroid Build Coastguard Worker write_nbyte = 0;
331*5c90c05cSAndroid Build Coastguard Worker EXPECT_THROW(pipe.write_end.write(&c, size), std::system_error);
332*5c90c05cSAndroid Build Coastguard Worker write_count = 0;
333*5c90c05cSAndroid Build Coastguard Worker EXPECT_EQ(UINT_MAX, write_nbyte);
334*5c90c05cSAndroid Build Coastguard Worker }
335*5c90c05cSAndroid Build Coastguard Worker # endif
336*5c90c05cSAndroid Build Coastguard Worker
TEST(file_test,dup_no_retry)337*5c90c05cSAndroid Build Coastguard Worker TEST(file_test, dup_no_retry) {
338*5c90c05cSAndroid Build Coastguard Worker int stdout_fd = FMT_POSIX(fileno(stdout));
339*5c90c05cSAndroid Build Coastguard Worker dup_count = 1;
340*5c90c05cSAndroid Build Coastguard Worker EXPECT_SYSTEM_ERROR(
341*5c90c05cSAndroid Build Coastguard Worker file::dup(stdout_fd), EINTR,
342*5c90c05cSAndroid Build Coastguard Worker fmt::format("cannot duplicate file descriptor {}", stdout_fd));
343*5c90c05cSAndroid Build Coastguard Worker dup_count = 0;
344*5c90c05cSAndroid Build Coastguard Worker }
345*5c90c05cSAndroid Build Coastguard Worker
TEST(file_test,dup2_retry)346*5c90c05cSAndroid Build Coastguard Worker TEST(file_test, dup2_retry) {
347*5c90c05cSAndroid Build Coastguard Worker int stdout_fd = FMT_POSIX(fileno(stdout));
348*5c90c05cSAndroid Build Coastguard Worker file f1 = file::dup(stdout_fd), f2 = file::dup(stdout_fd);
349*5c90c05cSAndroid Build Coastguard Worker EXPECT_RETRY(f1.dup2(f2.descriptor()), dup2,
350*5c90c05cSAndroid Build Coastguard Worker fmt::format("cannot duplicate file descriptor {} to {}",
351*5c90c05cSAndroid Build Coastguard Worker f1.descriptor(), f2.descriptor()));
352*5c90c05cSAndroid Build Coastguard Worker }
353*5c90c05cSAndroid Build Coastguard Worker
TEST(file_test,dup2_no_except_retry)354*5c90c05cSAndroid Build Coastguard Worker TEST(file_test, dup2_no_except_retry) {
355*5c90c05cSAndroid Build Coastguard Worker int stdout_fd = FMT_POSIX(fileno(stdout));
356*5c90c05cSAndroid Build Coastguard Worker file f1 = file::dup(stdout_fd), f2 = file::dup(stdout_fd);
357*5c90c05cSAndroid Build Coastguard Worker std::error_code ec;
358*5c90c05cSAndroid Build Coastguard Worker dup2_count = 1;
359*5c90c05cSAndroid Build Coastguard Worker f1.dup2(f2.descriptor(), ec);
360*5c90c05cSAndroid Build Coastguard Worker # ifndef _WIN32
361*5c90c05cSAndroid Build Coastguard Worker EXPECT_EQ(4, dup2_count);
362*5c90c05cSAndroid Build Coastguard Worker # else
363*5c90c05cSAndroid Build Coastguard Worker EXPECT_EQ(EINTR, ec.value());
364*5c90c05cSAndroid Build Coastguard Worker # endif
365*5c90c05cSAndroid Build Coastguard Worker dup2_count = 0;
366*5c90c05cSAndroid Build Coastguard Worker }
367*5c90c05cSAndroid Build Coastguard Worker
TEST(file_test,pipe_no_retry)368*5c90c05cSAndroid Build Coastguard Worker TEST(file_test, pipe_no_retry) {
369*5c90c05cSAndroid Build Coastguard Worker pipe_count = 1;
370*5c90c05cSAndroid Build Coastguard Worker EXPECT_SYSTEM_ERROR(fmt::pipe(), EINTR, "cannot create pipe");
371*5c90c05cSAndroid Build Coastguard Worker pipe_count = 0;
372*5c90c05cSAndroid Build Coastguard Worker }
373*5c90c05cSAndroid Build Coastguard Worker
TEST(file_test,fdopen_no_retry)374*5c90c05cSAndroid Build Coastguard Worker TEST(file_test, fdopen_no_retry) {
375*5c90c05cSAndroid Build Coastguard Worker auto pipe = fmt::pipe();
376*5c90c05cSAndroid Build Coastguard Worker fdopen_count = 1;
377*5c90c05cSAndroid Build Coastguard Worker EXPECT_SYSTEM_ERROR(pipe.read_end.fdopen("r"), EINTR,
378*5c90c05cSAndroid Build Coastguard Worker "cannot associate stream with file descriptor");
379*5c90c05cSAndroid Build Coastguard Worker fdopen_count = 0;
380*5c90c05cSAndroid Build Coastguard Worker }
381*5c90c05cSAndroid Build Coastguard Worker
TEST(buffered_file_test,open_retry)382*5c90c05cSAndroid Build Coastguard Worker TEST(buffered_file_test, open_retry) {
383*5c90c05cSAndroid Build Coastguard Worker write_file("temp", "there must be something here");
384*5c90c05cSAndroid Build Coastguard Worker std::unique_ptr<buffered_file> f{nullptr};
385*5c90c05cSAndroid Build Coastguard Worker EXPECT_RETRY(f.reset(new buffered_file("temp", "r")), fopen,
386*5c90c05cSAndroid Build Coastguard Worker "cannot open file temp");
387*5c90c05cSAndroid Build Coastguard Worker # ifndef _WIN32
388*5c90c05cSAndroid Build Coastguard Worker char c = 0;
389*5c90c05cSAndroid Build Coastguard Worker if (fread(&c, 1, 1, f->get()) < 1)
390*5c90c05cSAndroid Build Coastguard Worker throw fmt::system_error(errno, "fread failed");
391*5c90c05cSAndroid Build Coastguard Worker # endif
392*5c90c05cSAndroid Build Coastguard Worker }
393*5c90c05cSAndroid Build Coastguard Worker
TEST(buffered_file_test,close_no_retry_in_dtor)394*5c90c05cSAndroid Build Coastguard Worker TEST(buffered_file_test, close_no_retry_in_dtor) {
395*5c90c05cSAndroid Build Coastguard Worker auto pipe = fmt::pipe();
396*5c90c05cSAndroid Build Coastguard Worker std::unique_ptr<buffered_file> f(
397*5c90c05cSAndroid Build Coastguard Worker new buffered_file(pipe.read_end.fdopen("r")));
398*5c90c05cSAndroid Build Coastguard Worker int saved_fclose_count = 0;
399*5c90c05cSAndroid Build Coastguard Worker EXPECT_WRITE(
400*5c90c05cSAndroid Build Coastguard Worker stderr,
401*5c90c05cSAndroid Build Coastguard Worker {
402*5c90c05cSAndroid Build Coastguard Worker fclose_count = 1;
403*5c90c05cSAndroid Build Coastguard Worker f.reset(nullptr);
404*5c90c05cSAndroid Build Coastguard Worker saved_fclose_count = fclose_count;
405*5c90c05cSAndroid Build Coastguard Worker fclose_count = 0;
406*5c90c05cSAndroid Build Coastguard Worker },
407*5c90c05cSAndroid Build Coastguard Worker system_error_message(EINTR, "cannot close file") + "\n");
408*5c90c05cSAndroid Build Coastguard Worker EXPECT_EQ(2, saved_fclose_count);
409*5c90c05cSAndroid Build Coastguard Worker }
410*5c90c05cSAndroid Build Coastguard Worker
TEST(buffered_file_test,close_no_retry)411*5c90c05cSAndroid Build Coastguard Worker TEST(buffered_file_test, close_no_retry) {
412*5c90c05cSAndroid Build Coastguard Worker auto pipe = fmt::pipe();
413*5c90c05cSAndroid Build Coastguard Worker buffered_file f = pipe.read_end.fdopen("r");
414*5c90c05cSAndroid Build Coastguard Worker fclose_count = 1;
415*5c90c05cSAndroid Build Coastguard Worker EXPECT_SYSTEM_ERROR(f.close(), EINTR, "cannot close file");
416*5c90c05cSAndroid Build Coastguard Worker EXPECT_EQ(2, fclose_count);
417*5c90c05cSAndroid Build Coastguard Worker fclose_count = 0;
418*5c90c05cSAndroid Build Coastguard Worker }
419*5c90c05cSAndroid Build Coastguard Worker
TEST(buffered_file_test,fileno_no_retry)420*5c90c05cSAndroid Build Coastguard Worker TEST(buffered_file_test, fileno_no_retry) {
421*5c90c05cSAndroid Build Coastguard Worker auto pipe = fmt::pipe();
422*5c90c05cSAndroid Build Coastguard Worker buffered_file f = pipe.read_end.fdopen("r");
423*5c90c05cSAndroid Build Coastguard Worker fileno_count = 1;
424*5c90c05cSAndroid Build Coastguard Worker EXPECT_SYSTEM_ERROR((f.descriptor)(), EINTR, "cannot get file descriptor");
425*5c90c05cSAndroid Build Coastguard Worker EXPECT_EQ(2, fileno_count);
426*5c90c05cSAndroid Build Coastguard Worker fileno_count = 0;
427*5c90c05cSAndroid Build Coastguard Worker }
428*5c90c05cSAndroid Build Coastguard Worker #endif // FMT_USE_FCNTL
429*5c90c05cSAndroid Build Coastguard Worker
430*5c90c05cSAndroid Build Coastguard Worker struct test_mock {
431*5c90c05cSAndroid Build Coastguard Worker static test_mock* instance;
432*5c90c05cSAndroid Build Coastguard Worker }* test_mock::instance;
433*5c90c05cSAndroid Build Coastguard Worker
TEST(scoped_mock,scope)434*5c90c05cSAndroid Build Coastguard Worker TEST(scoped_mock, scope) {
435*5c90c05cSAndroid Build Coastguard Worker {
436*5c90c05cSAndroid Build Coastguard Worker scoped_mock<test_mock> mock;
437*5c90c05cSAndroid Build Coastguard Worker EXPECT_EQ(&mock, test_mock::instance);
438*5c90c05cSAndroid Build Coastguard Worker test_mock& copy = mock;
439*5c90c05cSAndroid Build Coastguard Worker static_cast<void>(copy);
440*5c90c05cSAndroid Build Coastguard Worker }
441*5c90c05cSAndroid Build Coastguard Worker EXPECT_EQ(nullptr, test_mock::instance);
442*5c90c05cSAndroid Build Coastguard Worker }
443