os/ossrv/stdcpp/src/fstream.cpp
author sl
Tue, 10 Jun 2014 14:32:02 +0200
changeset 1 260cb5ec6c19
permissions -rw-r--r--
Update contrib.
sl@0
     1
/*
sl@0
     2
 * © Portions copyright (c) 2006-2007 Nokia Corporation.  All rights reserved.
sl@0
     3
 *
sl@0
     4
 * Copyright (c) 1999
sl@0
     5
 * Silicon Graphics Computer Systems, Inc.
sl@0
     6
 *
sl@0
     7
 * Copyright (c) 1999 
sl@0
     8
 * Boris Fomitchev
sl@0
     9
 *
sl@0
    10
 * This material is provided "as is", with absolutely no warranty expressed
sl@0
    11
 * or implied. Any use is at your own risk.
sl@0
    12
 *
sl@0
    13
 * Permission to use or copy this software for any purpose is hereby granted 
sl@0
    14
 * without fee, provided the above notices are retained on all copies.
sl@0
    15
 * Permission to modify the code and to distribute modified code is granted,
sl@0
    16
 * provided the above notices are retained, and a notice that the code was
sl@0
    17
 * modified is included with the above copyright notice.
sl@0
    18
 *
sl@0
    19
 */ 
sl@0
    20
sl@0
    21
sl@0
    22
# include "stlport_prefix.h"
sl@0
    23
sl@0
    24
# if defined  (__SUNPPRO_CC)  && !defined (_STLP_NO_NEW_C_HEADERS)
sl@0
    25
#  include <time.h>
sl@0
    26
// For sunpro, it chokes if time.h is included through stat.h
sl@0
    27
# endif
sl@0
    28
sl@0
    29
#include <fstream>
sl@0
    30
sl@0
    31
sl@0
    32
#ifdef __CYGWIN__
sl@0
    33
#  define __int64 long long
sl@0
    34
#endif
sl@0
    35
sl@0
    36
#if defined (_STLP_USE_UNIX_IO)
sl@0
    37
extern "C" {
sl@0
    38
// open/close/read/write
sl@0
    39
#  include <sys/stat.h>           // For stat
sl@0
    40
#if !defined (_CRAY) && ! defined (__EMX__)
sl@0
    41
#  include <sys/mman.h>           // For mmap
sl@0
    42
#endif
sl@0
    43
sl@0
    44
//  on HP-UX 11, this one contradicts with pthread.h on pthread_atfork, unless we unset this
sl@0
    45
#  if defined (__hpux) && defined (__GNUC__)
sl@0
    46
#   undef _INCLUDE_POSIX1C_SOURCE
sl@0
    47
#  endif
sl@0
    48
sl@0
    49
#  include <unistd.h>
sl@0
    50
#  include <fcntl.h>
sl@0
    51
}
sl@0
    52
# ifdef __APPLE__
sl@0
    53
#  include <sys/sysctl.h>
sl@0
    54
# endif
sl@0
    55
#elif defined (_STLP_USE_WIN32_IO)
sl@0
    56
# define WIN32_LEAN_AND_MEAN
sl@0
    57
#  include <windows.h>
sl@0
    58
sl@0
    59
# ifdef __BORLANDC__
sl@0
    60
#  if (__BORLANDC__ > 0x530)
sl@0
    61
#  include <cfcntl.h>            // For _O_RDONLY, etc
sl@0
    62
#  else
sl@0
    63
#  include <fcntl.h>            // For _O_RDONLY, etc
sl@0
    64
#  endif
sl@0
    65
#  include <sys/stat.h>         // For _fstat
sl@0
    66
# elif !defined(_STLP_WINCE)
sl@0
    67
#  include <io.h>               // For _get_osfhandle
sl@0
    68
#  include <fcntl.h>            // For _O_RDONLY, etc
sl@0
    69
#  include <sys/stat.h>         // For _fstat
sl@0
    70
# endif
sl@0
    71
sl@0
    72
# define _TEXTBUF_SIZE 0x1000
sl@0
    73
#elif defined (_STLP_USE_UNIX_EMULATION_IO)
sl@0
    74
#  if defined( __MSL__ )
sl@0
    75
#   include <unistd.h>
sl@0
    76
#  else
sl@0
    77
#   include <io.h>
sl@0
    78
#  endif
sl@0
    79
#  include <fcntl.h>
sl@0
    80
#  include <sys/stat.h>
sl@0
    81
sl@0
    82
#elif defined (_STLP_USE_STDIO_IO)
sl@0
    83
#  include <cstdio>
sl@0
    84
sl@0
    85
#  if !(defined(__MRC__) || defined(__SC__) || defined(__ISCPP__) )
sl@0
    86
extern "C" {
sl@0
    87
#   include <sys/stat.h>
sl@0
    88
}
sl@0
    89
#  endif
sl@0
    90
#  if defined( __MSL__ )
sl@0
    91
#   include <unix.h>
sl@0
    92
#  endif
sl@0
    93
#  if defined(__ISCPP__)
sl@0
    94
#   include <c_locale_is/filestat.h>
sl@0
    95
#  endif
sl@0
    96
#  if (defined(__BEOS__) && defined(__INTEL__)) || defined(__SYMBIAN32__)
sl@0
    97
#   include <fcntl.h>
sl@0
    98
#   include <sys/stat.h>         // For _fstat
sl@0
    99
#   define _S_IREAD S_IREAD
sl@0
   100
#   define _S_IWRITE S_IWRITE
sl@0
   101
#   define _S_IFREG S_IFREG
sl@0
   102
#  endif
sl@0
   103
#else
sl@0
   104
#error "Configure i/o !"
sl@0
   105
#endif
sl@0
   106
sl@0
   107
// map permission masks
sl@0
   108
#if defined (_STLP_USE_UNIX_EMULATION_IO) || defined (_STLP_USE_STDIO_IO)
sl@0
   109
# ifndef S_IRUSR
sl@0
   110
#  define S_IRUSR _S_IREAD
sl@0
   111
#  define S_IWUSR _S_IWRITE
sl@0
   112
#  define S_IRGRP _S_IREAD
sl@0
   113
#  define S_IWGRP _S_IWRITE
sl@0
   114
#  define S_IROTH _S_IREAD
sl@0
   115
#  define S_IWOTH _S_IWRITE
sl@0
   116
# endif
sl@0
   117
# ifndef O_RDONLY
sl@0
   118
#   define O_RDONLY _O_RDONLY
sl@0
   119
#   define O_WRONLY _O_WRONLY
sl@0
   120
#   define O_RDWR   _O_RDWR
sl@0
   121
#   define O_APPEND _O_APPEND
sl@0
   122
#   define O_CREAT  _O_CREAT
sl@0
   123
#   define O_TRUNC  _O_TRUNC
sl@0
   124
#   define O_TEXT   _O_TEXT
sl@0
   125
#   define O_BINARY _O_BINARY
sl@0
   126
# endif
sl@0
   127
sl@0
   128
# ifdef __MSL__
sl@0
   129
#  define _O_TEXT 0x0
sl@0
   130
#  if !defined( O_TEXT )
sl@0
   131
#   define O_TEXT _O_TEXT
sl@0
   132
#  endif
sl@0
   133
#  define _S_IFREG S_IFREG
sl@0
   134
#  define S_IREAD        S_IRUSR
sl@0
   135
#  define S_IWRITE       S_IWUSR
sl@0
   136
#  define S_IEXEC        S_IXUSR
sl@0
   137
#  define _S_IWRITE S_IWRITE
sl@0
   138
#  define _S_IREAD S_IREAD
sl@0
   139
#  define _open open
sl@0
   140
#  define _lseek lseek
sl@0
   141
#  define _close close
sl@0
   142
#  define _read read
sl@0
   143
#  define _write write
sl@0
   144
# endif
sl@0
   145
sl@0
   146
#endif
sl@0
   147
sl@0
   148
#ifndef O_ACCMODE
sl@0
   149
#   define O_ACCMODE (O_RDONLY|O_WRONLY|O_RDWR)
sl@0
   150
#endif
sl@0
   151
sl@0
   152
#include "fstream_impl.h"
sl@0
   153
sl@0
   154
# ifdef _STLP_LONG_LONG
sl@0
   155
# define ULL(x) ((unsigned _STLP_LONG_LONG)x)
sl@0
   156
// # elif defined (_MSC_VER) || defined (__BORLANDC__)
sl@0
   157
// # define ULL(x) ((__int64)x)
sl@0
   158
# elif defined(__MRC__) || defined(__SC__)		//*TY 02/25/2000 - added support for MPW compilers
sl@0
   159
# include <Math64.h>
sl@0
   160
# define ULL(x) (U64SetU(x))
sl@0
   161
# elif defined(__ISCPP__)
sl@0
   162
#  include "uint64.h"
sl@0
   163
# else
sl@0
   164
#  error "there should be some long long type on the system!"
sl@0
   165
# endif
sl@0
   166
sl@0
   167
__SGI_BEGIN_NAMESPACE
sl@0
   168
sl@0
   169
#if !defined(__MSL__) && !defined(__MRC__) && !defined(__SC__) && !defined(_STLP_WINCE)		//*TY 04/15/2000 - exclude mpw compilers also
sl@0
   170
ios_base::openmode flag_to_openmode(int mode)
sl@0
   171
{
sl@0
   172
  ios_base::openmode ret = ios_base::in;
sl@0
   173
  
sl@0
   174
  switch(mode & O_ACCMODE) {
sl@0
   175
sl@0
   176
  case O_RDONLY:
sl@0
   177
    ret = ios_base::in; break;
sl@0
   178
sl@0
   179
  case O_WRONLY:
sl@0
   180
    ret = ios_base::out; break;
sl@0
   181
sl@0
   182
  case O_RDWR:
sl@0
   183
    ret = ios_base::in | ios_base::out; break;
sl@0
   184
sl@0
   185
  }
sl@0
   186
sl@0
   187
  if (mode & O_APPEND)
sl@0
   188
    ret |= ios_base::app;
sl@0
   189
  
sl@0
   190
#  ifdef _STLP_USE_WIN32_IO
sl@0
   191
  if (mode & O_BINARY)
sl@0
   192
    ret |= ios_base::binary;
sl@0
   193
#  endif // _MSC_VER
sl@0
   194
sl@0
   195
  return ret;
sl@0
   196
  
sl@0
   197
}
sl@0
   198
# endif /* MSL */
sl@0
   199
sl@0
   200
// Helper functions for _Filebuf_base.
sl@0
   201
sl@0
   202
bool __is_regular_file(_STLP_fd fd) {
sl@0
   203
sl@0
   204
#if defined (_STLP_UNIX)
sl@0
   205
sl@0
   206
  struct stat buf;
sl@0
   207
  return fstat(fd, &buf) == 0 && S_ISREG(buf.st_mode);
sl@0
   208
sl@0
   209
#elif defined(__MRC__) || defined(__SC__)		//*TY 02/25/2000 - added support for MPW compilers
sl@0
   210
sl@0
   211
  #pragma unused(fd)
sl@0
   212
  return true;  // each file is a regular file under mac os, isn't it? (we don't have fstat())
sl@0
   213
sl@0
   214
#elif defined(_STLP_USE_UNIX_EMULATION_IO) || defined (_STLP_USE_STDIO_IO)
sl@0
   215
sl@0
   216
  struct stat buf;
sl@0
   217
  return fstat(fd, &buf) == 0 && (buf.st_mode & _S_IFREG) != 0 ;
sl@0
   218
sl@0
   219
#   elif defined (_STLP_USE_WIN32_IO) && !defined(_STLP_WINCE)
sl@0
   220
sl@0
   221
  return (GetFileType(fd) & ~FILE_TYPE_REMOTE) == FILE_TYPE_DISK;
sl@0
   222
sl@0
   223
#else
sl@0
   224
  (void)fd;    // dwa 4/27/00 - suppress unused parameter warning
sl@0
   225
  return false;
sl@0
   226
sl@0
   227
#endif
sl@0
   228
}
sl@0
   229
sl@0
   230
// Number of characters in the file.  
sl@0
   231
streamoff __file_size(_STLP_fd fd) {
sl@0
   232
 streamoff ret = 0;
sl@0
   233
sl@0
   234
#if defined (_STLP_UNIX)
sl@0
   235
sl@0
   236
  struct stat buf;
sl@0
   237
  if(fstat(fd, &buf) == 0 && S_ISREG(buf.st_mode))
sl@0
   238
    ret = buf.st_size > 0 ? buf.st_size : 0;
sl@0
   239
sl@0
   240
#elif defined(__MRC__) || defined(__SC__)		//*TY 02/25/2000 - added support for MPW compilers
sl@0
   241
sl@0
   242
  #pragma unused(fd)
sl@0
   243
sl@0
   244
#elif defined(_STLP_USE_UNIX_EMULATION_IO) || defined (_STLP_USE_STDIO_IO)
sl@0
   245
sl@0
   246
  struct stat buf;
sl@0
   247
  if(fstat(fd, &buf) == 0 && (buf.st_mode & _S_IFREG) != 0)
sl@0
   248
    ret = buf.st_size > 0 ? buf.st_size : 0;
sl@0
   249
sl@0
   250
#   elif defined (_STLP_USE_WIN32_IO)
sl@0
   251
sl@0
   252
 LARGE_INTEGER li;
sl@0
   253
 li.LowPart = GetFileSize(fd, (unsigned long*) &li.HighPart);
sl@0
   254
 if (li.LowPart == 0xFFFFFFFF && GetLastError() != NO_ERROR )
sl@0
   255
 {
sl@0
   256
   ret = 0;
sl@0
   257
 } else
sl@0
   258
   ret = li.QuadPart;
sl@0
   259
 
sl@0
   260
# else
sl@0
   261
  (void)fd;    // dwa 4/27/00 - suppress unused parameter warning  
sl@0
   262
# endif
sl@0
   263
  
sl@0
   264
  return ret;
sl@0
   265
}
sl@0
   266
sl@0
   267
sl@0
   268
// Visual C++ and Intel use this, but not Metrowerks
sl@0
   269
// Also MinGW, msvcrt.dll (but not crtdll.dll) dependent version
sl@0
   270
#if (!defined(__MSL__) && !defined(_STLP_WINCE) && defined( _MSC_VER ) && defined(_WIN32)) || \
sl@0
   271
 (defined(__MINGW32__) && defined(__MSVCRT__) && !defined (__SYMBIAN32__))
sl@0
   272
sl@0
   273
// fcntl(fileno, F_GETFL) for Microsoft library
sl@0
   274
// 'semi-documented' defines:
sl@0
   275
#define IOINFO_L2E          5
sl@0
   276
#define IOINFO_ARRAY_ELTS   (1 << IOINFO_L2E)
sl@0
   277
#define _pioinfo(i) ( __pioinfo[(i) >> IOINFO_L2E] + \
sl@0
   278
              ((i) & (IOINFO_ARRAY_ELTS - 1)) )
sl@0
   279
#define FAPPEND         0x20    // O_APPEND flag
sl@0
   280
#define FTEXT           0x80    // O_TEXT flag
sl@0
   281
// end of 'semi-documented' defines
sl@0
   282
sl@0
   283
// fbp : empirical
sl@0
   284
#define F_WRITABLE           0x04
sl@0
   285
sl@0
   286
// 'semi-documented' internal structure
sl@0
   287
extern "C" {
sl@0
   288
  struct ioinfo {
sl@0
   289
    long osfhnd;    // the real os HANDLE
sl@0
   290
    char osfile;    // file handle flags
sl@0
   291
    char pipech;    // pipe buffer
sl@0
   292
#   ifdef _MT
sl@0
   293
    // multi-threaded locking
sl@0
   294
    int lockinitflag;
sl@0
   295
    CRITICAL_SECTION lock;
sl@0
   296
#   endif  /* _MT */
sl@0
   297
  };
sl@0
   298
#ifdef __MINGW32__ 
sl@0
   299
 __MINGW_IMPORT ioinfo * __pioinfo[];
sl@0
   300
#else
sl@0
   301
  extern _CRTIMP ioinfo * __pioinfo[];
sl@0
   302
#endif
sl@0
   303
sl@0
   304
} // extern "C"
sl@0
   305
// end of 'semi-documented' declarations 
sl@0
   306
sl@0
   307
ios_base::openmode _get_osfflags(int fd, HANDLE oshandle) {
sl@0
   308
  char dosflags = 0;
sl@0
   309
sl@0
   310
  dosflags = _pioinfo(fd)->osfile;
sl@0
   311
    // end of 'semi-documented' stuff 
sl@0
   312
  int mode = 0;
sl@0
   313
  
sl@0
   314
  if (dosflags & FAPPEND)
sl@0
   315
    mode |= O_APPEND;
sl@0
   316
  if (dosflags & FTEXT)
sl@0
   317
    mode |= O_TEXT;
sl@0
   318
  else
sl@0
   319
    mode |= O_BINARY;
sl@0
   320
sl@0
   321
  // we have to guess
sl@0
   322
# ifdef __macintosh
sl@0
   323
   DWORD dummy, dummy2;
sl@0
   324
   if (WriteFile(oshandle, &dummy2, 0, &dummy, 0))
sl@0
   325
     mode |= O_RDWR;
sl@0
   326
   else
sl@0
   327
     mode |= O_RDONLY;
sl@0
   328
# else
sl@0
   329
   // on Windows, this seems to work...
sl@0
   330
   if (dosflags & F_WRITABLE)
sl@0
   331
      mode |= O_RDWR;
sl@0
   332
   else
sl@0
   333
      mode |= O_RDONLY;
sl@0
   334
# endif
sl@0
   335
  
sl@0
   336
  return flag_to_openmode(mode);
sl@0
   337
}
sl@0
   338
sl@0
   339
#elif defined(__DMC__)
sl@0
   340
sl@0
   341
#define FHND_APPEND 0x04
sl@0
   342
#define FHND_DEVICE 0x08
sl@0
   343
#define FHND_TEXT   0x10
sl@0
   344
sl@0
   345
extern "C" unsigned char __fhnd_info[_NFILE];
sl@0
   346
sl@0
   347
ios_base::openmode _get_osfflags(int fd, HANDLE oshandle) {
sl@0
   348
  int mode = 0;
sl@0
   349
sl@0
   350
  if (__fhnd_info[fd] & FHND_APPEND)
sl@0
   351
    mode |= O_APPEND;
sl@0
   352
sl@0
   353
  if (__fhnd_info[fd] & FHND_TEXT == 0)
sl@0
   354
    mode |= O_BINARY;
sl@0
   355
sl@0
   356
  for (FILE *fp = &_iob[0]; fp < &_iob[_NFILE]; fp++)
sl@0
   357
  {
sl@0
   358
    if ((fileno(fp) == fd) && (fp->_flag & (_IOREAD | _IOWRT | _IORW)))
sl@0
   359
    {
sl@0
   360
      const int osflags = fp->_flag;
sl@0
   361
sl@0
   362
      if ((osflags & _IOREAD) && !(osflags & _IOWRT) && !(osflags & _IORW))
sl@0
   363
	mode |= O_RDONLY;
sl@0
   364
      else if ((osflags & _IOWRT) && !(osflags & _IOREAD) && !(osflags & _IORW))
sl@0
   365
	mode |= O_WRONLY;
sl@0
   366
      else
sl@0
   367
	mode |= O_RDWR;
sl@0
   368
sl@0
   369
      break;
sl@0
   370
    }
sl@0
   371
  }
sl@0
   372
sl@0
   373
  return flag_to_openmode(mode);
sl@0
   374
}
sl@0
   375
#endif // _MSC_VER
sl@0
   376
sl@0
   377
__SGI_END_NAMESPACE
sl@0
   378
sl@0
   379
// All version of Unix have mmap and lseek system calls.  Some also have
sl@0
   380
// longer versions of thos system calls to accommodate 64-bit offsets.
sl@0
   381
// If we're on a Unix system, define some macros to encapsulate those
sl@0
   382
// differences.
sl@0
   383
#ifdef _STLP_USE_UNIX_IO
sl@0
   384
# ifdef __sgi /* IRIX */
sl@0
   385
#  define LSEEK lseek64
sl@0
   386
#  define MMAP  mmap64
sl@0
   387
# else
sl@0
   388
#  define LSEEK lseek
sl@0
   389
#  define MMAP  mmap
sl@0
   390
# endif
sl@0
   391
sl@0
   392
#ifndef MAP_FAILED /* MMAP failure return code */
sl@0
   393
# define MAP_FAILED -1
sl@0
   394
#endif
sl@0
   395
sl@0
   396
#elif defined (_STLP_USE_UNIX_EMULATION_IO)
sl@0
   397
#  define LSEEK _lseek
sl@0
   398
#endif
sl@0
   399
sl@0
   400
#if defined(__LIBSTD_CPP_SYMBIAN32_WSD__) || defined(_STLP_LIBSTD_CPP_NO_STATIC_VAR_)
sl@0
   401
#include "libstdcppwsd.h"
sl@0
   402
# endif
sl@0
   403
sl@0
   404
_STLP_BEGIN_NAMESPACE
sl@0
   405
sl@0
   406
#if defined(__LIBSTD_CPP_SYMBIAN32_WSD__) || defined(_STLP_LIBSTD_CPP_NO_STATIC_VAR_)
sl@0
   407
void filebuf_page_size_init()
sl@0
   408
{
sl@0
   409
	get_fstream_Filebuf_Base_GetPageSize() = 4096;	
sl@0
   410
}
sl@0
   411
#else
sl@0
   412
size_t
sl@0
   413
_Filebuf_base::_M_page_size = 4096;
sl@0
   414
#endif //__LIBSTD_CPP_SYMBIAN32_WSD__
sl@0
   415
sl@0
   416
_STLP_EXP_DECLSPEC _Filebuf_base::_Filebuf_base()
sl@0
   417
  : _M_file_id((_STLP_fd)-1),
sl@0
   418
    _M_openmode(0),
sl@0
   419
    _M_is_open(false),
sl@0
   420
    _M_should_close(false)
sl@0
   421
{
sl@0
   422
#if defined(__LIBSTD_CPP_SYMBIAN32_WSD__) || defined(_STLP_LIBSTD_CPP_NO_STATIC_VAR_)
sl@0
   423
  if (!get_fstream_Filebuf_Base_GetPageSize())
sl@0
   424
#else
sl@0
   425
  if (!_M_page_size)
sl@0
   426
#endif  //__SYMBIAN32__
sl@0
   427
#if defined (_STLP_UNIX)  && !defined(__DJGPP) && !defined(_CRAY)
sl@0
   428
#  if defined (__APPLE__)
sl@0
   429
   {
sl@0
   430
   int mib[2];
sl@0
   431
   size_t pagesize, len;
sl@0
   432
   mib[0] = CTL_HW;
sl@0
   433
   mib[1] = HW_PAGESIZE;
sl@0
   434
   len = sizeof(pagesize);
sl@0
   435
   sysctl(mib, 2, &pagesize, &len, NULL, 0);
sl@0
   436
   _M_page_size = pagesize;
sl@0
   437
   }
sl@0
   438
# elif defined(__DJGPP) && defined(_CRAY)
sl@0
   439
   _M_page_size = BUFSIZ;
sl@0
   440
#  else
sl@0
   441
#if defined(__LIBSTD_CPP_SYMBIAN32_WSD__) || defined(_STLP_LIBSTD_CPP_NO_STATIC_VAR_)
sl@0
   442
  get_fstream_Filebuf_Base_GetPageSize() = sysconf(_SC_PAGESIZE);
sl@0
   443
#else
sl@0
   444
  _M_page_size = sysconf(_SC_PAGESIZE);
sl@0
   445
#endif
sl@0
   446
#  endif
sl@0
   447
# elif defined (_STLP_USE_WIN32_IO)
sl@0
   448
  {
sl@0
   449
  SYSTEM_INFO SystemInfo;
sl@0
   450
  GetSystemInfo(&SystemInfo);
sl@0
   451
  _M_page_size = SystemInfo.dwPageSize;
sl@0
   452
  // might be .dwAllocationGranularity
sl@0
   453
  }
sl@0
   454
  //  _M_CRLF_trans_buf = 0,
sl@0
   455
  //  _M_trans_buf_end=0,   
sl@0
   456
  _M_view_id = 0;
sl@0
   457
#endif
sl@0
   458
sl@0
   459
#if defined(__LIBSTD_CPP_SYMBIAN32_WSD__) || defined(_STLP_LIBSTD_CPP_NO_STATIC_VAR_)
sl@0
   460
  if (get_fstream_Filebuf_Base_GetPageSize() <= 0 )
sl@0
   461
      get_fstream_Filebuf_Base_GetPageSize() = 4096;
sl@0
   462
#else
sl@0
   463
  if (_M_page_size <=0 )
sl@0
   464
    _M_page_size = 4096;
sl@0
   465
#endif //__LIBSTD_CPP_SYMBIAN32_WSD__  
sl@0
   466
sl@0
   467
}
sl@0
   468
sl@0
   469
sl@0
   470
// Return the size of the file.  This is a wrapper for stat.
sl@0
   471
// Returns zero if the size cannot be determined or is ill-defined.
sl@0
   472
_STLP_EXP_DECLSPEC streamoff 
sl@0
   473
_Filebuf_base::_M_file_size()
sl@0
   474
{
sl@0
   475
  return _SgI::__file_size(_M_file_id);
sl@0
   476
}
sl@0
   477
sl@0
   478
_STLP_EXP_DECLSPEC bool _Filebuf_base::_M_open(const char* name, ios_base::openmode openmode,
sl@0
   479
                            long permission)
sl@0
   480
{
sl@0
   481
  _STLP_fd file_no;
sl@0
   482
sl@0
   483
  if (_M_is_open)
sl@0
   484
    return false;
sl@0
   485
sl@0
   486
#if defined (_STLP_USE_UNIX_IO) || defined (_STLP_USE_UNIX_EMULATION_IO)
sl@0
   487
sl@0
   488
  int flags = 0;
sl@0
   489
sl@0
   490
  // Unix makes no distinction between text and binary files.
sl@0
   491
  switch(openmode & (~ios_base::ate & ~ios_base::binary)) {
sl@0
   492
  case ios_base::out:
sl@0
   493
  case ios_base::out | ios_base::trunc:
sl@0
   494
    flags = O_WRONLY | O_CREAT | O_TRUNC;
sl@0
   495
    break;
sl@0
   496
  case ios_base::out | ios_base::app:
sl@0
   497
    flags = O_WRONLY | O_CREAT | O_APPEND;
sl@0
   498
    break;
sl@0
   499
  case ios_base::in:
sl@0
   500
    flags = O_RDONLY;
sl@0
   501
    permission = 0;             // Irrelevant unless we're writing.
sl@0
   502
    break;
sl@0
   503
  case ios_base::in | ios_base::out:
sl@0
   504
    flags = O_RDWR;
sl@0
   505
    break;
sl@0
   506
  case ios_base::in | ios_base::out | ios_base::trunc:
sl@0
   507
    flags = O_RDWR | O_CREAT | O_TRUNC;
sl@0
   508
    break;
sl@0
   509
  default:                      // The above are the only combinations of 
sl@0
   510
    return false;               // flags allowed by the C++ standard.
sl@0
   511
  }
sl@0
   512
sl@0
   513
# if defined (_STLP_USE_UNIX_EMULATION_IO)
sl@0
   514
sl@0
   515
  if (openmode & ios_base::binary)
sl@0
   516
    flags |= O_BINARY;
sl@0
   517
  else
sl@0
   518
    flags |= O_TEXT;
sl@0
   519
sl@0
   520
  file_no = _open(name, flags, permission);
sl@0
   521
sl@0
   522
# else
sl@0
   523
sl@0
   524
  file_no = open(name, flags, permission);
sl@0
   525
sl@0
   526
# endif /* _STLP_USE_UNIX_EMULATION_IO */
sl@0
   527
sl@0
   528
  if (file_no < 0)
sl@0
   529
    return false;
sl@0
   530
sl@0
   531
  _M_is_open = true;
sl@0
   532
sl@0
   533
  if (openmode & ios_base::ate)
sl@0
   534
    if (LSEEK(file_no, 0, SEEK_END) == -1)
sl@0
   535
      _M_is_open = false;
sl@0
   536
  
sl@0
   537
#elif defined (_STLP_USE_STDIO_IO)
sl@0
   538
  // use FILE-based i/o
sl@0
   539
  const char* flags;
sl@0
   540
sl@0
   541
  switch(openmode & (~ios_base::ate)) {
sl@0
   542
  case ios_base::out:
sl@0
   543
  case ios_base::out | ios_base::trunc:
sl@0
   544
    flags = "w";
sl@0
   545
    break;
sl@0
   546
sl@0
   547
  case ios_base::out | ios_base::binary:
sl@0
   548
  case ios_base::out | ios_base::trunc | ios_base::binary:
sl@0
   549
    flags = "wb";
sl@0
   550
    break;
sl@0
   551
sl@0
   552
  case ios_base::out | ios_base::app:
sl@0
   553
    flags = "a";
sl@0
   554
    break;
sl@0
   555
sl@0
   556
  case ios_base::out | ios_base::app | ios_base::binary:
sl@0
   557
    flags = "ab";
sl@0
   558
    break;
sl@0
   559
sl@0
   560
  case ios_base::in:
sl@0
   561
    flags = "r";
sl@0
   562
    break;
sl@0
   563
sl@0
   564
  case ios_base::in | ios_base::binary:
sl@0
   565
    flags = "rb";
sl@0
   566
    break;
sl@0
   567
sl@0
   568
  case ios_base::in | ios_base::out:
sl@0
   569
    flags = "r+";
sl@0
   570
    break;
sl@0
   571
sl@0
   572
  case ios_base::in | ios_base::out | ios_base::binary:
sl@0
   573
    flags = "r+b";
sl@0
   574
    break;
sl@0
   575
sl@0
   576
sl@0
   577
  case ios_base::in | ios_base::out | ios_base::trunc:
sl@0
   578
    flags = "w+";
sl@0
   579
    break;
sl@0
   580
sl@0
   581
  case ios_base::in | ios_base::out | ios_base::trunc | ios_base::binary:
sl@0
   582
    flags = "w+b";
sl@0
   583
    break;
sl@0
   584
sl@0
   585
  default:                      // The above are the only combinations of 
sl@0
   586
    return false;               // flags allowed by the C++ standard.
sl@0
   587
  }
sl@0
   588
sl@0
   589
  // fbp : TODO : set permissions !
sl@0
   590
  (void)permission; // currently unused		//*TY 02/26/2000 - added to suppress warning message
sl@0
   591
  _M_file = fopen(name, flags);
sl@0
   592
 
sl@0
   593
  if (_M_file) {
sl@0
   594
    file_no = fileno(_M_file);
sl@0
   595
  }
sl@0
   596
  else
sl@0
   597
    return false;
sl@0
   598
sl@0
   599
  // unset buffering immediately
sl@0
   600
  setbuf(_M_file, 0);
sl@0
   601
sl@0
   602
  _M_is_open = true;
sl@0
   603
sl@0
   604
  if (openmode & ios_base::ate)
sl@0
   605
    if (fseek(_M_file, 0, SEEK_END) == -1)
sl@0
   606
      _M_is_open = false;
sl@0
   607
  
sl@0
   608
#   elif defined (_STLP_USE_WIN32_IO)
sl@0
   609
sl@0
   610
  DWORD dwDesiredAccess, dwShareMode, dwCreationDisposition;
sl@0
   611
  bool  doTruncate = false;
sl@0
   612
sl@0
   613
  switch(openmode & (~ios_base::ate & ~ios_base::binary)) {
sl@0
   614
  case ios_base::out:
sl@0
   615
  case ios_base::out | ios_base::trunc:
sl@0
   616
    dwDesiredAccess = GENERIC_WRITE;
sl@0
   617
    dwShareMode = FILE_SHARE_READ | FILE_SHARE_WRITE;
sl@0
   618
    dwCreationDisposition = OPEN_ALWAYS;
sl@0
   619
    // boris : even though it is very non-intuitive, standard
sl@0
   620
    // requires them both to behave same.
sl@0
   621
    doTruncate = true;
sl@0
   622
    break;
sl@0
   623
sl@0
   624
  case ios_base::out | ios_base::app:
sl@0
   625
    dwDesiredAccess = GENERIC_WRITE;
sl@0
   626
    dwShareMode = FILE_SHARE_READ | FILE_SHARE_WRITE;
sl@0
   627
    dwCreationDisposition = OPEN_ALWAYS;
sl@0
   628
    break;
sl@0
   629
  case ios_base::in:
sl@0
   630
    dwDesiredAccess = GENERIC_READ;
sl@0
   631
    dwShareMode = FILE_SHARE_READ | FILE_SHARE_WRITE;
sl@0
   632
    dwCreationDisposition = OPEN_EXISTING;
sl@0
   633
    permission = 0;             // Irrelevant unless we're writing.
sl@0
   634
    break;
sl@0
   635
  case ios_base::in | ios_base::out:
sl@0
   636
    dwDesiredAccess = GENERIC_READ | GENERIC_WRITE;
sl@0
   637
    dwShareMode = FILE_SHARE_READ | FILE_SHARE_WRITE;
sl@0
   638
    dwCreationDisposition = OPEN_EXISTING;
sl@0
   639
    break;
sl@0
   640
  case ios_base::in | ios_base::out | ios_base::trunc:
sl@0
   641
    dwDesiredAccess = GENERIC_READ | GENERIC_WRITE;
sl@0
   642
    dwShareMode = FILE_SHARE_READ | FILE_SHARE_WRITE;
sl@0
   643
    dwCreationDisposition = OPEN_ALWAYS;
sl@0
   644
    doTruncate = true;
sl@0
   645
    break;
sl@0
   646
  default:                      // The above are the only combinations of
sl@0
   647
    return false;               // flags allowed by the C++ standard.
sl@0
   648
  }
sl@0
   649
sl@0
   650
  #if defined(_STLP_WINCE)
sl@0
   651
    file_no = CreateFile(__ASCIIToWide(name).c_str(),
sl@0
   652
  #else
sl@0
   653
    file_no = CreateFileA(name,
sl@0
   654
  #endif
sl@0
   655
                  dwDesiredAccess, dwShareMode, 0,
sl@0
   656
			dwCreationDisposition, permission, 0);
sl@0
   657
  
sl@0
   658
  if ( file_no == INVALID_HANDLE_VALUE )
sl@0
   659
  return false;
sl@0
   660
sl@0
   661
  if ((doTruncate && SetEndOfFile(file_no) == 0) ||
sl@0
   662
      ((openmode & ios_base::ate) && SetFilePointer(file_no, 0, NULL, FILE_END) == -1)) {
sl@0
   663
    CloseHandle(file_no);
sl@0
   664
    return false;
sl@0
   665
  }
sl@0
   666
sl@0
   667
  _M_is_open = true;
sl@0
   668
 
sl@0
   669
#else
sl@0
   670
# error "Port!"  
sl@0
   671
#endif /* __unix */
sl@0
   672
sl@0
   673
sl@0
   674
  _M_file_id = file_no;
sl@0
   675
  _M_should_close = _M_is_open;
sl@0
   676
  _M_openmode = openmode;
sl@0
   677
sl@0
   678
  if (_M_is_open)
sl@0
   679
    _M_regular_file = _SgI::__is_regular_file(_M_file_id);
sl@0
   680
  
sl@0
   681
  return _M_is_open;
sl@0
   682
}
sl@0
   683
sl@0
   684
  
sl@0
   685
_STLP_EXP_DECLSPEC bool _Filebuf_base::_M_open(const char* name, ios_base::openmode openmode)
sl@0
   686
{
sl@0
   687
  // This doesn't really grant everyone in the world read/write
sl@0
   688
  // access.  On Unix, file-creation system calls always clear 
sl@0
   689
  // bits that are set in the umask from the permissions flag.
sl@0
   690
# ifdef _STLP_USE_WIN32_IO
sl@0
   691
  return this->_M_open(name, openmode, FILE_ATTRIBUTE_NORMAL);
sl@0
   692
# elif defined(__MRC__) || defined(__SC__)		//*TY 02/26/2000 - added support for MPW compilers
sl@0
   693
  return this->_M_open(name, openmode, 0);
sl@0
   694
# else
sl@0
   695
  return this->_M_open(name, openmode, S_IRUSR | S_IWUSR | S_IRGRP | 
sl@0
   696
                                       S_IWGRP | S_IROTH | S_IWOTH);
sl@0
   697
# endif
sl@0
   698
}
sl@0
   699
sl@0
   700
      
sl@0
   701
// Associated the filebuf with a file descriptor pointing to an already-
sl@0
   702
// open file.  Mode is set to be consistent with the way that the file
sl@0
   703
// was opened.
sl@0
   704
bool _Filebuf_base::_M_open(int file_no, ios_base::openmode init_mode) {
sl@0
   705
sl@0
   706
  if (_M_is_open || file_no < 0)
sl@0
   707
    return false;
sl@0
   708
sl@0
   709
# if defined (_STLP_UNIX)
sl@0
   710
  (void)init_mode;    // dwa 4/27/00 - suppress unused parameter warning
sl@0
   711
  int mode ;
sl@0
   712
  mode = fcntl(file_no, F_GETFL);
sl@0
   713
sl@0
   714
  if (mode == -1)
sl@0
   715
    return false;
sl@0
   716
sl@0
   717
  _M_openmode = _SgI::flag_to_openmode(mode);
sl@0
   718
  
sl@0
   719
# elif defined(__MRC__) || defined(__SC__)		//*TY 02/26/2000 - added support for MPW compilers
sl@0
   720
  (void)init_mode;    // dwa 4/27/00 - suppress unused parameter warning
sl@0
   721
  switch( _iob[file_no]._flag & (_IOREAD|_IOWRT|_IORW) )
sl@0
   722
  {
sl@0
   723
  case _IOREAD:
sl@0
   724
    _M_openmode = ios_base::in; break;
sl@0
   725
  case _IOWRT:
sl@0
   726
    _M_openmode = ios_base::out; break;
sl@0
   727
  case _IORW:
sl@0
   728
    _M_openmode = ios_base::in | ios_base::out; break;
sl@0
   729
  default:
sl@0
   730
  	return false;
sl@0
   731
  }
sl@0
   732
  
sl@0
   733
# elif defined (_STLP_USE_UNIX_EMULATION_IO) || defined (_STLP_USE_STDIO_IO) 
sl@0
   734
  (void)init_mode;    // dwa 4/27/00 - suppress unused parameter warning
sl@0
   735
  int mode ;
sl@0
   736
  struct stat buf;
sl@0
   737
  if (fstat(file_no, &buf) != 0)
sl@0
   738
    return false;
sl@0
   739
  mode = buf.st_mode;
sl@0
   740
  
sl@0
   741
  switch(mode & (_S_IWRITE | _S_IREAD) ) {
sl@0
   742
  case _S_IREAD:
sl@0
   743
    _M_openmode = ios_base::in; break;
sl@0
   744
  case _S_IWRITE:
sl@0
   745
    _M_openmode = ios_base::out; break;
sl@0
   746
  case (_S_IWRITE | _S_IREAD):
sl@0
   747
    _M_openmode = ios_base::in | ios_base::out; break;
sl@0
   748
  default:
sl@0
   749
    return false;
sl@0
   750
  }
sl@0
   751
# elif (defined(_STLP_USE_WIN32_IO) && defined (_MSC_VER) && !defined(_STLP_WINCE)) || \
sl@0
   752
        (defined(__MINGW32__) && defined(__MSVCRT__)) || defined(__DMC__)
sl@0
   753
sl@0
   754
  if (_M_is_open || file_no == -1)
sl@0
   755
    return false;
sl@0
   756
sl@0
   757
  HANDLE oshandle = (HANDLE)_get_osfhandle(file_no);
sl@0
   758
  
sl@0
   759
  if ((long)oshandle != -1)
sl@0
   760
	file_no = (int)oshandle;
sl@0
   761
  else
sl@0
   762
    return false;
sl@0
   763
  
sl@0
   764
  if (init_mode != ios_base::__default_mode)
sl@0
   765
    _M_openmode = init_mode;
sl@0
   766
  else
sl@0
   767
    _M_openmode = _SgI::_get_osfflags(file_no, oshandle);
sl@0
   768
  
sl@0
   769
# else
sl@0
   770
  (void)init_mode;    // dwa 4/27/00 - suppress unused parameter warning
sl@0
   771
sl@0
   772
  // not available for the API
sl@0
   773
  return false;
sl@0
   774
sl@0
   775
# endif
sl@0
   776
  
sl@0
   777
sl@0
   778
  _M_is_open = true;
sl@0
   779
  _M_file_id = (_STLP_fd)file_no;
sl@0
   780
  _M_should_close = false;
sl@0
   781
  _M_regular_file = _SgI::__is_regular_file(_M_file_id);
sl@0
   782
sl@0
   783
  return true;
sl@0
   784
}
sl@0
   785
sl@0
   786
_STLP_EXP_DECLSPEC bool _Filebuf_base::_M_close() {
sl@0
   787
  if (!_M_is_open)
sl@0
   788
    return false;
sl@0
   789
sl@0
   790
  bool ok;
sl@0
   791
sl@0
   792
  if (!_M_should_close)
sl@0
   793
    ok = true;
sl@0
   794
  else {
sl@0
   795
    
sl@0
   796
#   if defined (_STLP_USE_UNIX_IO)
sl@0
   797
sl@0
   798
    ok = (close(_M_file_id) == 0);
sl@0
   799
sl@0
   800
#   elif defined (_STLP_USE_UNIX_EMULATION_IO)
sl@0
   801
sl@0
   802
    ok = (_close(_M_file_id) == 0);
sl@0
   803
sl@0
   804
#   elif defined (_STLP_USE_STDIO_IO)
sl@0
   805
sl@0
   806
    ok = (fclose(_M_file) == 0);
sl@0
   807
sl@0
   808
#   elif defined (_STLP_USE_WIN32_IO)
sl@0
   809
sl@0
   810
    if ( _M_file_id != INVALID_HANDLE_VALUE ) {
sl@0
   811
      ok = (CloseHandle(_M_file_id) != 0);
sl@0
   812
    }
sl@0
   813
    else {
sl@0
   814
      ok = false;
sl@0
   815
    }
sl@0
   816
    
sl@0
   817
#   else
sl@0
   818
sl@0
   819
    ok = false;
sl@0
   820
sl@0
   821
#   endif /* _STLP_USE_UNIX_IO */
sl@0
   822
  }
sl@0
   823
sl@0
   824
  _M_is_open = _M_should_close = false;
sl@0
   825
  _M_openmode = 0;
sl@0
   826
  return ok;
sl@0
   827
}
sl@0
   828
sl@0
   829
sl@0
   830
# define _STLP_LF 10
sl@0
   831
# define _STLP_CR 13
sl@0
   832
# define _STLP_CTRLZ 26
sl@0
   833
sl@0
   834
// Read up to n characters into a buffer.  Return value is number of
sl@0
   835
// characters read.
sl@0
   836
_STLP_EXP_DECLSPEC ptrdiff_t _Filebuf_base::_M_read(char* buf, ptrdiff_t n) {
sl@0
   837
#   if defined (_STLP_USE_UNIX_IO)
sl@0
   838
sl@0
   839
  return read(_M_file_id, buf, n);
sl@0
   840
sl@0
   841
#   elif defined (_STLP_USE_UNIX_EMULATION_IO)
sl@0
   842
sl@0
   843
  return _read(_M_file_id, buf, n);
sl@0
   844
sl@0
   845
#   elif defined (_STLP_USE_WIN32_IO)
sl@0
   846
  
sl@0
   847
  DWORD NumberOfBytesRead;
sl@0
   848
  ReadFile(_M_file_id, (LPVOID)buf, (DWORD)n, 
sl@0
   849
           &NumberOfBytesRead, 0);
sl@0
   850
sl@0
   851
  if ((! (_M_openmode & ios_base::binary)) && NumberOfBytesRead) { 
sl@0
   852
    // translate CR-LFs to LFs in the buffer
sl@0
   853
    char * to = buf, * last = buf + NumberOfBytesRead - 1;
sl@0
   854
    char * from;
sl@0
   855
    for (from = buf; from <= last && * from != _STLP_CTRLZ; ++ from ) {
sl@0
   856
      if (* from != _STLP_CR)
sl@0
   857
        * to ++ = * from;
sl@0
   858
      else { // found CR
sl@0
   859
        if (from < last) { // not at buffer end
sl@0
   860
          if (* (from + 1) != _STLP_LF)
sl@0
   861
            * to ++ = _STLP_CR;
sl@0
   862
        }
sl@0
   863
        else { // last char is CR, peek for LF
sl@0
   864
          char peek = ' ';
sl@0
   865
          DWORD NumberOfBytesPeeked;
sl@0
   866
          ReadFile(_M_file_id, (LPVOID)&peek, 
sl@0
   867
                        1, &NumberOfBytesPeeked, 0);
sl@0
   868
          if (NumberOfBytesPeeked) {
sl@0
   869
            if (peek != _STLP_LF) { //not a <CR><LF> combination
sl@0
   870
            * to ++ = _STLP_CR;
sl@0
   871
              SetFilePointer(_M_file_id,(LONG)-1,0,SEEK_CUR);
sl@0
   872
			}
sl@0
   873
            else {
sl@0
   874
              //We ignore the complete combinaison:
sl@0
   875
              SetFilePointer(_M_file_id,(LONG)-2,0,SEEK_CUR);
sl@0
   876
			}
sl@0
   877
		  }
sl@0
   878
        }
sl@0
   879
      } // found CR
sl@0
   880
    } // for
sl@0
   881
    // seek back to TEXT end of file if hit CTRL-Z
sl@0
   882
    if (from <= last) // terminated due to CTRLZ
sl@0
   883
      SetFilePointer(_M_file_id,(LONG)((last+1) - from),0,SEEK_CUR);
sl@0
   884
    NumberOfBytesRead = to - buf;
sl@0
   885
  }
sl@0
   886
  return (ptrdiff_t)NumberOfBytesRead;
sl@0
   887
  
sl@0
   888
#   elif defined (_STLP_USE_STDIO_IO)
sl@0
   889
  
sl@0
   890
  return fread(buf, 1, n, _M_file);
sl@0
   891
  
sl@0
   892
#   else 
sl@0
   893
#    error "Port!"
sl@0
   894
# endif /* __unix */
sl@0
   895
}
sl@0
   896
sl@0
   897
// Write n characters from a buffer.  Return value: true if we managed
sl@0
   898
// to write the entire buffer, false if we didn't.  
sl@0
   899
_STLP_EXP_DECLSPEC bool _Filebuf_base::_M_write(char* buf, ptrdiff_t n) {
sl@0
   900
  
sl@0
   901
  while (true) {
sl@0
   902
    ptrdiff_t written;
sl@0
   903
    
sl@0
   904
#   if defined (_STLP_USE_UNIX_IO)
sl@0
   905
sl@0
   906
    written = write(_M_file_id, buf, n);
sl@0
   907
sl@0
   908
#   elif defined (_STLP_USE_UNIX_EMULATION_IO)
sl@0
   909
sl@0
   910
    written = _write(_M_file_id, buf, n);
sl@0
   911
sl@0
   912
#   elif defined (_STLP_USE_WIN32_IO)
sl@0
   913
sl@0
   914
    // In append mode, every write does an implicit seek to the end
sl@0
   915
    // of the file.
sl@0
   916
    if (_M_openmode & ios_base::app)
sl@0
   917
      _M_seek( 0, ios_base::end);
sl@0
   918
    
sl@0
   919
    if (_M_openmode & ios_base::binary) { 
sl@0
   920
      // binary mode
sl@0
   921
      DWORD NumberOfBytesWritten;
sl@0
   922
      WriteFile(_M_file_id, (LPVOID)buf, (DWORD)n, 
sl@0
   923
                &NumberOfBytesWritten, 0);
sl@0
   924
      written = (ptrdiff_t)NumberOfBytesWritten;
sl@0
   925
    }
sl@0
   926
    else {
sl@0
   927
      char textbuf[_TEXTBUF_SIZE + 1]; // extra 1 in case LF at end
sl@0
   928
      char * nextblock = buf, * ptrtextbuf = textbuf;
sl@0
   929
      char * endtextbuf = textbuf + _TEXTBUF_SIZE;
sl@0
   930
      char * endblock = buf + n;
sl@0
   931
      ptrdiff_t nextblocksize = (min) (n, (ptrdiff_t)_TEXTBUF_SIZE);
sl@0
   932
      char * nextlf;
sl@0
   933
      
sl@0
   934
      while ( (nextblocksize > 0) && 
sl@0
   935
              (nextlf = (char *)memchr(nextblock, _STLP_LF, nextblocksize)) != 0) {
sl@0
   936
        ptrdiff_t linelength = nextlf - nextblock;
sl@0
   937
        memcpy(ptrtextbuf, nextblock, linelength);
sl@0
   938
        ptrtextbuf += linelength;
sl@0
   939
        nextblock += (linelength + 1);
sl@0
   940
        * ptrtextbuf ++ = _STLP_CR;
sl@0
   941
        * ptrtextbuf ++ = _STLP_LF;
sl@0
   942
        nextblocksize = (min) (ptrdiff_t(endblock - nextblock), 
sl@0
   943
                                             (max) (ptrdiff_t(0), ptrdiff_t(endtextbuf - ptrtextbuf)));
sl@0
   944
      }
sl@0
   945
      // write out what's left, > condition is here since for LF at the end , 
sl@0
   946
      // endtextbuf may get < ptrtextbuf ...
sl@0
   947
      if (nextblocksize > 0) {
sl@0
   948
        memcpy(ptrtextbuf, nextblock, nextblocksize);
sl@0
   949
        ptrtextbuf += nextblocksize;
sl@0
   950
        nextblock += nextblocksize;
sl@0
   951
      }
sl@0
   952
      // now write out the translated buffer
sl@0
   953
      char * writetextbuf = textbuf;
sl@0
   954
      for (ptrdiff_t NumberOfBytesToWrite = ptrtextbuf - textbuf; 
sl@0
   955
           NumberOfBytesToWrite;) {
sl@0
   956
        DWORD NumberOfBytesWritten;
sl@0
   957
        WriteFile((HANDLE)_M_file_id, (LPVOID)writetextbuf, 
sl@0
   958
                  NumberOfBytesToWrite, &NumberOfBytesWritten, 0);
sl@0
   959
        if (NumberOfBytesWritten == NumberOfBytesToWrite)
sl@0
   960
          break;
sl@0
   961
        if (!NumberOfBytesWritten) // write shortfall
sl@0
   962
          return false;
sl@0
   963
        writetextbuf += NumberOfBytesWritten;
sl@0
   964
        NumberOfBytesToWrite -=	NumberOfBytesWritten;
sl@0
   965
      }
sl@0
   966
      // count non-translated characters
sl@0
   967
      written = (nextblock - buf);
sl@0
   968
    }
sl@0
   969
sl@0
   970
#   elif defined (_STLP_USE_STDIO_IO)
sl@0
   971
sl@0
   972
    written = fwrite(buf, 1, n, _M_file);
sl@0
   973
sl@0
   974
#   else 
sl@0
   975
#    error "Port!"
sl@0
   976
#   endif /* __unix */ 
sl@0
   977
    
sl@0
   978
    if (n == written)
sl@0
   979
      return true;
sl@0
   980
    else if (written > 0 && written < n) {
sl@0
   981
      n -= written;
sl@0
   982
      buf += written;
sl@0
   983
    }
sl@0
   984
    else
sl@0
   985
      return false;
sl@0
   986
  }
sl@0
   987
}
sl@0
   988
sl@0
   989
sl@0
   990
#ifdef _STLP_USE_WIN32_IO
sl@0
   991
# define STL_SEEK_SET FILE_BEGIN
sl@0
   992
# define STL_SEEK_CUR FILE_CURRENT
sl@0
   993
# define STL_SEEK_END FILE_END
sl@0
   994
#else
sl@0
   995
# define STL_SEEK_SET SEEK_SET
sl@0
   996
# define STL_SEEK_CUR SEEK_CUR
sl@0
   997
# define STL_SEEK_END SEEK_END
sl@0
   998
#endif
sl@0
   999
sl@0
  1000
// Wrapper for lseek or the like.
sl@0
  1001
_STLP_EXP_DECLSPEC streamoff _Filebuf_base::_M_seek(streamoff offset, ios_base::seekdir dir)
sl@0
  1002
{
sl@0
  1003
  streamoff result = -1;
sl@0
  1004
sl@0
  1005
  int whence;
sl@0
  1006
sl@0
  1007
  switch(dir) {
sl@0
  1008
  case ios_base::beg:
sl@0
  1009
    if (offset < 0 /* || offset > _M_file_size() */ )
sl@0
  1010
      return streamoff(-1);
sl@0
  1011
    whence = STL_SEEK_SET;
sl@0
  1012
    break;
sl@0
  1013
  case ios_base::cur:
sl@0
  1014
    whence = STL_SEEK_CUR;
sl@0
  1015
    break;
sl@0
  1016
  case ios_base::end:
sl@0
  1017
    if (/* offset > 0 || */  -offset > _M_file_size() )
sl@0
  1018
      return streamoff(-1);
sl@0
  1019
    whence = STL_SEEK_END;
sl@0
  1020
    break;
sl@0
  1021
  default:
sl@0
  1022
    return streamoff(-1);
sl@0
  1023
  }
sl@0
  1024
sl@0
  1025
#if defined (_STLP_USE_UNIX_IO) || defined (_STLP_USE_UNIX_EMULATION_IO)
sl@0
  1026
sl@0
  1027
  result = LSEEK(_M_file_id, offset, whence);
sl@0
  1028
sl@0
  1029
#elif defined (_STLP_USE_STDIO_IO)
sl@0
  1030
sl@0
  1031
  result = fseek(_M_file, offset, whence);
sl@0
  1032
sl@0
  1033
#elif defined (_STLP_USE_WIN32_IO)
sl@0
  1034
sl@0
  1035
  LARGE_INTEGER li;
sl@0
  1036
  li.QuadPart = offset;
sl@0
  1037
  li.LowPart = SetFilePointer(_M_file_id, li.LowPart, &li.HighPart, whence);
sl@0
  1038
  if (li.LowPart == 0xFFFFFFFF && GetLastError() != NO_ERROR)
sl@0
  1039
    result = -1; // Error
sl@0
  1040
  else
sl@0
  1041
    result = li.QuadPart;
sl@0
  1042
sl@0
  1043
#else
sl@0
  1044
#   error "Port!"
sl@0
  1045
#endif
sl@0
  1046
sl@0
  1047
  return result;
sl@0
  1048
}
sl@0
  1049
sl@0
  1050
sl@0
  1051
// Attempts to memory-map len bytes of the current file, starting
sl@0
  1052
// at position offset.  Precondition: offset is a multiple of the
sl@0
  1053
// page size.  Postcondition: return value is a null pointer if the
sl@0
  1054
// memory mapping failed.  Otherwise the return value is a pointer to 
sl@0
  1055
// the memory-mapped file and the file position is set to offset.
sl@0
  1056
_STLP_EXP_DECLSPEC void* _Filebuf_base::_M_mmap(streamoff offset, streamoff len) {
sl@0
  1057
  void* base;
sl@0
  1058
#if defined (_STLP_UNIX) && !defined(__DJGPP) && !defined(_CRAY)
sl@0
  1059
  base = MMAP(0, len, PROT_READ, MAP_PRIVATE, _M_file_id, offset);
sl@0
  1060
  if (base != (void*)MAP_FAILED) {
sl@0
  1061
    if (LSEEK(_M_file_id, offset + len, SEEK_SET) < 0) {
sl@0
  1062
      this->_M_unmap(base, len);
sl@0
  1063
      base = 0;
sl@0
  1064
    }
sl@0
  1065
  } else
sl@0
  1066
    base =0;
sl@0
  1067
sl@0
  1068
#elif defined (_STLP_USE_WIN32_IO)
sl@0
  1069
sl@0
  1070
  _M_view_id = CreateFileMapping(_M_file_id, (PSECURITY_ATTRIBUTES)0 ,
sl@0
  1071
				 PAGE_READONLY, 0 /* len >> 32 */ , 
sl@0
  1072
				 0 /* len & 0xFFFFFFFF */ ,             // low-order DWORD of size
sl@0
  1073
				 0);
sl@0
  1074
sl@0
  1075
  if (_M_view_id) {
sl@0
  1076
# if 0
sl@0
  1077
    printf("view %x created from file %x, error = %d, size = %d, map_offset = %d map_len = %d\n", 
sl@0
  1078
	   _M_view_id, _M_file_id, GetLastError(), 
sl@0
  1079
	   (int)cur_filesize, ULL(offset) & 0xffffffff, len);
sl@0
  1080
# endif
sl@0
  1081
    
sl@0
  1082
    base = MapViewOfFile(_M_view_id, FILE_MAP_READ, ULL(offset)>>32, 
sl@0
  1083
			 ULL(offset) & 0xffffffff, len);
sl@0
  1084
    // check if mapping succeded and is usable
sl@0
  1085
    if (base ==0  || _M_seek(offset+len, ios_base::beg) < 0) {
sl@0
  1086
      this->_M_unmap(base, len);
sl@0
  1087
      base = 0;
sl@0
  1088
    }
sl@0
  1089
  } else
sl@0
  1090
    base = 0;
sl@0
  1091
#else
sl@0
  1092
  (void)len;		//*TY 02/26/2000 - unused variables
sl@0
  1093
  (void)offset;		//*TY 02/26/2000 - 
sl@0
  1094
  base = 0;
sl@0
  1095
#endif
sl@0
  1096
  return base;  
sl@0
  1097
}
sl@0
  1098
sl@0
  1099
_STLP_EXP_DECLSPEC void _Filebuf_base::_M_unmap(void* base, streamoff len) {
sl@0
  1100
  // precondition : there is a valid mapping at the moment
sl@0
  1101
#if defined (_STLP_UNIX)  && !defined(__DJGPP) && !defined(_CRAY)
sl@0
  1102
  munmap((char*)base, len);
sl@0
  1103
#elif defined (_STLP_USE_WIN32_IO)
sl@0
  1104
  if ( base != NULL )
sl@0
  1105
    UnmapViewOfFile(base);
sl@0
  1106
  // destroy view handle as well
sl@0
  1107
  if ( _M_view_id != NULL )
sl@0
  1108
    CloseHandle(_M_view_id);
sl@0
  1109
  _M_view_id = 0;
sl@0
  1110
  base = 0;
sl@0
  1111
#else
sl@0
  1112
  (void)len;		//*TY 02/26/2000 - unused variables
sl@0
  1113
  (void)base;		//*TY 02/26/2000 - 
sl@0
  1114
#endif
sl@0
  1115
}
sl@0
  1116
sl@0
  1117
_STLP_EXP_DECLSPEC int _STLP_CALL
sl@0
  1118
_Underflow<char, char_traits<char> >::_M_doit (basic_filebuf<char, char_traits<char> >* __this)  
sl@0
  1119
{
sl@0
  1120
  if (!__this->_M_in_input_mode) {
sl@0
  1121
    if (!__this->_M_switch_to_input_mode())
sl@0
  1122
      return traits_type::eof();
sl@0
  1123
  }
sl@0
  1124
  
sl@0
  1125
  else if (__this->_M_in_putback_mode) {
sl@0
  1126
    __this->_M_exit_putback_mode();
sl@0
  1127
    if (__this->gptr() != __this->egptr()) {
sl@0
  1128
      int_type __c = traits_type::to_int_type(*__this->gptr());
sl@0
  1129
      return __c;
sl@0
  1130
    }
sl@0
  1131
  }
sl@0
  1132
    
sl@0
  1133
  // If it's a disk file, and if the internal and external character
sl@0
  1134
  // sequences are guaranteed to be identical, then try to use memory
sl@0
  1135
  // mapped I/O.  Otherwise, revert to ordinary read.
sl@0
  1136
  if (__this->_M_base.__regular_file() 
sl@0
  1137
      && __this->_M_always_noconv
sl@0
  1138
      && __this->_M_base._M_in_binary_mode()) {
sl@0
  1139
    // If we've mmapped part of the file already, then unmap it.
sl@0
  1140
    if (__this->_M_mmap_base)
sl@0
  1141
      __this->_M_base._M_unmap(__this->_M_mmap_base, __this->_M_mmap_len);
sl@0
  1142
    __this->_M_mmap_base = 0;
sl@0
  1143
    __this->_M_mmap_len = 0;
sl@0
  1144
      
sl@0
  1145
    // Determine the position where we start mapping.  It has to be
sl@0
  1146
    // a multiple of the page size.
sl@0
  1147
    streamoff __cur = __this->_M_base._M_seek(0, ios_base::cur);
sl@0
  1148
    streamoff __size = __this->_M_base._M_file_size();
sl@0
  1149
    if (__size > 0 && __cur >= 0 && __cur < __size) {
sl@0
  1150
      streamoff __offset    = (__cur / __this->_M_base.__page_size())
sl@0
  1151
	* __this->_M_base.__page_size();
sl@0
  1152
      streamoff __remainder = __cur - __offset;
sl@0
  1153
sl@0
  1154
      __this->_M_mmap_len = __size - __offset;
sl@0
  1155
sl@0
  1156
      if (__this->_M_mmap_len > MMAP_CHUNK)
sl@0
  1157
	__this->_M_mmap_len = MMAP_CHUNK;
sl@0
  1158
sl@0
  1159
      if ((__this->_M_mmap_base =
sl@0
  1160
	   __this->_M_base._M_mmap(__offset, __this->_M_mmap_len)) != 0) {
sl@0
  1161
	__this->setg((char*) __this->_M_mmap_base,
sl@0
  1162
		     (char*) __this->_M_mmap_base + __remainder,
sl@0
  1163
		     (char*) __this->_M_mmap_base + __this->_M_mmap_len);
sl@0
  1164
	return traits_type::to_int_type(*__this->gptr());
sl@0
  1165
      } 
sl@0
  1166
    } else /* size > 0 ... */ {
sl@0
  1167
      // There is nothing to map. We unmapped the file above, now zap pointers.
sl@0
  1168
      __this->_M_mmap_base = 0;
sl@0
  1169
      __this->_M_mmap_len = 0;
sl@0
  1170
    }
sl@0
  1171
  }
sl@0
  1172
    
sl@0
  1173
  return __this->_M_underflow_aux();
sl@0
  1174
}
sl@0
  1175
sl@0
  1176
sl@0
  1177
//----------------------------------------------------------------------
sl@0
  1178
// Force instantiation of filebuf and fstream classes.
sl@0
  1179
#if !defined(_STLP_NO_FORCE_INSTANTIATE)
sl@0
  1180
sl@0
  1181
template class basic_filebuf<char, char_traits<char> >;
sl@0
  1182
template class basic_ifstream<char, char_traits<char> >;
sl@0
  1183
template class basic_ofstream<char, char_traits<char> >;
sl@0
  1184
template class basic_fstream<char, char_traits<char> >;
sl@0
  1185
sl@0
  1186
#  ifndef _STLP_NO_WCHAR_T
sl@0
  1187
template class _Underflow<wchar_t, char_traits<wchar_t> >;
sl@0
  1188
template class basic_filebuf<wchar_t, char_traits<wchar_t> >;
sl@0
  1189
template class basic_ifstream<wchar_t, char_traits<wchar_t> >;
sl@0
  1190
template class basic_ofstream<wchar_t, char_traits<wchar_t> >;
sl@0
  1191
template class basic_fstream<wchar_t, char_traits<wchar_t> >;
sl@0
  1192
#  endif /* _STLP_NO_WCHAR_T */
sl@0
  1193
sl@0
  1194
#endif
sl@0
  1195
sl@0
  1196
_STLP_END_NAMESPACE
sl@0
  1197
sl@0
  1198
// Local Variables:
sl@0
  1199
// mode:C++
sl@0
  1200
// End:
sl@0
  1201