linkio.c
来自「开放源码的编译器open watcom 1.6.0版的源代码」· C语言 代码 · 共 470 行
C
470 行
/****************************************************************************
*
* 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: Linker binary i/o interface routines (DOS flavour).
*
****************************************************************************/
#include <stdio.h>
#include <dos.h>
#include <string.h>
#include <stdlib.h>
#include <process.h>
#include <conio.h>
#include <unistd.h>
#include "linkstd.h"
#include "msg.h"
#include "alloc.h"
#include "wlnkmsg.h"
#include "wressetr.h"
#include "tinyio.h"
#include "objio.h"
#include "fileio.h"
typedef enum {
NOT_HIT,
BREAK_DETECTED,
BREAK_HANDLED
} break_status;
static bool BreakCond;
static break_status CaughtBreak = NOT_HIT;
static int OpenFiles; // the number of open files
static unsigned LastResult;
static int AuxFilesClosed = FALSE;
static char Rc_Buffer[RESOURCE_MAX_SIZE];
#define TOOMANY 4
#define ACCESS_DENIED 5
extern void LnkFilesInit( void )
/******************************/
// the linker doesn't use stdaux or stdprn, so close these.
{
CaughtBreak = NOT_HIT;
if( !AuxFilesClosed ) {
OpenFiles = 2; // will be 0 when done closing stdaux & stdprn.
QClose( STDAUX_HANDLE, "stdaux" );
QClose( STDPRN_HANDLE, "stdprn" );
AuxFilesClosed = TRUE;
OpenFiles = 0;
}
}
extern void PrintIOError( unsigned msg, char *types, char *name )
/***************************************************************/
{
char rc_buff[RESOURCE_MAX_SIZE];
Msg_Get( MSG_IOERRLIST_0 + LastResult, rc_buff );
LnkMsg( msg, types, name, rc_buff );
}
extern void CheckBreak( void )
/****************************/
{
if( CaughtBreak == BREAK_DETECTED ) {
CaughtBreak = BREAK_HANDLED;
LnkMsg( FTL+MSG_BREAK_HIT, NULL ); /* suicides */
}
}
static tiny_ret_t DoOpen( char *name, bool create, unsigned mode )
/****************************************************************/
{
tiny_ret_t h;
CheckBreak();
for( ;; ) {
if( OpenFiles >= MAX_OPEN_FILES ) CleanCachedHandles();
if( create ) {
h = TinyCreate( name, mode );
} else {
h = TinyOpen( name, mode );
}
if( TINY_OK( h ) ) {
OpenFiles++;
break;
}
if( TINY_INFO( h ) != TOOMANY ) break;
if( !CleanCachedHandles() ) break;
}
return( h );
}
static char *QErrMsg( unsigned status )
/*************************************/
{
Msg_Get( MSG_IOERRLIST_0 + TINY_INFO( status ), Rc_Buffer );
return( Rc_Buffer );
}
extern f_handle QOpenR( char *name )
/**********************************/
{
tiny_ret_t h;
h = DoOpen( name, FALSE, TIO_READ );
if( TINY_OK( h ) ) return( (f_handle)TINY_INFO( h ) );
LnkMsg( FTL+MSG_CANT_OPEN, "12", name, QErrMsg( TINY_INFO( h ) ) );
return( NIL_HANDLE );
}
extern f_handle QOpenRW( char *name )
/***********************************/
{
tiny_ret_t h;
h = DoOpen( name, TRUE, TIO_NORMAL );
if( TINY_OK( h ) ) return( (f_handle)TINY_INFO( h ) );
LnkMsg( FTL+MSG_CANT_OPEN, "12", name, QErrMsg( TINY_INFO( h ) ) );
return( NIL_HANDLE );
}
extern unsigned QRead( f_handle file, void *buffer, unsigned len, char *name )
/****************************************************************************/
/* read into far memory */
{
tiny_ret_t h;
CheckBreak();
h = TinyRead( file, buffer, len );
if( TINY_ERROR( h ) ) {
if( name != NULL ) {
LnkMsg( ERR+MSG_IO_PROBLEM, "12", name, QErrMsg( TINY_INFO( h ) ) );
} else {
return( -1 );
}
}
return( TINY_INFO(h) );
}
static unsigned TestWrite( f_handle file, void *buffer, unsigned len, char *name )
/*****************************************************************************/
{
tiny_ret_t h;
char rc_buff[RESOURCE_MAX_SIZE];
CheckBreak();
if( len == 0 ) return( 0 );
h = TinyWrite( file, buffer, len );
if( name != NULL ) {
if( TINY_ERROR( h ) ) {
LnkMsg( ERR+MSG_IO_PROBLEM, "12", name, QErrMsg( TINY_INFO( h ) ) );
return( ~0 );
} else if( TINY_INFO( h ) != len ) {
if( name != NULL ) {
Msg_Get( MSG_IOERRLIST_7, rc_buff );
LnkMsg( (FTL+MSG_IO_PROBLEM) & ~OUT_MAP, "12", name, rc_buff );
}
}
}
return( TINY_INFO(h) );
}
extern unsigned QWrite( f_handle file, void *buffer, unsigned len, char *name )
/*****************************************************************************/
{
while( len > (16*1024) ) {
if( TestWrite( file, buffer, 16*1024, name ) != 16*1024 ) return 0;
len -= 16*1024;
buffer = ((char *) buffer) + 16*1024;
}
return TestWrite( file, buffer, len, name );
}
char NLSeq[] = { "\r\n" };
extern void QWriteNL( f_handle file, char *name )
/***********************************************/
{
QWrite( file, NLSeq, sizeof( NLSeq ) - 1, name );
}
extern void QClose( f_handle file, char *name )
/*********************************************/
/* file close */
{
tiny_ret_t h;
CheckBreak();
h = TinyClose( file );
OpenFiles--;
if( TINY_OK( h ) || name == NULL ) return;
LnkMsg( ERR+MSG_IO_PROBLEM, "12", name, QErrMsg( TINY_INFO( h ) ) );
}
extern long QLSeek( f_handle file, long position, int start, char *name )
/***********************************************************************/
{
tiny_ret_t rc;
unsigned long pos;
CheckBreak();
rc = TinyLSeek( file, position, start, (void __near *)&pos );
if( TINY_ERROR( rc ) ) {
if( name != NULL )
LnkMsg( ERR+MSG_IO_PROBLEM, "12", name, QErrMsg( TINY_INFO( rc ) ) );
return( -1L ); /* for xmem checking */
}
return( pos );
}
extern void QSeek( f_handle file, long position, char *name )
/***********************************************************/
{
QLSeek( file, position, TIO_SEEK_START, name );
}
extern unsigned long QPos( f_handle file )
/****************************************/
{
unsigned long pos;
CheckBreak();
if( TINY_ERROR( TinyLSeek( file, 0L, TIO_SEEK_CURR, (void __near *)&pos ) ) ) {
return( -1L );
}
return( pos );
}
extern unsigned long QFileSize( f_handle file )
/*********************************************/
{
unsigned long curpos;
unsigned long size;
curpos = QPos( file );
if( TINY_ERROR( TinyLSeek( file, 0L, TIO_SEEK_END, (void __near *)&size ) ) )
size = 0;
TinySeek( file, curpos, TIO_SEEK_START );
return( size );
}
extern void QDelete( char *name )
/*******************************/
{
tiny_ret_t h;
if( name == NULL ) return;
h = TinyDelete( name );
if( TINY_ERROR( h ) ) {
if( TINY_INFO( h ) != 2 ) { /* file not found is OK */
LnkMsg( ERR+MSG_IO_PROBLEM, "12", name, QErrMsg( TINY_INFO( h ) ) );
}
}
}
extern bool QReadStr( f_handle file, char *dest, unsigned size, char *name )
/**************************************************************************/
/* quick read string (for reading directive file) */
{
bool eof;
char ch;
eof = FALSE;
while( --size > 0 ) {
if( QRead( file, &ch, 1, name ) == 0 ) {
eof = TRUE;
break;
} else if( ch != '\r' ) {
*dest++ = ch;
}
if( ch == '\n' ) break;
}
*dest = '\0';
return( eof );
}
extern bool QIsDevice( f_handle file )
/************************************/
{
if( TinyGetDeviceInfo( file ) & TIO_CTL_DEVICE ) {
return( TRUE );
} else {
return( FALSE ); // don't write the prompt if input not from stdin
}
}
extern f_handle ExeCreate( char *name )
/*************************************/
{
tiny_ret_t h;
h = DoOpen( name, TRUE, TIO_NORMAL );
LastResult = TINY_INFO( h );
if( TINY_OK( h ) ) return( TINY_INFO( h ) );
return( NIL_HANDLE );
}
static f_handle NSOpen( char *name, unsigned mode )
/*************************************************/
{
tiny_ret_t h;
h = DoOpen( name, FALSE, mode );
LastResult = TINY_INFO( h );
if( TINY_OK( h ) ) return( TINY_INFO( h ) );
return( NIL_HANDLE );
}
extern f_handle ExeOpen( char *name )
/***********************************/
{
return( NSOpen( name, TIO_READ_WRITE ) );
}
extern f_handle QObjOpen( char *name )
/************************************/
{
return( NSOpen( name, TIO_READ ) );
}
extern f_handle TempFileOpen( char *name )
/****************************************/
// open without suiciding. Don't create the file
{
return( NSOpen( name, TIO_READ ) );
}
extern int QMakeFileName( char **pos, char *name, char *fname )
/*************************************************************/
{
char *pathptr;
unsigned int path_len;
char *file_ptr;
pathptr = *pos;
if( pathptr == NULL ) return( 0 );
while( *pathptr != '\0' ) {
if( *pathptr == PATH_LIST_SEP ) *pos = ++pathptr;
for(;;) {
if( *pathptr == '\0' ) break;
if( *pathptr == PATH_LIST_SEP ) break;
pathptr++;
}
path_len = pathptr - *pos;
if( path_len != 0 ) {
memcpy( fname, *pos, path_len );
file_ptr = fname + path_len;
switch( file_ptr[ -1 ] ) {
CASE_PATH_SEP:
break;
default:
*file_ptr++ = PATH_SEP;
break;
}
strcpy( file_ptr, name );
*pos = pathptr;
return( 1 );
}
*pos = pathptr;
}
return( 0 );
}
extern bool QHavePath( char *name )
/*********************************/
{
return( *name == '\\' || *name == '/' || *(name + 1) == ':' );
}
extern bool QSysHelp( char **cmd_ptr )
{
cmd_ptr = cmd_ptr;
return( FALSE );
}
extern bool QModTime( char *name, time_t *time )
/**********************************************/
{
int result;
struct stat buf;
result = stat( name, &buf );
*time = buf.st_mtime;
return result != 0;
}
extern time_t QFModTime( int handle )
/***********************************/
{
struct stat buf;
fstat( handle, &buf );
return buf.st_mtime;
}
int ResOpen( const char *name, int access, ... )
/******************************************/
/* this just ignores the access parameter, and assumes it knows what the
* resource stuff really wants to do */
{
access = access; /* to avoid a warning */
return( NSOpen( (char *) name, TIO_READ ) );
}
extern char WaitForKey( void )
/****************************/
{
return getch();
}
extern void GetCmdLine( char *buff )
/**********************************/
{
getcmd( buff );
}
extern void TrapBreak( int sig_num )
/**********************************/
{
sig_num = sig_num; // to avoid a warning, will be optimized out.
if( CaughtBreak != BREAK_HANDLED ) {
CaughtBreak = BREAK_DETECTED;
}
}
extern void SetBreak( void )
/**************************/
{
BreakCond = TinyGetCtrlBreak();
TinySetCtrlBreak( 1 );
}
extern void RestoreBreak( void )
/******************************/
{
TinySetCtrlBreak( BreakCond );
}
⌨️ 快捷键说明
复制代码Ctrl + C
搜索代码Ctrl + F
全屏模式F11
增大字号Ctrl + =
减小字号Ctrl + -
显示快捷键?