1 /* 2 * Copyright (c) 1999 3 * Silicon Graphics Computer Systems, Inc. 4 * 5 * Copyright (c) 1999 6 * Boris Fomitchev 7 * 8 * This material is provided "as is", with absolutely no warranty expressed 9 * or implied. Any use is at your own risk. 10 * 11 * Permission to use or copy this software for any purpose is hereby granted 12 * without fee, provided the above notices are retained on all copies. 13 * Permission to modify the code and to distribute modified code is granted, 14 * provided the above notices are retained, and a notice that the code was 15 * modified is included with the above copyright notice. 16 * 17 */ 18 19 #if defined (__SUNPPRO_CC) && !defined (_STLP_NO_NEW_C_HEADERS) 20 # include <time.h> 21 // For sunpro, it chokes if time.h is included through stat.h 22 #endif 23 24 #include <fstream> 25 26 #ifdef __CYGWIN__ 27 # define __int64 long long 28 #endif 29 30 #include <cstdio> 31 #if !defined(__ISCPP__) 32 extern "C" { 33 # include <sys/stat.h> 34 } 35 #endif 36 37 #if defined( __MSL__ ) 38 # include <unix.h> 39 #endif 40 41 #if defined(__ISCPP__) 42 # include <c_locale_is/filestat.h> 43 #endif 44 45 #if defined(__BEOS__) && defined(__INTEL__) 46 # include <fcntl.h> 47 # include <sys/stat.h> // For _fstat 48 #endif 49 50 #if defined (_STLP_MSVC) || defined (__MINGW32__) 51 # include <fcntl.h> 52 # define S_IREAD _S_IREAD 53 # define S_IWRITE _S_IWRITE 54 # define S_IFREG _S_IFREG 55 // map permission masks 56 # ifndef S_IRUSR 57 # define S_IRUSR _S_IREAD 58 # define S_IWUSR _S_IWRITE 59 # endif 60 # ifndef S_IRGRP 61 # define S_IRGRP _S_IREAD 62 # define S_IWGRP _S_IWRITE 63 # endif 64 # ifndef S_IROTH 65 # define S_IROTH _S_IREAD 66 # define S_IWOTH _S_IWRITE 67 # endif 68 69 # ifndef O_RDONLY 70 # define O_RDONLY _O_RDONLY 71 # define O_WRONLY _O_WRONLY 72 # define O_RDWR _O_RDWR 73 # define O_APPEND _O_APPEND 74 # define O_CREAT _O_CREAT 75 # define O_TRUNC _O_TRUNC 76 # define O_TEXT _O_TEXT 77 # define O_BINARY _O_BINARY 78 # endif 79 80 # ifndef O_ACCMODE 81 # define O_ACCMODE (O_RDONLY|O_WRONLY|O_RDWR) 82 # endif 83 #endif 84 85 const _STLP_fd INVALID_STLP_FD = -1; 86 87 88 # ifdef __MSL__ 89 # define _O_TEXT 0x0 90 # if !defined( O_TEXT ) 91 # define O_TEXT _O_TEXT 92 # endif 93 # define _S_IFREG S_IFREG 94 # define S_IREAD S_IRUSR 95 # define S_IWRITE S_IWUSR 96 # define S_IEXEC S_IXUSR 97 # define _S_IWRITE S_IWRITE 98 # define _S_IREAD S_IREAD 99 # define _open open 100 # define _close close 101 # define _read read 102 # define _write write 103 # endif 104 105 _STLP_BEGIN_NAMESPACE 106 107 // Compare with streamoff definition in stl/char_traits.h! 108 109 #if defined (_STLP_USE_DEFAULT_FILE_OFFSET) || \ 110 (!defined(_LARGEFILE_SOURCE) && !defined(_LARGEFILE64_SOURCE)) 111 # define FOPEN fopen 112 # define FSEEK fseek 113 # define FSTAT fstat 114 # define STAT stat 115 # define FTELL ftell 116 #else 117 # define FOPEN fopen64 118 # define FSEEK fseeko64 119 # define FSTAT fstat64 120 # define STAT stat64 121 # define FTELL ftello64 122 #endif 123 124 _STLP_MOVE_TO_PRIV_NAMESPACE 125 126 // Helper functions for _Filebuf_base. 127 128 static bool __is_regular_file(_STLP_fd fd) { 129 struct STAT buf; 130 return FSTAT(fd, &buf) == 0 && (buf.st_mode & S_IFREG) != 0 ; 131 } 132 133 // Number of characters in the file. 134 static streamoff __file_size(_STLP_fd fd) { 135 streamoff ret = 0; 136 137 struct STAT buf; 138 if (FSTAT(fd, &buf) == 0 && (buf.st_mode & S_IFREG) != 0) 139 ret = buf.st_size > 0 ? buf.st_size : 0; 140 141 return ret; 142 } 143 144 _STLP_MOVE_TO_STD_NAMESPACE 145 146 // All version of Unix have mmap and lseek system calls. Some also have 147 // longer versions of those system calls to accommodate 64-bit offsets. 148 // If we're on a Unix system, define some macros to encapsulate those 149 // differences. 150 151 size_t _Filebuf_base::_M_page_size = 4096; 152 153 _Filebuf_base::_Filebuf_base() 154 : _M_file_id(INVALID_STLP_FD), 155 _M_openmode(0), 156 _M_is_open(false), 157 _M_should_close(false) 158 {} 159 160 void _Filebuf_base::_S_initialize() 161 { 162 163 } 164 165 // Return the size of the file. This is a wrapper for stat. 166 // Returns zero if the size cannot be determined or is ill-defined. 167 streamoff _Filebuf_base::_M_file_size() 168 { 169 return _STLP_PRIV __file_size(_M_file_id); 170 } 171 172 bool _Filebuf_base::_M_open(const char* name, ios_base::openmode openmode, 173 long permission) 174 { 175 _STLP_fd file_no; 176 177 if (_M_is_open) 178 return false; 179 180 // use FILE-based i/o 181 const char* flags; 182 183 switch (openmode & (~ios_base::ate)) { 184 case ios_base::out: 185 case ios_base::out | ios_base::trunc: 186 flags = "w"; 187 break; 188 189 case ios_base::out | ios_base::binary: 190 case ios_base::out | ios_base::trunc | ios_base::binary: 191 flags = "wb"; 192 break; 193 194 case ios_base::out | ios_base::app: 195 flags = "a"; 196 break; 197 198 case ios_base::out | ios_base::app | ios_base::binary: 199 flags = "ab"; 200 break; 201 202 case ios_base::in: 203 flags = "r"; 204 break; 205 206 case ios_base::in | ios_base::binary: 207 flags = "rb"; 208 break; 209 210 case ios_base::in | ios_base::out: 211 flags = "r+"; 212 break; 213 214 case ios_base::in | ios_base::out | ios_base::binary: 215 flags = "r+b"; 216 break; 217 218 case ios_base::in | ios_base::out | ios_base::trunc: 219 flags = "w+"; 220 break; 221 222 case ios_base::in | ios_base::out | ios_base::trunc | ios_base::binary: 223 flags = "w+b"; 224 break; 225 226 default: // The above are the only combinations of 227 return false; // flags allowed by the C++ standard. 228 } 229 230 // fbp : TODO : set permissions ! 231 (void)permission; // currently unused //*TY 02/26/2000 - added to suppress warning message 232 _M_file = FOPEN(name, flags); 233 234 if (_M_file) { 235 file_no = fileno(_M_file); 236 } else { 237 return false; 238 } 239 240 // unset buffering immediately 241 setbuf(_M_file, 0); 242 243 _M_is_open = true; 244 245 if (openmode & ios_base::ate) { 246 if (FSEEK(_M_file, 0, SEEK_END) != 0) 247 _M_is_open = false; 248 } 249 250 _M_file_id = file_no; 251 _M_should_close = _M_is_open; 252 _M_openmode = openmode; 253 254 if (_M_is_open) 255 _M_regular_file = _STLP_PRIV __is_regular_file(_M_file_id); 256 257 return (_M_is_open != 0); 258 } 259 260 261 bool _Filebuf_base::_M_open(const char* name, ios_base::openmode openmode) 262 { 263 // This doesn't really grant everyone in the world read/write 264 // access. On Unix, file-creation system calls always clear 265 // bits that are set in the umask from the permissions flag. 266 return this->_M_open(name, openmode, S_IRUSR | S_IWUSR | S_IRGRP | 267 S_IWGRP | S_IROTH | S_IWOTH); 268 } 269 270 // Associated the filebuf with a file descriptor pointing to an already- 271 // open file. Mode is set to be consistent with the way that the file 272 // was opened. 273 bool _Filebuf_base::_M_open( int file_no, ios_base::openmode ) 274 { 275 if (_M_is_open || file_no < 0) 276 return false; 277 278 struct STAT buf; 279 if (FSTAT(file_no, &buf) != 0) 280 return false; 281 int mode = buf.st_mode; 282 283 switch ( mode & (S_IWRITE | S_IREAD) ) { 284 case S_IREAD: 285 _M_openmode = ios_base::in; 286 break; 287 case S_IWRITE: 288 _M_openmode = ios_base::out; 289 break; 290 case (S_IWRITE | S_IREAD): 291 _M_openmode = ios_base::in | ios_base::out; 292 break; 293 default: 294 return false; 295 } 296 _M_file_id = file_no; 297 _M_is_open = true; 298 _M_should_close = false; 299 _M_regular_file = _STLP_PRIV __is_regular_file(_M_file_id); 300 return true; 301 } 302 303 bool _Filebuf_base::_M_open(FILE *file, ios_base::openmode openmode) 304 { 305 _STLP_fd file_no; 306 307 if (_M_is_open) 308 return false; 309 310 _M_file = file; 311 312 if (_M_file) { 313 file_no = fileno(_M_file); 314 } else { 315 return false; 316 } 317 318 // unset buffering immediately 319 setbuf(_M_file, 0); 320 321 _M_is_open = true; 322 323 if (openmode & ios_base::ate) { 324 if (FSEEK(_M_file, 0, SEEK_END) != 0) 325 _M_is_open = false; 326 } 327 328 _M_file_id = file_no; 329 _M_should_close = _M_is_open; 330 _M_openmode = openmode; 331 332 if (_M_is_open) 333 _M_regular_file = _STLP_PRIV __is_regular_file(_M_file_id); 334 335 return (_M_is_open != 0); 336 } 337 338 bool _Filebuf_base::_M_close() 339 { 340 if (!_M_is_open) 341 return false; 342 343 bool ok = _M_should_close ? (fclose(_M_file) == 0) : true; 344 345 _M_is_open = _M_should_close = false; 346 _M_openmode = 0; 347 return ok; 348 } 349 350 // Read up to n characters into a buffer. Return value is number of 351 // characters read. 352 ptrdiff_t _Filebuf_base::_M_read(char* buf, ptrdiff_t n) { 353 return fread(buf, 1, n, _M_file); 354 } 355 356 // Write n characters from a buffer. Return value: true if we managed 357 // to write the entire buffer, false if we didn't. 358 bool _Filebuf_base::_M_write(char* buf, ptrdiff_t n) 359 { 360 for (;;) { 361 ptrdiff_t written = fwrite(buf, 1, n, _M_file); 362 363 if (n == written) { 364 return true; 365 } 366 367 if (written > 0 && written < n) { 368 n -= written; 369 buf += written; 370 } else { 371 return false; 372 } 373 } 374 } 375 376 // Wrapper for lseek or the like. 377 streamoff _Filebuf_base::_M_seek(streamoff offset, ios_base::seekdir dir) 378 { 379 int whence; 380 381 switch ( dir ) { 382 case ios_base::beg: 383 if (offset < 0 /* || offset > _M_file_size() */ ) 384 return streamoff(-1); 385 whence = SEEK_SET; 386 break; 387 case ios_base::cur: 388 whence = SEEK_CUR; 389 break; 390 case ios_base::end: 391 if (/* offset > 0 || */ -offset > _M_file_size() ) 392 return streamoff(-1); 393 whence = SEEK_END; 394 break; 395 default: 396 return streamoff(-1); 397 } 398 399 if ( FSEEK(_M_file, offset, whence) == 0 ) { 400 return FTELL(_M_file); 401 } 402 403 return streamoff(-1); 404 } 405 406 407 // Attempts to memory-map len bytes of the current file, starting 408 // at position offset. Precondition: offset is a multiple of the 409 // page size. Postcondition: return value is a null pointer if the 410 // memory mapping failed. Otherwise the return value is a pointer to 411 // the memory-mapped file and the file position is set to offset. 412 void *_Filebuf_base::_M_mmap(streamoff, streamoff ) 413 { 414 return 0; 415 } 416 417 void _Filebuf_base::_M_unmap(void*, streamoff) 418 { 419 // precondition : there is a valid mapping at the moment 420 } 421 422 _STLP_END_NAMESPACE 423