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