2020-09-30 17:12:29 +02:00

374 lines
8.9 KiB
C

/***
*write.c - write to a file handle
*
* Copyright (c) 1989-1995, Microsoft Corporation. All rights reserved.
*
*Purpose:
* defines _write() - write to a file handle
*
*Revision History:
* 06-14-89 PHG Module created, based on asm version
* 03-13-90 GJF Made calling type _CALLTYPE2 (for now), added #include
* <cruntime.h>, fixed compiler warnings and fixed the
* copyright. Also, cleaned up the formatting a bit.
* 04-03-90 GJF Now _CALLTYPE1.
* 07-24-90 SBM Removed '32' from API names
* 08-14-90 SBM Compiles cleanly with -W3
* 10-01-90 GJF New-style function declarators.
* 12-04-90 GJF Appended Win32 version onto source with #ifdef-s.
* Should come back latter and do a better merge.
* 12-04-90 SRW Changed to include <oscalls.h> instead of <doscalls.h>
* 12-06-90 SRW Changed to use _osfile and _osfhnd instead of _osfinfo
* 12-28-90 SRW Added _CRUISER_ conditional around check_stack pragma
* 12-28-90 SRW Added cast of void * to char * for Mips C Compiler
* 01-17-91 GJF ANSI naming.
* 02-25-91 MHL Adapt to ReadFile/WriteFile changes (_WIN32_)
* 04-09-91 PNT Added _MAC_ conditional
* 07-18-91 GJF Removed unreferenced local variable from _write_lk
* routine [_WIN32_].
* 10-24-91 GJF Added LPDWORD casts to make MIPS compiler happy.
* ASSUMES THAT sizeof(int) == sizeof(DWORD).
* 02-13-92 GJF Replaced _nfile by _nhandle for Win32.
* 02-15-92 GJF Increased BUF_SIZE and simplified LF translation code
* for Win32.
* 04-06-93 SKS Replace _CRTAPI* with __cdecl
* 09-06-94 CFW Remove Cruiser support.
* 09-06-94 CFW Replace MTHREAD with _MT.
* 12-03-94 SKS Clean up OS/2 references
* 01-04-95 GJF _WIN32_ -> _WIN32
* 02-15-95 GJF Appended Mac version of source file (somewhat cleaned
* up), with appropriate #ifdef-s.
* 06-12-95 GJF Changed _osfile[] and _osfhnd[] to _osfile() and
* _osfhnd(), which reference __pioinfo[].
* 06-27-95 GJF Added check that the file handle is open.
*
*******************************************************************************/
#ifdef _WIN32
#include <cruntime.h>
#include <oscalls.h>
#include <io.h>
#include <errno.h>
#include <msdos.h>
#include <mtdll.h>
#include <stdlib.h>
#include <string.h>
#include <internal.h>
#ifdef _WIN32
#define BUF_SIZE 1025 /* size of LF translation buffer */
#else /* ndef _WIN32 */
#define BUF_SIZE 513 /* size of LF translation buffer, sector size+1 is ok */
#endif /* _WIN32 */
#define LF '\n' /* line feed */
#define CR '\r' /* carriage return */
#define CTRLZ 26 /* ctrl-z */
/***
*int _write(fh, buf, cnt) - write bytes to a file handle
*
*Purpose:
* Writes count bytes from the buffer to the handle specified.
* If the file was opened in text mode, each LF is translated to
* CR-LF. This does not affect the return value. In text
* mode ^Z indicates end of file.
*
* Multi-thread notes:
* (1) _write() - Locks/unlocks file handle
* _write_lk() - Does NOT lock/unlock file handle
*
*Entry:
* int fh - file handle to write to
* char *buf - buffer to write from
* unsigned int cnt - number of bytes to write
*
*Exit:
* returns number of bytes actually written.
* This may be less than cnt, for example, if out of disk space.
* returns -1 (and set errno) if fails.
*
*Exceptions:
*
*******************************************************************************/
#ifdef _MT
/* define normal version that locks/unlocks, validates fh */
int __cdecl _write (
int fh,
const void *buf,
unsigned cnt
)
{
int r; /* return value */
/* validate handle */
if ( ((unsigned)fh >= (unsigned)_nhandle) ||
!(_osfile(fh) & FOPEN) )
{
/* out of range -- return error */
errno = EBADF;
_doserrno = 0; /* not o.s. error */
return -1;
}
_lock_fh(fh); /* lock file */
r = _write_lk(fh, buf, cnt); /* write bytes */
_unlock_fh(fh); /* unlock file */
return r;
}
/* now define version that doesn't lock/unlock, validate fh */
int __cdecl _write_lk (
int fh,
const void *buf,
unsigned cnt
)
{
int lfcount; /* count of line feeds */
int charcount; /* count of chars written so far */
int written; /* count of chars written on this write */
ULONG dosretval; /* o.s. return value */
char ch; /* current character */
char *p, *q; /* pointers into buf and lfbuf resp. */
char lfbuf[BUF_SIZE]; /* lf translation buffer */
#else
/* now define normal version */
int __cdecl _write (
int fh,
const void *buf,
unsigned cnt
)
{
int lfcount; /* count of line feeds */
int charcount; /* count of chars written so far */
int written; /* count of chars written on this write */
ULONG dosretval; /* o.s. return value */
char ch; /* current character */
char *p, *q; /* pointers into buf and lfbuf resp. */
char lfbuf[BUF_SIZE]; /* lf translation buffer */
/* validate handle */
if ( ((unsigned)fh >= (unsigned)_nhandle) ||
!(_osfile(fh) & FOPEN) )
{
/* out of range -- return error */
errno = EBADF;
_doserrno = 0; /* not o.s. error */
return -1;
}
#endif
lfcount = charcount = 0; /* nothing written yet */
if (cnt == 0)
return 0; /* nothing to do */
if (_osfile(fh) & FAPPEND) {
/* appending - seek to end of file; ignore error, because maybe
file doesn't allow seeking */
(void)_lseek_lk(fh, 0, FILE_END);
}
/* check for text mode with LF's in the buffer */
if ( _osfile(fh) & FTEXT ) {
/* text mode, translate LF's to CR/LF's on output */
p = (char *)buf; /* start at beginning of buffer */
dosretval = 0; /* no OS error yet */
while ( (unsigned)(p - (char *)buf) < cnt ) {
q = lfbuf; /* start at beginning of lfbuf */
/* fill the lf buf, except maybe last char */
while ( q - lfbuf < BUF_SIZE - 1 &&
(unsigned)(p - (char *)buf) < cnt ) {
ch = *p++;
if ( ch == LF ) {
++lfcount;
*q++ = CR;
}
*q++ = ch;
}
/* write the lf buf and update total */
if ( WriteFile( (HANDLE)_osfhnd(fh),
lfbuf,
q - lfbuf,
(LPDWORD)&written,
NULL) )
{
charcount += written;
if (written < q - lfbuf)
break;
}
else {
dosretval = GetLastError();
break;
}
}
}
else {
/* binary mode, no translation */
if ( WriteFile( (HANDLE)_osfhnd(fh),
(LPVOID)buf,
cnt,
(LPDWORD)&written,
NULL) )
{
dosretval = 0;
charcount = written;
}
else
dosretval = GetLastError();
}
if (charcount == 0) {
/* If nothing was written, first check if an o.s. error,
otherwise we return -1 and set errno to ENOSPC,
unless a device and first char was CTRL-Z */
if (dosretval != 0) {
/* o.s. error happened, map error */
if (dosretval == ERROR_ACCESS_DENIED) {
/* wrong read/write mode should return EBADF, not
EACCES */
errno = EBADF;
_doserrno = dosretval;
}
else
_dosmaperr(dosretval);
return -1;
}
else if ((_osfile(fh) & FDEV) && *(char *)buf == CTRLZ)
return 0;
else {
errno = ENOSPC;
_doserrno = 0; /* no o.s. error */
return -1;
}
}
else
/* return adjusted bytes written */
return charcount - lfcount;
}
#else /* ndef _WIN32 */
#if defined(_M_MPPC) || defined(_M_M68K)
#include <cruntime.h>
#include <errno.h>
#include <msdos.h>
#include <stdlib.h>
#include <internal.h>
#include <macos\files.h>
#include <macos\errors.h>
#include <mpw.h>
/***
*int _write(fh, buf, cnt) - write bytes to a file handle
*
*Purpose:
* Writes count bytes from the buffer to the handle specified.
*
*Entry:
* int fh - file handle to write to
* char *buf - buffer to write from
* unsigned int cnt - number of bytes to write
*
*Exit:
* returns number of bytes actually written.
* This may be less than cnt, for example, if out of disk space.
* returns -1 (and set errno) if fails.
*
*Exceptions:
*
*******************************************************************************/
int __cdecl _write (
int fh,
const void *buf,
unsigned cnt
)
{
OSErr osErr;
int cbReturn;
/* validate handle */
if ((unsigned)fh >= (unsigned)_nfile || !(_osfile[fh] & FOPEN) ||
_osfile[fh] & FRDONLY)
{
/* out of range -- return error */
errno = EBADF;
_macerrno = 0;
return -1;
}
if (_osfile[fh] & FDEV)
{
MPWFILE *pparm;
/* MPW console */
pparm = (MPWFILE *)_osfhnd[fh];
pparm->count = cnt;
(const void *)(pparm->pBuff) = buf;
osErr = (*((pparm->pDevice)->write))(pparm);
if (osErr)
{
osErr = pparm->err;
}
else
{
cbReturn = cnt - pparm->count;
}
}
else
{
ParamBlockRec parm;
/* File */
parm.ioParam.ioRefNum = _osfhnd[fh];
(const void *)parm.ioParam.ioBuffer = buf;
parm.ioParam.ioReqCount = cnt;
parm.ioParam.ioPosOffset = 0;
if ( _osfile[fh] & FAPPEND)
{
parm.ioParam.ioPosMode = fsFromLEOF;
}
else
{
parm.ioParam.ioPosMode = fsAtMark;
}
osErr = PBWriteSync(&parm);
cbReturn = parm.ioParam.ioActCount;
}
if (!osErr)
{
return cbReturn;
}
else
{
_dosmaperr(osErr);
return -1;
}
}
#endif /* defined(_M_MPPC) || defined(_M_M68K) */
#endif /* _WIN32 */