228 lines
6.0 KiB
C
228 lines
6.0 KiB
C
|
/****************************************************************************
|
||
|
*
|
||
|
* Open Watcom Project
|
||
|
*
|
||
|
* Portions Copyright (c) 1983-2002 Sybase, Inc. All Rights Reserved.
|
||
|
*
|
||
|
* ========================================================================
|
||
|
*
|
||
|
* This file contains Original Code and/or Modifications of Original
|
||
|
* Code as defined in and that are subject to the Sybase Open Watcom
|
||
|
* Public License version 1.0 (the 'License'). You may not use this file
|
||
|
* except in compliance with the License. BY USING THIS FILE YOU AGREE TO
|
||
|
* ALL TERMS AND CONDITIONS OF THE LICENSE. A copy of the License is
|
||
|
* provided with the Original Code and Modifications, and is also
|
||
|
* available at www.sybase.com/developer/opensource.
|
||
|
*
|
||
|
* The Original Code and all software distributed under the License are
|
||
|
* distributed on an 'AS IS' basis, WITHOUT WARRANTY OF ANY KIND, EITHER
|
||
|
* EXPRESS OR IMPLIED, AND SYBASE AND ALL CONTRIBUTORS HEREBY DISCLAIM
|
||
|
* ALL SUCH WARRANTIES, INCLUDING WITHOUT LIMITATION, ANY WARRANTIES OF
|
||
|
* MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE, QUIET ENJOYMENT OR
|
||
|
* NON-INFRINGEMENT. Please see the License for the specific language
|
||
|
* governing rights and limitations under the License.
|
||
|
*
|
||
|
* ========================================================================
|
||
|
*
|
||
|
* Description: Platform independent fgetc() implementation.
|
||
|
*
|
||
|
****************************************************************************/
|
||
|
|
||
|
|
||
|
#include "variety.h"
|
||
|
#include "widechar.h"
|
||
|
#include <stdio.h>
|
||
|
#include <unistd.h>
|
||
|
#ifndef __UNIX__
|
||
|
#include <conio.h>
|
||
|
#endif
|
||
|
#include "fileacc.h"
|
||
|
#include <errno.h>
|
||
|
#include "rtdata.h"
|
||
|
#include "seterrno.h"
|
||
|
#ifdef __WIDECHAR__
|
||
|
#include <mbstring.h>
|
||
|
#include <wchar.h>
|
||
|
#endif
|
||
|
#include "qread.h"
|
||
|
#include "orient.h"
|
||
|
#include "flush.h"
|
||
|
#include "streamio.h"
|
||
|
|
||
|
|
||
|
#define DOS_EOF_CHAR 0x1a
|
||
|
|
||
|
#ifdef __WIDECHAR__
|
||
|
#define CHARMASK 0xffff
|
||
|
#else
|
||
|
#define CHARMASK 0xff
|
||
|
#endif
|
||
|
|
||
|
int __F_NAME(__fill_buffer,__wfill_buffer)( FILE *fp )
|
||
|
{
|
||
|
if( _FP_BASE(fp) == NULL ) {
|
||
|
__ioalloc( fp );
|
||
|
}
|
||
|
if( fp->_flag & _ISTTY ) { /* 20-aug-90 */
|
||
|
if( fp->_flag & (_IONBF | _IOLBF) ) {
|
||
|
__flushall( _ISTTY ); /* flush all TTY output */
|
||
|
}
|
||
|
}
|
||
|
fp->_flag &= ~_UNGET; /* 10-mar-90 */
|
||
|
fp->_ptr = _FP_BASE(fp);
|
||
|
#ifdef __UNIX__
|
||
|
fp->_cnt = __qread( fileno( fp ), fp->_ptr,
|
||
|
(fp->_flag & _IONBF) ? CHARSIZE : fp->_bufsize );
|
||
|
#else
|
||
|
if(( fp->_flag & (_IONBF | _ISTTY)) == (_IONBF | _ISTTY) &&
|
||
|
( fileno( fp ) == STDIN_FILENO ))
|
||
|
{
|
||
|
int c; /* JBS 31-may-91 */
|
||
|
|
||
|
fp->_cnt = 0;
|
||
|
// c = getche();
|
||
|
if( c != EOF ) {
|
||
|
*(CHAR_TYPE *)fp->_ptr = c;
|
||
|
fp->_cnt = CHARSIZE;
|
||
|
}
|
||
|
} else {
|
||
|
fp->_cnt = __qread( fileno( fp ), fp->_ptr,
|
||
|
(fp->_flag & _IONBF) ? CHARSIZE : fp->_bufsize );
|
||
|
}
|
||
|
#endif
|
||
|
if( fp->_cnt <= 0 ) {
|
||
|
if( fp->_cnt == 0 ) {
|
||
|
fp->_flag |= _EOF;
|
||
|
} else {
|
||
|
fp->_flag |= _SFERR;
|
||
|
fp->_cnt = 0;
|
||
|
}
|
||
|
}
|
||
|
return( fp->_cnt );
|
||
|
}
|
||
|
|
||
|
int __F_NAME(__filbuf,__wfilbuf)( FILE *fp )
|
||
|
{
|
||
|
if( __F_NAME(__fill_buffer,__wfill_buffer)( fp ) == 0 ) {
|
||
|
return( EOF );
|
||
|
}
|
||
|
else {
|
||
|
fp->_cnt -= CHARSIZE;
|
||
|
fp->_ptr += CHARSIZE;
|
||
|
return( *(CHAR_TYPE *)(fp->_ptr - CHARSIZE) & CHARMASK );
|
||
|
}
|
||
|
}
|
||
|
|
||
|
|
||
|
#ifndef __WIDECHAR__
|
||
|
|
||
|
_WCRTLINK int fgetc( FILE *fp )
|
||
|
{
|
||
|
int c;
|
||
|
|
||
|
_ValidFile( fp, EOF );
|
||
|
_AccessFile( fp );
|
||
|
|
||
|
/*** Deal with stream orientation ***/
|
||
|
ORIENT_STREAM(fp,EOF);
|
||
|
|
||
|
if( (fp->_flag & _READ) == 0 ) {
|
||
|
__set_errno( EBADF );
|
||
|
fp->_flag |= _SFERR;
|
||
|
c = EOF;
|
||
|
} else {
|
||
|
fp->_cnt--;
|
||
|
// it is important that this remain a relative comparison
|
||
|
// to ensure that the getc() macro works properly
|
||
|
if( fp->_cnt < 0 ) {
|
||
|
c = __F_NAME(__filbuf,__wfilbuf)( fp );
|
||
|
} else {
|
||
|
c = *(char *)fp->_ptr;
|
||
|
fp->_ptr++;
|
||
|
}
|
||
|
}
|
||
|
#ifndef __UNIX__
|
||
|
if( !(fp->_flag & _BINARY) ) {
|
||
|
if( c == '\r' ) {
|
||
|
fp->_cnt--;
|
||
|
// it is important that this remain a relative comparison
|
||
|
// to ensure that the getc() macro works properly
|
||
|
if( fp->_cnt < 0 ) {
|
||
|
c = __F_NAME(__filbuf,__wfilbuf)( fp );
|
||
|
} else {
|
||
|
c = *(CHAR_TYPE*)fp->_ptr & CHARMASK;
|
||
|
fp->_ptr += CHARSIZE;
|
||
|
}
|
||
|
}
|
||
|
if( c == DOS_EOF_CHAR ) {
|
||
|
fp->_flag |= _EOF;
|
||
|
c = EOF;
|
||
|
}
|
||
|
}
|
||
|
#endif
|
||
|
_ReleaseFile( fp );
|
||
|
return( c );
|
||
|
}
|
||
|
|
||
|
#else
|
||
|
|
||
|
static int __read_wide_char( FILE *fp, wchar_t *wc )
|
||
|
/**************************************************/
|
||
|
{
|
||
|
if( fp->_flag & _BINARY ) {
|
||
|
/*** Read a wide character ***/
|
||
|
return( fread( wc, sizeof( wchar_t ), 1, fp ) );
|
||
|
} else {
|
||
|
char mbc[MB_CUR_MAX];
|
||
|
wchar_t wcTemp;
|
||
|
int rc;
|
||
|
|
||
|
/*** Read the multibyte character ***/
|
||
|
if( !fread( &mbc[0], 1, 1, fp ) )
|
||
|
return( 0 );
|
||
|
|
||
|
if( _ismbblead( mbc[0] ) ) {
|
||
|
if( !fread( &mbc[1], 1, 1, fp ) )
|
||
|
return( 0 );
|
||
|
}
|
||
|
|
||
|
/*** Convert it to wide form ***/
|
||
|
rc = mbtowc( &wcTemp, mbc, MB_CUR_MAX );
|
||
|
if( rc >= 0 ) {
|
||
|
*wc = wcTemp;
|
||
|
return( 1 );
|
||
|
} else {
|
||
|
__set_errno( EILSEQ );
|
||
|
return( 0 );
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
_WCRTLINK wint_t fgetwc( FILE *fp )
|
||
|
{
|
||
|
wchar_t c;
|
||
|
|
||
|
_ValidFile( fp, WEOF );
|
||
|
_AccessFile( fp );
|
||
|
|
||
|
/*** Deal with stream orientation ***/
|
||
|
ORIENT_STREAM(fp,WEOF);
|
||
|
|
||
|
/*** Read the character ***/
|
||
|
if( !__read_wide_char( fp, &c ) ) {
|
||
|
_ReleaseFile( fp );
|
||
|
return( WEOF );
|
||
|
}
|
||
|
if( !(fp->_flag & _BINARY) && (c == L'\r') ) {
|
||
|
if( !__read_wide_char( fp, &c ) ) {
|
||
|
_ReleaseFile( fp );
|
||
|
return( WEOF );
|
||
|
}
|
||
|
}
|
||
|
|
||
|
_ReleaseFile( fp );
|
||
|
return( (wint_t)c );
|
||
|
}
|
||
|
|
||
|
#endif
|