| 
									
										
										
										
											2014-05-15 07:45:44 -07:00
										 |  |  | /*
 | 
					
						
							| 
									
										
										
										
											2015-06-24 08:43:30 -07:00
										 |  |  |  Tests of the C++ interface to POSIX functions that require mocks | 
					
						
							| 
									
										
										
										
											2014-05-15 07:45:44 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-06-24 08:43:30 -07:00
										 |  |  |  Copyright (c) 2012-2015, Victor Zverovich | 
					
						
							| 
									
										
										
										
											2014-05-15 07:45:44 -07:00
										 |  |  |  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. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-12-17 06:53:32 -08:00
										 |  |  | // Disable bogus MSVC warnings.
 | 
					
						
							|  |  |  | #define _CRT_SECURE_NO_WARNINGS
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-06-24 08:43:30 -07:00
										 |  |  | #include "posix-mock.h"
 | 
					
						
							| 
									
										
										
										
											2016-04-24 09:06:12 -07:00
										 |  |  | #include "fmt/posix.cc"
 | 
					
						
							| 
									
										
										
										
											2014-05-15 07:45:44 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  | #include <errno.h>
 | 
					
						
							|  |  |  | #include <fcntl.h>
 | 
					
						
							| 
									
										
										
										
											2014-05-19 08:25:08 -07:00
										 |  |  | #include <climits>
 | 
					
						
							| 
									
										
										
										
											2014-05-15 07:45:44 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-05-18 11:09:37 -07:00
										 |  |  | #ifdef _WIN32
 | 
					
						
							|  |  |  | # include <io.h>
 | 
					
						
							| 
									
										
										
										
											2014-09-12 15:18:11 -07:00
										 |  |  | # undef max
 | 
					
						
							| 
									
										
										
										
											2014-09-12 17:16:22 -07:00
										 |  |  | # undef ERROR
 | 
					
						
							| 
									
										
										
										
											2014-05-18 11:09:37 -07:00
										 |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-02-02 17:21:09 -08:00
										 |  |  | #include "gmock/gmock.h"
 | 
					
						
							| 
									
										
										
										
											2014-05-15 07:45:44 -07:00
										 |  |  | #include "gtest-extra.h"
 | 
					
						
							| 
									
										
										
										
											2015-06-24 09:16:03 -07:00
										 |  |  | #include "util.h"
 | 
					
						
							| 
									
										
										
										
											2014-05-15 07:45:44 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-06-09 09:07:27 -07:00
										 |  |  | using fmt::BufferedFile; | 
					
						
							|  |  |  | using fmt::ErrorCode; | 
					
						
							|  |  |  | using fmt::File; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-07-31 08:24:37 -07:00
										 |  |  | using testing::internal::scoped_ptr; | 
					
						
							| 
									
										
										
										
											2016-02-02 17:21:09 -08:00
										 |  |  | using testing::_; | 
					
						
							|  |  |  | using testing::StrEq; | 
					
						
							|  |  |  | using testing::Return; | 
					
						
							| 
									
										
										
										
											2015-07-31 08:24:37 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-05-15 07:45:44 -07:00
										 |  |  | namespace { | 
					
						
							|  |  |  | int open_count; | 
					
						
							| 
									
										
										
										
											2014-05-15 08:58:10 -07:00
										 |  |  | int close_count; | 
					
						
							|  |  |  | int dup_count; | 
					
						
							|  |  |  | int dup2_count; | 
					
						
							|  |  |  | int fdopen_count; | 
					
						
							|  |  |  | int read_count; | 
					
						
							|  |  |  | int write_count; | 
					
						
							| 
									
										
										
										
											2014-05-18 10:05:29 -07:00
										 |  |  | int pipe_count; | 
					
						
							| 
									
										
										
										
											2014-08-28 11:53:05 -07:00
										 |  |  | int fopen_count; | 
					
						
							| 
									
										
										
										
											2014-05-18 11:09:37 -07:00
										 |  |  | int fclose_count; | 
					
						
							| 
									
										
										
										
											2014-05-19 07:31:05 -07:00
										 |  |  | int fileno_count; | 
					
						
							| 
									
										
										
										
											2014-05-19 08:25:08 -07:00
										 |  |  | std::size_t read_nbyte; | 
					
						
							| 
									
										
										
										
											2014-05-19 08:37:28 -07:00
										 |  |  | std::size_t write_nbyte; | 
					
						
							| 
									
										
										
										
											2014-09-12 13:53:52 -07:00
										 |  |  | bool sysconf_error; | 
					
						
							| 
									
										
										
										
											2014-09-12 16:52:15 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  | enum FStatSimulation { NONE, MAX_SIZE, ERROR } fstat_sim; | 
					
						
							| 
									
										
										
										
											2014-05-15 07:45:44 -07:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-05-15 08:58:10 -07:00
										 |  |  | #define EMULATE_EINTR(func, error_result) \
 | 
					
						
							|  |  |  |   if (func##_count != 0) { \ | 
					
						
							|  |  |  |     if (func##_count++ != 3) { \ | 
					
						
							|  |  |  |       errno = EINTR; \ | 
					
						
							|  |  |  |       return error_result; \ | 
					
						
							|  |  |  |     } \ | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-05-06 07:29:58 -07:00
										 |  |  | #ifndef _MSC_VER
 | 
					
						
							| 
									
										
										
										
											2014-05-15 07:45:44 -07:00
										 |  |  | int test::open(const char *path, int oflag, int mode) { | 
					
						
							| 
									
										
										
										
											2014-05-15 08:58:10 -07:00
										 |  |  |   EMULATE_EINTR(open, -1); | 
					
						
							|  |  |  |   return ::open(path, oflag, mode); | 
					
						
							| 
									
										
										
										
											2014-05-15 07:45:44 -07:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2015-05-06 07:29:58 -07:00
										 |  |  | #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); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2014-09-12 13:53:52 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-05-06 07:29:58 -07:00
										 |  |  | #ifndef _WIN32
 | 
					
						
							| 
									
										
										
										
											2015-05-11 19:10:31 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-09-12 13:53:52 -07:00
										 |  |  | long test::sysconf(int name) { | 
					
						
							|  |  |  |   long result = ::sysconf(name); | 
					
						
							|  |  |  |   if (!sysconf_error) | 
					
						
							|  |  |  |     return result; | 
					
						
							|  |  |  |   // Simulate an error.
 | 
					
						
							|  |  |  |   errno = EINVAL; | 
					
						
							|  |  |  |   return -1; | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2015-05-11 19:10:31 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  | 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 (fstat_sim == MAX_SIZE) | 
					
						
							|  |  |  |     buf->st_size = max_file_size(); | 
					
						
							|  |  |  |   return result; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-05-15 07:45:44 -07:00
										 |  |  | #else
 | 
					
						
							| 
									
										
										
										
											2015-05-11 19:10:31 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  | static LONGLONG max_file_size() { return std::numeric_limits<LONGLONG>::max(); } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-03-16 08:52:23 -07:00
										 |  |  | DWORD test::GetFileSize(HANDLE hFile, LPDWORD lpFileSizeHigh) { | 
					
						
							| 
									
										
										
										
											2014-09-12 16:52:15 -07:00
										 |  |  |   if (fstat_sim == ERROR) { | 
					
						
							|  |  |  |     SetLastError(ERROR_ACCESS_DENIED); | 
					
						
							| 
									
										
										
										
											2015-03-16 09:17:18 -07:00
										 |  |  |     return INVALID_FILE_SIZE; | 
					
						
							| 
									
										
										
										
											2014-09-12 16:52:15 -07:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2015-03-16 08:52:23 -07:00
										 |  |  |   if (fstat_sim == MAX_SIZE) { | 
					
						
							|  |  |  |     DWORD max = std::numeric_limits<DWORD>::max(); | 
					
						
							| 
									
										
										
										
											2015-03-16 09:05:46 -07:00
										 |  |  |     *lpFileSizeHigh = max >> 1; | 
					
						
							| 
									
										
										
										
											2015-03-16 08:52:23 -07:00
										 |  |  |     return max; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   return ::GetFileSize(hFile, lpFileSizeHigh); | 
					
						
							| 
									
										
										
										
											2014-09-12 11:12:22 -07:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2015-05-11 19:10:31 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-05-15 07:45:44 -07:00
										 |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-05-15 08:58:10 -07:00
										 |  |  | int test::close(int fildes) { | 
					
						
							|  |  |  |   // Close the file first because close shouldn't be retried.
 | 
					
						
							| 
									
										
										
										
											2014-05-19 07:31:05 -07:00
										 |  |  |   int result = ::FMT_POSIX(close(fildes)); | 
					
						
							| 
									
										
										
										
											2014-05-15 08:58:10 -07:00
										 |  |  |   EMULATE_EINTR(close, -1); | 
					
						
							|  |  |  |   return result; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int test::dup(int fildes) { | 
					
						
							|  |  |  |   EMULATE_EINTR(dup, -1); | 
					
						
							| 
									
										
										
										
											2014-05-19 07:31:05 -07:00
										 |  |  |   return ::FMT_POSIX(dup(fildes)); | 
					
						
							| 
									
										
										
										
											2014-05-15 08:58:10 -07:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int test::dup2(int fildes, int fildes2) { | 
					
						
							|  |  |  |   EMULATE_EINTR(dup2, -1); | 
					
						
							| 
									
										
										
										
											2014-05-19 07:31:05 -07:00
										 |  |  |   return ::FMT_POSIX(dup2(fildes, fildes2)); | 
					
						
							| 
									
										
										
										
											2014-05-15 08:58:10 -07:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | FILE *test::fdopen(int fildes, const char *mode) { | 
					
						
							|  |  |  |   EMULATE_EINTR(fdopen, 0); | 
					
						
							| 
									
										
										
										
											2014-05-19 07:31:05 -07:00
										 |  |  |   return ::FMT_POSIX(fdopen(fildes, mode)); | 
					
						
							| 
									
										
										
										
											2014-05-15 08:58:10 -07:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | test::ssize_t test::read(int fildes, void *buf, test::size_t nbyte) { | 
					
						
							| 
									
										
										
										
											2014-05-19 08:25:08 -07:00
										 |  |  |   read_nbyte = nbyte; | 
					
						
							| 
									
										
										
										
											2014-05-15 08:58:10 -07:00
										 |  |  |   EMULATE_EINTR(read, -1); | 
					
						
							| 
									
										
										
										
											2014-05-19 07:31:05 -07:00
										 |  |  |   return ::FMT_POSIX(read(fildes, buf, nbyte)); | 
					
						
							| 
									
										
										
										
											2014-05-15 08:58:10 -07:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | test::ssize_t test::write(int fildes, const void *buf, test::size_t nbyte) { | 
					
						
							| 
									
										
										
										
											2014-05-19 08:37:28 -07:00
										 |  |  |   write_nbyte = nbyte; | 
					
						
							| 
									
										
										
										
											2014-05-15 08:58:10 -07:00
										 |  |  |   EMULATE_EINTR(write, -1); | 
					
						
							| 
									
										
										
										
											2014-05-19 07:31:05 -07:00
										 |  |  |   return ::FMT_POSIX(write(fildes, buf, nbyte)); | 
					
						
							| 
									
										
										
										
											2014-05-15 08:58:10 -07:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-05-18 11:09:37 -07:00
										 |  |  | #ifndef _WIN32
 | 
					
						
							| 
									
										
										
										
											2014-05-18 10:05:29 -07:00
										 |  |  | int test::pipe(int fildes[2]) { | 
					
						
							|  |  |  |   EMULATE_EINTR(pipe, -1); | 
					
						
							|  |  |  |   return ::pipe(fildes); | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2014-05-18 11:09:37 -07:00
										 |  |  | #else
 | 
					
						
							|  |  |  | int test::pipe(int *pfds, unsigned psize, int textmode) { | 
					
						
							|  |  |  |   EMULATE_EINTR(pipe, -1); | 
					
						
							| 
									
										
										
										
											2014-05-18 12:38:01 -07:00
										 |  |  |   return _pipe(pfds, psize, textmode); | 
					
						
							| 
									
										
										
										
											2014-05-18 11:09:37 -07:00
										 |  |  | } | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-08-28 11:53:05 -07:00
										 |  |  | FILE *test::fopen(const char *filename, const char *mode) { | 
					
						
							|  |  |  |   EMULATE_EINTR(fopen, 0); | 
					
						
							|  |  |  |   return ::fopen(filename, mode); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-05-18 11:09:37 -07:00
										 |  |  | int test::fclose(FILE *stream) { | 
					
						
							|  |  |  |   EMULATE_EINTR(fclose, EOF); | 
					
						
							|  |  |  |   return ::fclose(stream); | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2014-05-18 10:05:29 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-03-19 07:39:24 -07:00
										 |  |  | int (test::fileno)(FILE *stream) { | 
					
						
							| 
									
										
										
										
											2014-05-19 07:31:05 -07:00
										 |  |  |   EMULATE_EINTR(fileno, -1); | 
					
						
							| 
									
										
										
										
											2015-05-06 07:19:42 -07:00
										 |  |  | #ifdef fileno
 | 
					
						
							|  |  |  |   return FMT_POSIX(fileno(stream)); | 
					
						
							|  |  |  | #else
 | 
					
						
							| 
									
										
										
										
											2014-05-19 07:53:16 -07:00
										 |  |  |   return ::FMT_POSIX(fileno(stream)); | 
					
						
							| 
									
										
										
										
											2015-05-06 07:19:42 -07:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2014-05-19 07:31:05 -07:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-05-15 07:45:44 -07:00
										 |  |  | #ifndef _WIN32
 | 
					
						
							|  |  |  | # define EXPECT_RETRY(statement, func, message) \
 | 
					
						
							| 
									
										
										
										
											2014-05-15 08:58:10 -07:00
										 |  |  |     func##_count = 1; \ | 
					
						
							| 
									
										
										
										
											2014-05-15 07:45:44 -07:00
										 |  |  |     statement; \ | 
					
						
							| 
									
										
										
										
											2014-05-15 08:58:10 -07:00
										 |  |  |     EXPECT_EQ(4, func##_count); \ | 
					
						
							|  |  |  |     func##_count = 0; | 
					
						
							| 
									
										
										
										
											2014-05-16 09:16:29 -07:00
										 |  |  | # define EXPECT_EQ_POSIX(expected, actual) EXPECT_EQ(expected, actual)
 | 
					
						
							| 
									
										
										
										
											2014-05-15 07:45:44 -07:00
										 |  |  | #else
 | 
					
						
							|  |  |  | # define EXPECT_RETRY(statement, func, message) \
 | 
					
						
							| 
									
										
										
										
											2014-05-15 08:58:10 -07:00
										 |  |  |     func##_count = 1; \ | 
					
						
							|  |  |  |     EXPECT_SYSTEM_ERROR(statement, EINTR, message); \ | 
					
						
							|  |  |  |     func##_count = 0; | 
					
						
							| 
									
										
										
										
											2014-05-16 09:16:29 -07:00
										 |  |  | # define EXPECT_EQ_POSIX(expected, actual)
 | 
					
						
							| 
									
										
										
										
											2014-05-15 07:45:44 -07:00
										 |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-06-26 07:43:54 -07:00
										 |  |  | void write_file(fmt::CStringRef filename, fmt::StringRef content) { | 
					
						
							| 
									
										
										
										
											2014-08-28 11:53:05 -07:00
										 |  |  |   fmt::BufferedFile f(filename, "w"); | 
					
						
							| 
									
										
										
										
											2014-09-12 12:05:32 -07:00
										 |  |  |   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.
 | 
					
						
							| 
									
										
										
										
											2014-08-28 11:53:05 -07:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-09-12 13:53:52 -07:00
										 |  |  | TEST(UtilTest, GetPageSize) { | 
					
						
							|  |  |  | #ifdef _WIN32
 | 
					
						
							|  |  |  |   SYSTEM_INFO si = {}; | 
					
						
							|  |  |  |   GetSystemInfo(&si); | 
					
						
							|  |  |  |   EXPECT_EQ(si.dwPageSize, fmt::getpagesize()); | 
					
						
							|  |  |  | #else
 | 
					
						
							|  |  |  |   EXPECT_EQ(sysconf(_SC_PAGESIZE), fmt::getpagesize()); | 
					
						
							|  |  |  |   sysconf_error = true; | 
					
						
							|  |  |  |   EXPECT_SYSTEM_ERROR( | 
					
						
							|  |  |  |       fmt::getpagesize(), EINVAL, "cannot get memory page size"); | 
					
						
							|  |  |  |   sysconf_error = false; | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-05-15 07:45:44 -07:00
										 |  |  | TEST(FileTest, OpenRetry) { | 
					
						
							| 
									
										
										
										
											2014-08-28 11:53:05 -07:00
										 |  |  |   write_file("test", "there must be something here"); | 
					
						
							| 
									
										
										
										
											2015-07-31 08:24:37 -07:00
										 |  |  |   scoped_ptr<File> f; | 
					
						
							|  |  |  |   EXPECT_RETRY(f.reset(new File("test", File::RDONLY)), | 
					
						
							| 
									
										
										
										
											2014-08-28 11:53:05 -07:00
										 |  |  |                open, "cannot open file test"); | 
					
						
							| 
									
										
										
										
											2014-05-16 07:07:43 -07:00
										 |  |  | #ifndef _WIN32
 | 
					
						
							|  |  |  |   char c = 0; | 
					
						
							|  |  |  |   f->read(&c, 1); | 
					
						
							|  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2014-05-15 07:45:44 -07:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-05-16 06:25:20 -07:00
										 |  |  | TEST(FileTest, CloseNoRetryInDtor) { | 
					
						
							|  |  |  |   File read_end, write_end; | 
					
						
							|  |  |  |   File::pipe(read_end, write_end); | 
					
						
							| 
									
										
										
										
											2015-07-31 08:24:37 -07:00
										 |  |  |   scoped_ptr<File> f(new File(std::move(read_end))); | 
					
						
							| 
									
										
										
										
											2014-05-16 06:25:20 -07:00
										 |  |  |   int saved_close_count = 0; | 
					
						
							|  |  |  |   EXPECT_WRITE(stderr, { | 
					
						
							|  |  |  |     close_count = 1; | 
					
						
							| 
									
										
										
										
											2015-07-28 07:18:14 -07:00
										 |  |  |     f.reset(); | 
					
						
							| 
									
										
										
										
											2014-05-16 06:25:20 -07:00
										 |  |  |     saved_close_count = close_count; | 
					
						
							|  |  |  |     close_count = 0; | 
					
						
							| 
									
										
										
										
											2014-07-27 15:09:05 -07:00
										 |  |  |   }, format_system_error(EINTR, "cannot close file") + "\n"); | 
					
						
							| 
									
										
										
										
											2014-05-16 06:25:20 -07:00
										 |  |  |   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; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-09-12 10:50:16 -07:00
										 |  |  | TEST(FileTest, Size) { | 
					
						
							| 
									
										
										
										
											2014-09-12 12:05:32 -07:00
										 |  |  |   std::string content = "top secret, destroy before reading"; | 
					
						
							|  |  |  |   write_file("test", content); | 
					
						
							| 
									
										
										
										
											2014-09-12 10:50:16 -07:00
										 |  |  |   File f("test", File::RDONLY); | 
					
						
							| 
									
										
										
										
											2014-10-13 08:39:38 -07:00
										 |  |  |   EXPECT_GE(f.size(), 0); | 
					
						
							| 
									
										
										
										
											2016-03-02 07:53:14 -08:00
										 |  |  |   EXPECT_EQ(content.size(), static_cast<fmt::ULongLong>(f.size())); | 
					
						
							| 
									
										
										
										
											2014-09-12 16:52:15 -07:00
										 |  |  | #ifdef _WIN32
 | 
					
						
							| 
									
										
										
										
											2014-09-29 09:28:44 -07:00
										 |  |  |   fmt::MemoryWriter message; | 
					
						
							| 
									
										
										
										
											2014-09-12 17:41:04 -07:00
										 |  |  |   fmt::internal::format_windows_error( | 
					
						
							| 
									
										
										
										
											2014-09-12 17:16:22 -07:00
										 |  |  |       message, ERROR_ACCESS_DENIED, "cannot get file size"); | 
					
						
							| 
									
										
										
										
											2014-09-12 16:52:15 -07:00
										 |  |  |   fstat_sim = ERROR; | 
					
						
							|  |  |  |   EXPECT_THROW_MSG(f.size(), fmt::WindowsError, message.str()); | 
					
						
							| 
									
										
										
										
											2014-09-12 17:16:22 -07:00
										 |  |  |   fstat_sim = NONE; | 
					
						
							| 
									
										
										
										
											2014-09-12 16:52:15 -07:00
										 |  |  | #else
 | 
					
						
							| 
									
										
										
										
											2014-09-12 18:22:45 -07:00
										 |  |  |   f.close(); | 
					
						
							| 
									
										
										
										
											2014-09-12 11:12:22 -07:00
										 |  |  |   EXPECT_SYSTEM_ERROR(f.size(), EBADF, "cannot get file attributes"); | 
					
						
							| 
									
										
										
										
											2014-09-12 16:52:15 -07:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2014-09-12 10:50:16 -07:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-09-12 11:12:22 -07:00
										 |  |  | TEST(FileTest, MaxSize) { | 
					
						
							| 
									
										
										
										
											2014-09-12 12:05:32 -07:00
										 |  |  |   write_file("test", ""); | 
					
						
							| 
									
										
										
										
											2014-09-12 11:12:22 -07:00
										 |  |  |   File f("test", File::RDONLY); | 
					
						
							| 
									
										
										
										
											2014-09-12 16:52:15 -07:00
										 |  |  |   fstat_sim = MAX_SIZE; | 
					
						
							| 
									
										
										
										
											2014-09-12 11:12:22 -07:00
										 |  |  |   EXPECT_GE(f.size(), 0); | 
					
						
							|  |  |  |   EXPECT_EQ(max_file_size(), f.size()); | 
					
						
							| 
									
										
										
										
											2014-09-12 16:52:15 -07:00
										 |  |  |   fstat_sim = NONE; | 
					
						
							| 
									
										
										
										
											2014-09-12 11:12:22 -07:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-05-16 07:07:43 -07:00
										 |  |  | 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]; | 
					
						
							| 
									
										
										
										
											2016-03-02 07:53:14 -08:00
										 |  |  |   std::size_t count = 0; | 
					
						
							| 
									
										
										
										
											2014-05-16 07:07:43 -07:00
										 |  |  |   EXPECT_RETRY(count = read_end.read(buffer, SIZE), | 
					
						
							|  |  |  |       read, "cannot read from file"); | 
					
						
							| 
									
										
										
										
											2014-06-07 07:11:34 -07:00
										 |  |  |   EXPECT_EQ_POSIX(static_cast<std::streamsize>(SIZE), count); | 
					
						
							| 
									
										
										
										
											2014-05-16 07:07:43 -07:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | TEST(FileTest, WriteRetry) { | 
					
						
							|  |  |  |   File read_end, write_end; | 
					
						
							|  |  |  |   File::pipe(read_end, write_end); | 
					
						
							|  |  |  |   enum { SIZE = 4 }; | 
					
						
							| 
									
										
										
										
											2016-03-02 07:53:14 -08:00
										 |  |  |   std::size_t count = 0; | 
					
						
							| 
									
										
										
										
											2014-05-16 07:07:43 -07:00
										 |  |  |   EXPECT_RETRY(count = write_end.write("test", SIZE), | 
					
						
							|  |  |  |       write, "cannot write to file"); | 
					
						
							|  |  |  |   write_end.close(); | 
					
						
							|  |  |  | #ifndef _WIN32
 | 
					
						
							| 
									
										
										
										
											2014-06-07 07:11:34 -07:00
										 |  |  |   EXPECT_EQ(static_cast<std::streamsize>(SIZE), count); | 
					
						
							| 
									
										
										
										
											2014-05-16 07:07:43 -07:00
										 |  |  |   char buffer[SIZE + 1]; | 
					
						
							|  |  |  |   read_end.read(buffer, SIZE); | 
					
						
							|  |  |  |   buffer[SIZE] = '\0'; | 
					
						
							|  |  |  |   EXPECT_STREQ("test", buffer); | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-05-19 08:25:08 -07:00
										 |  |  | #ifdef _WIN32
 | 
					
						
							|  |  |  | TEST(FileTest, ConvertReadCount) { | 
					
						
							|  |  |  |   File read_end, write_end; | 
					
						
							|  |  |  |   File::pipe(read_end, write_end); | 
					
						
							|  |  |  |   char c; | 
					
						
							| 
									
										
										
										
											2014-05-19 08:37:28 -07:00
										 |  |  |   std::size_t size = UINT_MAX; | 
					
						
							|  |  |  |   if (sizeof(unsigned) != sizeof(std::size_t)) | 
					
						
							|  |  |  |     ++size; | 
					
						
							| 
									
										
										
										
											2014-05-19 08:25:08 -07:00
										 |  |  |   read_count = 1; | 
					
						
							| 
									
										
										
										
											2014-05-19 08:37:28 -07:00
										 |  |  |   read_nbyte = 0; | 
					
						
							|  |  |  |   EXPECT_THROW(read_end.read(&c, size), fmt::SystemError); | 
					
						
							| 
									
										
										
										
											2014-05-19 08:25:08 -07:00
										 |  |  |   read_count = 0; | 
					
						
							|  |  |  |   EXPECT_EQ(UINT_MAX, read_nbyte); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-05-19 08:37:28 -07:00
										 |  |  | 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
 | 
					
						
							| 
									
										
										
										
											2014-05-19 08:25:08 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-05-18 10:05:29 -07:00
										 |  |  | TEST(FileTest, DupNoRetry) { | 
					
						
							| 
									
										
										
										
											2014-05-19 07:53:16 -07:00
										 |  |  |   int stdout_fd = FMT_POSIX(fileno(stdout)); | 
					
						
							| 
									
										
										
										
											2014-05-18 10:05:29 -07:00
										 |  |  |   dup_count = 1; | 
					
						
							|  |  |  |   EXPECT_SYSTEM_ERROR(File::dup(stdout_fd), EINTR, | 
					
						
							| 
									
										
										
										
											2014-06-29 11:51:10 -07:00
										 |  |  |       fmt::format("cannot duplicate file descriptor {}", stdout_fd)); | 
					
						
							| 
									
										
										
										
											2014-05-18 10:05:29 -07:00
										 |  |  |   dup_count = 0; | 
					
						
							| 
									
										
										
										
											2014-05-16 07:07:43 -07:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | TEST(FileTest, Dup2Retry) { | 
					
						
							| 
									
										
										
										
											2014-05-19 07:53:16 -07:00
										 |  |  |   int stdout_fd = FMT_POSIX(fileno(stdout)); | 
					
						
							|  |  |  |   File f1 = File::dup(stdout_fd), f2 = File::dup(stdout_fd); | 
					
						
							| 
									
										
										
										
											2014-05-16 07:07:43 -07:00
										 |  |  |   EXPECT_RETRY(f1.dup2(f2.descriptor()), dup2, | 
					
						
							| 
									
										
										
										
											2014-06-29 11:51:10 -07:00
										 |  |  |       fmt::format("cannot duplicate file descriptor {} to {}", | 
					
						
							|  |  |  |       f1.descriptor(), f2.descriptor())); | 
					
						
							| 
									
										
										
										
											2014-05-16 07:07:43 -07:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-05-18 10:05:29 -07:00
										 |  |  | TEST(FileTest, Dup2NoExceptRetry) { | 
					
						
							| 
									
										
										
										
											2014-05-19 07:53:16 -07:00
										 |  |  |   int stdout_fd = FMT_POSIX(fileno(stdout)); | 
					
						
							|  |  |  |   File f1 = File::dup(stdout_fd), f2 = File::dup(stdout_fd); | 
					
						
							| 
									
										
										
										
											2014-05-18 10:05:29 -07:00
										 |  |  |   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; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-08-28 11:53:05 -07:00
										 |  |  | TEST(BufferedFileTest, OpenRetry) { | 
					
						
							|  |  |  |   write_file("test", "there must be something here"); | 
					
						
							| 
									
										
										
										
											2015-07-31 08:24:37 -07:00
										 |  |  |   scoped_ptr<BufferedFile> f; | 
					
						
							|  |  |  |   EXPECT_RETRY(f.reset(new BufferedFile("test", "r")), | 
					
						
							| 
									
										
										
										
											2014-08-28 11:53:05 -07:00
										 |  |  |                fopen, "cannot open file test"); | 
					
						
							|  |  |  | #ifndef _WIN32
 | 
					
						
							|  |  |  |   char c = 0; | 
					
						
							| 
									
										
										
										
											2014-08-29 06:57:26 -07:00
										 |  |  |   if (fread(&c, 1, 1, f->get()) < 1) | 
					
						
							|  |  |  |     throw fmt::SystemError(errno, "fread failed"); | 
					
						
							| 
									
										
										
										
											2014-08-28 11:53:05 -07:00
										 |  |  | #endif
 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-05-19 07:31:05 -07:00
										 |  |  | TEST(BufferedFileTest, CloseNoRetryInDtor) { | 
					
						
							|  |  |  |   File read_end, write_end; | 
					
						
							|  |  |  |   File::pipe(read_end, write_end); | 
					
						
							| 
									
										
										
										
											2015-07-31 08:24:37 -07:00
										 |  |  |   scoped_ptr<BufferedFile> f(new BufferedFile(read_end.fdopen("r"))); | 
					
						
							| 
									
										
										
										
											2014-05-19 07:31:05 -07:00
										 |  |  |   int saved_fclose_count = 0; | 
					
						
							|  |  |  |   EXPECT_WRITE(stderr, { | 
					
						
							|  |  |  |     fclose_count = 1; | 
					
						
							| 
									
										
										
										
											2015-07-31 08:24:37 -07:00
										 |  |  |     f.reset(); | 
					
						
							| 
									
										
										
										
											2014-05-19 07:31:05 -07:00
										 |  |  |     saved_fclose_count = fclose_count; | 
					
						
							|  |  |  |     fclose_count = 0; | 
					
						
							| 
									
										
										
										
											2014-07-27 15:09:05 -07:00
										 |  |  |   }, format_system_error(EINTR, "cannot close file") + "\n"); | 
					
						
							| 
									
										
										
										
											2014-05-19 07:31:05 -07:00
										 |  |  |   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; | 
					
						
							| 
									
										
										
										
											2015-05-06 07:19:42 -07:00
										 |  |  |   EXPECT_SYSTEM_ERROR((f.fileno)(), EINTR, "cannot get file descriptor"); | 
					
						
							| 
									
										
										
										
											2014-05-19 07:31:05 -07:00
										 |  |  |   EXPECT_EQ(2, fileno_count); | 
					
						
							|  |  |  |   fileno_count = 0; | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2016-02-02 17:21:09 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-02-09 11:31:04 -08:00
										 |  |  | struct TestMock { | 
					
						
							|  |  |  |   static TestMock *instance; | 
					
						
							|  |  |  | } *TestMock::instance; | 
					
						
							| 
									
										
										
										
											2016-02-02 17:21:09 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  | TEST(ScopedMock, Scope) { | 
					
						
							|  |  |  |   { | 
					
						
							| 
									
										
										
										
											2016-02-09 11:31:04 -08:00
										 |  |  |     ScopedMock<TestMock> mock; | 
					
						
							|  |  |  |     EXPECT_EQ(&mock, TestMock::instance); | 
					
						
							| 
									
										
										
										
											2016-02-02 17:21:09 -08:00
										 |  |  |     TestMock © = mock; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2016-02-09 11:31:04 -08:00
										 |  |  |   EXPECT_EQ(0, TestMock::instance); | 
					
						
							| 
									
										
										
										
											2016-02-02 17:21:09 -08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #ifdef FMT_LOCALE
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | typedef fmt::Locale::Type LocaleType; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | struct LocaleMock { | 
					
						
							| 
									
										
										
										
											2016-02-09 11:31:04 -08:00
										 |  |  |   static LocaleMock *instance; | 
					
						
							| 
									
										
										
										
											2016-02-02 17:21:09 -08:00
										 |  |  |   MOCK_METHOD3(newlocale, LocaleType (int category_mask, const char *locale, | 
					
						
							|  |  |  |                                       LocaleType base)); | 
					
						
							|  |  |  |   MOCK_METHOD1(freelocale, void (LocaleType locale)); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   MOCK_METHOD3(strtod_l, double (const char *nptr, char **endptr, | 
					
						
							|  |  |  |                                  LocaleType locale)); | 
					
						
							| 
									
										
										
										
											2016-02-09 11:31:04 -08:00
										 |  |  | } *LocaleMock::instance; | 
					
						
							| 
									
										
										
										
											2016-02-02 17:21:09 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  | #ifdef _MSC_VER
 | 
					
						
							| 
									
										
										
										
											2016-03-19 06:39:33 -07:00
										 |  |  | # pragma warning(push)
 | 
					
						
							|  |  |  | # pragma warning(disable: 4273)
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-02-02 17:21:09 -08:00
										 |  |  | _locale_t _create_locale(int category, const char *locale) { | 
					
						
							| 
									
										
										
										
											2016-02-09 11:31:04 -08:00
										 |  |  |   return LocaleMock::instance->newlocale(category, locale, 0); | 
					
						
							| 
									
										
										
										
											2016-02-02 17:21:09 -08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void _free_locale(_locale_t locale) { | 
					
						
							| 
									
										
										
										
											2016-02-09 11:31:04 -08:00
										 |  |  |   LocaleMock::instance->freelocale(locale); | 
					
						
							| 
									
										
										
										
											2016-02-02 17:21:09 -08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | double _strtod_l(const char *nptr, char **endptr, _locale_t locale) { | 
					
						
							| 
									
										
										
										
											2016-02-09 11:31:04 -08:00
										 |  |  |   return LocaleMock::instance->strtod_l(nptr, endptr, locale); | 
					
						
							| 
									
										
										
										
											2016-02-02 17:21:09 -08:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2016-03-19 06:39:33 -07:00
										 |  |  | # pragma warning(pop)
 | 
					
						
							| 
									
										
										
										
											2016-02-02 17:21:09 -08:00
										 |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | LocaleType newlocale(int category_mask, const char *locale, LocaleType base) { | 
					
						
							| 
									
										
										
										
											2016-02-09 11:31:04 -08:00
										 |  |  |   return LocaleMock::instance->newlocale(category_mask, locale, base); | 
					
						
							| 
									
										
										
										
											2016-02-02 17:21:09 -08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-14 06:28:41 -08:00
										 |  |  | #if defined(__APPLE__) || defined(__FreeBSD__)
 | 
					
						
							| 
									
										
										
										
											2016-02-02 22:06:54 -08:00
										 |  |  | typedef int FreeLocaleResult; | 
					
						
							| 
									
										
										
										
											2016-02-02 17:21:09 -08:00
										 |  |  | #else
 | 
					
						
							| 
									
										
										
										
											2016-02-02 22:06:54 -08:00
										 |  |  | typedef void FreeLocaleResult; | 
					
						
							| 
									
										
										
										
											2016-02-02 17:21:09 -08:00
										 |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | FreeLocaleResult freelocale(LocaleType locale) { | 
					
						
							| 
									
										
										
										
											2016-02-09 11:31:04 -08:00
										 |  |  |   LocaleMock::instance->freelocale(locale); | 
					
						
							| 
									
										
										
										
											2016-02-11 06:55:53 -08:00
										 |  |  |   return FreeLocaleResult(); | 
					
						
							| 
									
										
										
										
											2016-02-02 17:21:09 -08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | double strtod_l(const char *nptr, char **endptr, LocaleType locale) { | 
					
						
							| 
									
										
										
										
											2016-02-09 11:31:04 -08:00
										 |  |  |   return LocaleMock::instance->strtod_l(nptr, endptr, locale); | 
					
						
							| 
									
										
										
										
											2016-02-02 17:21:09 -08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | TEST(LocaleTest, LocaleMock) { | 
					
						
							| 
									
										
										
										
											2016-02-09 11:31:04 -08:00
										 |  |  |   ScopedMock<LocaleMock> mock; | 
					
						
							| 
									
										
										
										
											2016-02-02 17:21:09 -08:00
										 |  |  |   LocaleType locale = reinterpret_cast<LocaleType>(11); | 
					
						
							|  |  |  |   EXPECT_CALL(mock, newlocale(222, StrEq("foo"), locale)); | 
					
						
							|  |  |  |   newlocale(222, "foo", locale); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | TEST(LocaleTest, Locale) { | 
					
						
							|  |  |  | #ifndef LC_NUMERIC_MASK
 | 
					
						
							|  |  |  |   enum { LC_NUMERIC_MASK = LC_NUMERIC }; | 
					
						
							|  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2016-02-09 11:31:04 -08:00
										 |  |  |   ScopedMock<LocaleMock> mock; | 
					
						
							| 
									
										
										
										
											2016-02-02 17:21:09 -08:00
										 |  |  |   LocaleType impl = reinterpret_cast<LocaleType>(42); | 
					
						
							|  |  |  |   EXPECT_CALL(mock, newlocale(LC_NUMERIC_MASK, StrEq("C"), 0)) | 
					
						
							|  |  |  |       .WillOnce(Return(impl)); | 
					
						
							|  |  |  |   EXPECT_CALL(mock, freelocale(impl)); | 
					
						
							|  |  |  |   fmt::Locale locale; | 
					
						
							|  |  |  |   EXPECT_EQ(impl, locale.get()); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | TEST(LocaleTest, Strtod) { | 
					
						
							| 
									
										
										
										
											2016-02-09 11:31:04 -08:00
										 |  |  |   ScopedMock<LocaleMock> mock; | 
					
						
							| 
									
										
										
										
											2016-02-02 17:21:09 -08:00
										 |  |  |   EXPECT_CALL(mock, newlocale(_, _, _)) | 
					
						
							|  |  |  |       .WillOnce(Return(reinterpret_cast<LocaleType>(42))); | 
					
						
							|  |  |  |   EXPECT_CALL(mock, freelocale(_)); | 
					
						
							|  |  |  |   fmt::Locale locale; | 
					
						
							|  |  |  |   const char *str = "4.2"; | 
					
						
							|  |  |  |   char end = 'x'; | 
					
						
							|  |  |  |   EXPECT_CALL(mock, strtod_l(str, _, locale.get())) | 
					
						
							|  |  |  |       .WillOnce(testing::DoAll(testing::SetArgPointee<1>(&end), Return(777))); | 
					
						
							|  |  |  |   EXPECT_EQ(777, locale.strtod(str)); | 
					
						
							|  |  |  |   EXPECT_EQ(&end, str); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #endif  // FMT_LOCALE
 |