oldcache.c
来自「开放源码的编译器open watcom 1.6.0版的源代码」· C语言 代码 · 共 286 行
C
286 行
/****************************************************************************
*
* 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: WHEN YOU FIGURE OUT WHAT THIS FILE DOES, PLEASE
* DESCRIBE IT HERE!
*
****************************************************************************/
/*
OLDCACHE - object file caching routines usable on 16-bit systems.
*/
#include <string.h>
#include <limits.h>
#include "linkstd.h"
#include "msg.h"
#include "alloc.h"
#include "wlnkmsg.h"
#include "fileio.h"
#include "objio.h"
#include "objcache.h"
static bool DumpFileCache( infilelist *, bool );
static bool Multipage;
#define CACHE_PAGE_SIZE (8*1024)
static unsigned NumCacheBlocks( unsigned long len )
/*************************************************/
// figure out the number of cache blocks necessary
{
unsigned numblocks;
numblocks = len / CACHE_PAGE_SIZE;
if( len % CACHE_PAGE_SIZE != 0 ) {
numblocks++;
}
return numblocks;
}
extern bool CacheOpen( file_list *list )
/**************************************/
{
infilelist *file;
unsigned numblocks;
char ** cache;
if( list == NULL ) return TRUE;
file = list->file;
if( file->flags & INSTAT_IOERR ) return( FALSE );
if( DoObjOpen( file ) ) {
file->flags |= INSTAT_IN_USE;
} else {
file->flags |= INSTAT_IOERR;
return( FALSE );
}
if( file->cache == NULL ) { // haven't set up cache for this yet.
file->len = QLSeek( file->handle, 0, LSEEK_END, file->name );
file->currpos = file->len;
if( file->len == 0 ) {
file->flags |= INSTAT_IOERR;
LnkMsg( ERR+MSG_BAD_OBJECT, "s", file->name );
return FALSE;
}
numblocks = NumCacheBlocks( file->len );
_PermAlloc( file->cache, numblocks * sizeof(char *) );
cache = file->cache;
while( numblocks > 0 ) {
*cache = NULL;
cache++;
numblocks--;
}
}
return TRUE;
}
extern void CacheClose( file_list *list, unsigned pass )
/******************************************************/
{
infilelist *file;
bool nukecache;
if( list == NULL ) return;
nukecache = TRUE;
file = list->file;
if( file->handle == NIL_HANDLE ) return;
file->flags &= ~INSTAT_IN_USE;
switch( pass ) {
case 1: /* first pass */
#if !defined( __QNX__ )
nukecache = !(file->flags & INSTAT_LIBRARY);
DumpFileCache( file, nukecache ); // don't cache .obj's
#endif
break;
case 2: /* second pass */
nukecache = !(file->flags & INSTAT_LIBRARY);
case 3: /* freeing structure */
#if !defined( __QNX__ )
DumpFileCache( file, nukecache );
#endif
QClose( file->handle, file->name );
file->handle = NIL_HANDLE;
FreeObjCache( list );
break;
}
}
extern void * CachePermRead( file_list *list, unsigned long pos, unsigned len )
/*****************************************************************************/
{
char * buf;
char * result;
buf = CacheRead( list, pos, len );
if( Multipage ) {
_LnkReAlloc( result, buf, len );
_ChkAlloc( TokBuff, TokSize );
} else {
_ChkAlloc( result, len );
memcpy( result, buf, len );
}
return result;
}
extern void * CacheRead( file_list * list, unsigned long pos, unsigned len )
/**************************************************************************/
/* read len bytes out of the cache. */
{
unsigned bufnum;
unsigned startnum;
unsigned offset;
unsigned amtread;
char * result;
char ** cache;
unsigned long newpos;
infilelist *file;
Multipage = FALSE;
file = list->file;
offset = pos % CACHE_PAGE_SIZE;
amtread = CACHE_PAGE_SIZE - offset;
startnum = pos / CACHE_PAGE_SIZE;
bufnum = startnum;
cache = file->cache;
for(;;) {
if( cache[bufnum] == NULL ) { // make sure page is in.
_ChkAlloc( cache[bufnum], CACHE_PAGE_SIZE );
newpos = (unsigned long) bufnum * CACHE_PAGE_SIZE;
if( file->currpos != newpos ) {
QSeek( file->handle, newpos, file->name );
}
file->currpos = newpos + CACHE_PAGE_SIZE;
QRead( file->handle, cache[bufnum], CACHE_PAGE_SIZE, file->name );
}
if( amtread >= len ) break;
amtread += CACHE_PAGE_SIZE; // it spans pages.
bufnum++;
Multipage = TRUE;
}
if( !Multipage ) {
result = cache[startnum] + offset;
} else {
if( len > TokSize ) {
TokSize = ROUND_UP( len, SECTOR_SIZE );
_LnkReAlloc( TokBuff, TokBuff, TokSize );
}
amtread = CACHE_PAGE_SIZE - offset;
memcpy( TokBuff, cache[startnum] + offset, amtread );
len -= amtread;
result = TokBuff + amtread;
for(;;) {
startnum++;
if( len <= CACHE_PAGE_SIZE ) {
memcpy( result, cache[startnum], len );
break;
} else {
memcpy( result, cache[startnum], CACHE_PAGE_SIZE );
len -= CACHE_PAGE_SIZE;
result += CACHE_PAGE_SIZE;
}
}
result = TokBuff;
}
return result;
}
extern bool CacheEnd( file_list * list, unsigned long pos )
/*********************************************************/
{
return pos >= list->file->len;
}
extern void CacheFree( file_list *list, void *mem )
/*************************************************/
// used for disposing things allocated by CachePermRead
{
list = list;
_LnkFree( mem );
}
static bool DumpFileCache( infilelist *file, bool nuke )
/******************************************************/
{
unsigned num;
unsigned savenum;
unsigned index;
char ** blocklist;
bool blockfreed;
blockfreed = FALSE;
if( nuke ) {
savenum = UINT_MAX;
} else {
savenum = file->currpos / CACHE_PAGE_SIZE;
}
if( file->cache != NULL ) {
num = NumCacheBlocks( file->len );
blocklist = file->cache;
for( index = 0; index < num; index++ ) {
if( index != savenum && *blocklist != NULL ) {
_LnkFree( *blocklist );
*blocklist = NULL;
blockfreed = TRUE;
}
blocklist++;
}
}
return blockfreed;
}
extern void FreeObjCache( file_list *list )
/*****************************************/
{
if( list == NULL ) return;
DumpFileCache( list->file, TRUE );
list->file->cache = NULL;
}
extern bool DumpObjCache( void )
/******************************/
// find and dump an object file cache.
{
infilelist *file;
file = CachedFiles;
while( file != NULL ) {
if( CurrMod == NULL || CurrMod->source == NULL
|| CurrMod->source->file != file ) {
if( DumpFileCache( file, TRUE ) ) return TRUE;
}
file = file->next;
}
return FALSE;
}
extern void CacheFini( void )
/***************************/
{
}
⌨️ 快捷键说明
复制代码Ctrl + C
搜索代码Ctrl + F
全屏模式F11
增大字号Ctrl + =
减小字号Ctrl + -
显示快捷键?