1359 lines
48 KiB
C
1359 lines
48 KiB
C
/* SCCSID = %W% %E% */
|
|
/*
|
|
* Copyright Microsoft Corporation, 1983-1987
|
|
*
|
|
* This Module contains Proprietary Information of Microsoft
|
|
* Corporation and should be treated as Confidential.
|
|
*/
|
|
/****************************************************************
|
|
* *
|
|
* LIBRARY PROCESSING ROUTINES *
|
|
* *
|
|
****************************************************************/
|
|
|
|
#include <minlit.h> /* Types and constants */
|
|
#include <bndtrn.h> /* Types and constants */
|
|
#include <bndrel.h> /* Types and constants */
|
|
#include <lnkio.h> /* Linker I/O definitions */
|
|
#include <lnkmsg.h> /* Error messages */
|
|
#include <extern.h> /* External declarations */
|
|
#include <stdlib.h>
|
|
#if OSMSDOS
|
|
#include <dos.h> /* DOS interface definitions */
|
|
#if CPU286
|
|
#define INCL_BASE
|
|
#include <os2.h> /* OS/2 interface definitions */
|
|
#if defined(M_I86LM)
|
|
#undef NEAR
|
|
#define NEAR
|
|
#endif
|
|
#endif
|
|
#endif
|
|
|
|
#define DICHDR 0xF1 /* Dictionary header type (F1H) */
|
|
|
|
#if OSXENIX
|
|
#define ShrOpenRd(f) fopen(f,RDBIN)
|
|
#endif
|
|
#if NEWIO
|
|
#include <errno.h> /* System error codes */
|
|
#endif
|
|
|
|
#define PNSORTMAX 512 /* Maximum # modules can be sorted */
|
|
|
|
typedef struct _edmt /* Extended Dictionary Module Table */
|
|
{
|
|
WORD page;
|
|
WORD list;
|
|
}
|
|
edmt;
|
|
|
|
LOCAL FTYPE fUndefHit; /* found this undef in the library */
|
|
LOCAL FTYPE fFileExtracted; /* Took file from library flag */
|
|
LOCAL FTYPE fUndefsSeen; /* True if externals seen in library */
|
|
LOCAL WORD ipnMac; /* Count of page numbers in sort table */
|
|
LOCAL WORD *pnSort; /* Sort table for library page numbers */
|
|
/* f(ifh) = pointer to dictionary */
|
|
LOCAL WORD mpifhcpnHash[IFHLIBMAX];
|
|
/* f(ifh) = # pages in hash table */
|
|
LOCAL BYTE mpifhAlign[IFHLIBMAX];
|
|
/* f(ifh) = lib alignment factor */
|
|
LOCAL RBTYPE vrpNewList; /* List of unprocessed files */
|
|
LOCAL FTYPE vfLibOpen; /* Library open flag */
|
|
#if M_BYTESWAP OR defined( _WIN32 )
|
|
#define getfarword getword /* This assumes no far data */
|
|
#else
|
|
#define getfarword(x) (((WORD FAR *)(x))[0])
|
|
#endif
|
|
|
|
/*
|
|
* INTERFACE WITH ASSEMBLY LANGUAGE FUNCTION
|
|
*/
|
|
|
|
WORD libAlign; /* Library alignment factor */
|
|
WORD libcpnHash; /* Length of hash table in pages */
|
|
BYTE FAR *mpifhDict[IFHLIBMAX];
|
|
|
|
/*
|
|
* FUNCTION PROTOTYPES
|
|
*/
|
|
|
|
LOCAL unsigned char NEAR OpenLibrary(unsigned char *sbLib);
|
|
LOCAL void NEAR FreeDictionary(void);
|
|
#if CPU8086 OR CPU286
|
|
LOCAL WORD NEAR readfar(int fh, char FAR *buf,int n);
|
|
#endif
|
|
LOCAL void NEAR GetDictionary(void);
|
|
LOCAL WORD NEAR GetLib(void);
|
|
LOCAL void ProcessAnUndef(APROPNAMEPTR papropUndef,
|
|
RBTYPE rhte,
|
|
RBTYPE rprop,
|
|
WORD fNewHte);
|
|
LOCAL int cdecl FGtNum(const WORD *pn1, const WORD *pn2);
|
|
LOCAL void NEAR LookMod(edmt *modtab,unsigned short iMod);
|
|
LOCAL void NEAR LookPage(edmt *modtab,unsigned short cMod,unsigned short page);
|
|
LOCAL void NEAR ProcExtDic(char *pExtDic);
|
|
LOCAL char * NEAR GetExtDic(void);
|
|
|
|
#if NEW_LIB_SEARCH
|
|
|
|
/// undef lookaside list
|
|
|
|
typedef struct tag_UND
|
|
{
|
|
struct tag_UND * pNext;
|
|
APROPNAMEPTR papropUndef;
|
|
DWORD dwLibMask;
|
|
RBTYPE rhte;
|
|
|
|
} UND;
|
|
|
|
#define C_UNDS_POOL 128
|
|
|
|
typedef struct tag_UNDPOOL
|
|
{
|
|
struct tag_UNDPOOL *pNext;
|
|
UND und[C_UNDS_POOL];
|
|
} UNDPOOL;
|
|
|
|
|
|
// pool storage management variables
|
|
|
|
UNDPOOL *pundpoolCur;
|
|
UNDPOOL *pundpoolHead;
|
|
int iundPool = C_UNDS_POOL;
|
|
UND * pundFree;
|
|
UND * pundListHead;
|
|
|
|
#define FUndefsLeft() (pundListHead != NULL)
|
|
|
|
void StoreUndef(APROPNAMEPTR, RBTYPE, RBTYPE, WORD);
|
|
|
|
#else
|
|
#define FUndefsLeft() (fUndefsSeen)
|
|
#endif
|
|
|
|
FTYPE fStoreUndefsInLookaside = FALSE;
|
|
|
|
/////
|
|
|
|
|
|
#if NOASM
|
|
LOCAL WORD NEAR rolw(WORD x, WORD n) /* Rotate word left */
|
|
{
|
|
return(LO16BITS((x << n) | ((x >> (WORDLN - n)) & ~(~0 << n))));
|
|
}
|
|
|
|
LOCAL WORD NEAR rorw(WORD x, WORD n) /* Rotate word right */
|
|
{
|
|
return(LO16BITS((x << (WORDLN - n)) | ((x >> n) & ~(~0 << (WORDLN - n)))));
|
|
}
|
|
#endif
|
|
|
|
#if OSMSDOS
|
|
BSTYPE NEAR ShrOpenRd(pname)
|
|
char *pname; /* Name of file (null-terminated) */
|
|
{
|
|
int fh; /* File handle */
|
|
|
|
|
|
#if NEWIO
|
|
if(mpifhfh[ifhLibCur])
|
|
{
|
|
fh = mpifhfh[ifhLibCur];
|
|
/* If dictionary not allocated, seek to beginning since we're
|
|
* somewhere else now.
|
|
*/
|
|
if(!mpifhDict[ifhLibCur])
|
|
if (_lseek(fh,0L,0) == -1) {
|
|
return NULL;
|
|
}
|
|
}
|
|
else
|
|
fh = SmartOpen(pname,ifhLibCur);
|
|
if(fh > 0)
|
|
{
|
|
fflush(bsInput);
|
|
bsInput->_file = (char) fh;
|
|
return(bsInput);
|
|
}
|
|
else
|
|
return(NULL);
|
|
#else
|
|
if((fh = _sopen(pname,O_RDONLY | O_BINARY,SH_DENYWR)) < 0)
|
|
return(NULL);
|
|
return(fdopen(fh,RDBIN));
|
|
#endif
|
|
}
|
|
#endif /* OSMSDOS */
|
|
|
|
#pragma check_stack(on)
|
|
|
|
/****************************************************************
|
|
* *
|
|
* OpenLibrary: *
|
|
* *
|
|
* This function takes as its arguments a pointer to the text *
|
|
* of the name of the library to open, a count of the bytes in *
|
|
* that name, an index into a global table in which to place *
|
|
* the file handle for the opened library. It returns TRUE if *
|
|
* it succeeds, FALSE if it fails to open the file; and it *
|
|
* dies gracefully if the file is not a valid library. *
|
|
* *
|
|
****************************************************************/
|
|
|
|
LOCAL FTYPE NEAR OpenLibrary(sbLib)
|
|
BYTE *sbLib; /* Library name */
|
|
{
|
|
SBTYPE libnam; /* Library name */
|
|
WORD reclen; /* Library header record length */
|
|
BSTYPE bsLib; /* File stream pointer for library */
|
|
|
|
memcpy(libnam,&sbLib[1],min(sizeof(libnam), B2W(sbLib[0])));
|
|
/* Copy library name */
|
|
libnam[B2W(sbLib[0])] = '\0'; /* Null-terminate name */
|
|
/* WARNING: do not assign bsInput to NULL if open fails, it
|
|
* screws up NEWIO.
|
|
*/
|
|
if((bsLib = ShrOpenRd(libnam)) != NULL)
|
|
{ /* If open successful */
|
|
bsInput = bsLib;
|
|
/* If dictionary already allocated, no need to do anything */
|
|
if(mpifhDict[ifhLibCur])
|
|
return((FTYPE) TRUE);
|
|
#if OSMSDOS
|
|
/* Reduce buffer size. We can avoid calling setvbuf() because
|
|
* everything is set up properly at this point.
|
|
*/
|
|
#if OWNSTDIO
|
|
bsInput->_bsize = 512;
|
|
#else
|
|
setvbuf(bsInput, bsInput->_base, _IOFBF, 512);
|
|
#endif
|
|
#endif
|
|
if(getc(bsInput) == LIBHDR) /* If we have a library */
|
|
{
|
|
reclen = (WORD) (3 + WSGets());
|
|
/* Get record length */
|
|
for(libAlign = 15; libAlign &&
|
|
!(reclen & (1 << libAlign)); --libAlign);
|
|
/* Calculate alignment factor */
|
|
mpifhAlign[ifhLibCur] = (BYTE) libAlign;
|
|
if(libAlign >= 4 && reclen == (WORD) (1 << libAlign))
|
|
{ /* Check legality of alignment */
|
|
libHTAddr = (long) WSGets();
|
|
libHTAddr += (long) WSGets() << WORDLN;
|
|
/* Get the offset of the hash table */
|
|
if (libHTAddr <= 0L)
|
|
Fatal(ER_badlib,libnam);
|
|
if ((mpifhcpnHash[ifhLibCur] = WSGets()) <= 0)
|
|
/* Get size of hash table in pages */
|
|
Fatal(ER_badlib,libnam);
|
|
#if OSMSDOS
|
|
/* Restore big buffer size. Avoid calling setvbuf(). */
|
|
#if OWNSTDIO
|
|
bsInput->_bsize = LBUFSIZ;
|
|
#else
|
|
setvbuf(bsInput, bsInput->_base, _IOFBF, LBUFSIZ);
|
|
#endif
|
|
#endif
|
|
return((FTYPE) TRUE); /* Success */
|
|
}
|
|
}
|
|
Fatal(ER_badlib,libnam);
|
|
}
|
|
return(FALSE); /* Failure */
|
|
}
|
|
|
|
#pragma check_stack(off)
|
|
|
|
|
|
/*
|
|
* LookupLibSym: look up a symbol in library dictionary
|
|
*
|
|
* The minimum page size is 16, so we can return paragraph offsets.
|
|
* This is a win because offsets are stored as paragraphs in the
|
|
* sorting table anyway. Also, the majority of libraries have page
|
|
* size of 16.
|
|
*
|
|
* Parameters:
|
|
* char *psb - pointer to length-prefixed string
|
|
* Returns:
|
|
* Long paragraph offset to location of module which defines
|
|
* the symbol, or 0L if not found.
|
|
*/
|
|
#if NOASM
|
|
LOCAL WORD NEAR LookupLibSym(psb)
|
|
BYTE *psb; /* Symbol to look up */
|
|
{
|
|
WORD i1; /* First hash value */
|
|
WORD d1; /* First hash delta */
|
|
WORD i2; /* Second hash value */
|
|
WORD d2; /* Second hash delta */
|
|
WORD pn; /* Page number */
|
|
WORD dpn; /* Page number delta */
|
|
WORD pslot; /* Page slot */
|
|
WORD dpslot; /* Page slot delta */
|
|
WORD ipn; /* Initial page number */
|
|
BYTE FAR *hpg;
|
|
#if NOASM
|
|
WORD ch1; /* Character */
|
|
WORD ch2; /* Character */
|
|
char *pc1; /* Character pointer */
|
|
char *pc2; /* Character pointer */
|
|
WORD length; /* Symbol length */
|
|
#endif
|
|
|
|
#if LIBDEBUG
|
|
OutSb(stderr,psb);
|
|
fprintf(stderr," is wanted; dictionary is %d pages\r\n",libcpnHash);
|
|
#endif
|
|
#if NOASM
|
|
length = B2W(psb[0]); /* Get symbol length */
|
|
pc1 = (char *) psb; /* Initialize */
|
|
pc2 = (char *) &psb[B2W(psb[0])]; /* Initialize */
|
|
i1 = 0; /* Initialize */
|
|
d1 = 0; /* Initialize */
|
|
i2 = 0; /* Initialize */
|
|
d2 = 0; /* Initialize */
|
|
while(length--) /* Hashing loop */
|
|
{
|
|
ch1 = (WORD) (B2W(*pc1++) | 040);/* Force to lower case */
|
|
ch2 = (WORD) (B2W(*pc2--) | 040);/* Force to lower case */
|
|
i1 = (WORD) (rolw(i1,2) ^ ch1); /* Hash */
|
|
d1 = (WORD) (rolw(d1,2) ^ ch2); /* Hash */
|
|
i2 = (WORD) (rorw(i2,2) ^ ch2); /* Hash */
|
|
d2 = (WORD) (rorw(d2,2) ^ ch1); /* Hash */
|
|
}
|
|
#else
|
|
i1 = libhash(psb,&d1,&i2,&d2); /* Hash */
|
|
#endif
|
|
pn = (WORD) (i1 % libcpnHash); /* Calculate page number index */
|
|
if(!(dpn = (WORD) (d1 % libcpnHash))) dpn = 1;
|
|
/* Calculate page number delta */
|
|
pslot = (WORD) (i2 % CSLOTMAX); /* Calculate page slot index */
|
|
if(!(dpslot = (WORD) (d2 % CSLOTMAX))) dpslot = 1;
|
|
/* Calculate page slot delta */
|
|
#if LIBDEBUG
|
|
fprintf(stderr,"page index %d, delta %d, bucket index %d, delta %d\r\n",
|
|
pn,dpn,pslot,dpslot);
|
|
#endif
|
|
ipn = pn; /* Remember initial page number */
|
|
for(;;) /* Search loop */
|
|
{
|
|
#if LIBDEBUG
|
|
fprintf(stderr,"Page %d:\r\n",pn);
|
|
#endif
|
|
// Get pointer to the dictionary page
|
|
|
|
hpg = mpifhDict[ifhLibCur] + (pn << LG2PAG);
|
|
|
|
for(i2 = 0; i2 < CSLOTMAX; ++i2)/* Loop to check slots */
|
|
{
|
|
#if LIBDEBUG
|
|
fprintf(stderr,"Bucket %d %sempty, page %sfull\r\n",
|
|
pslot,hpg[pslot]? "not ": "",
|
|
B2W(hpg[CSLOTMAX]) == 0xFF? "": "not ");
|
|
#endif
|
|
if(!(i1 = (WORD) (B2W(hpg[pslot]) << 1)))
|
|
{ /* If slot is empty */
|
|
if(B2W(hpg[CSLOTMAX]) == 0xFF) break;
|
|
/* If page is full, break */
|
|
return(0); /* Search failed */
|
|
}
|
|
#if LIBDEBUG
|
|
fprintf(stderr," Comparing ");
|
|
OutSb(stderr,psb);
|
|
fprintf(stderr," to ");
|
|
OutSb(stderr,&hpg[i1]);
|
|
fprintf(stderr," %signoring case\r\n",fIgnoreCase? "": "not ");
|
|
#endif
|
|
if(psb[0] == hpg[i1] && SbNewComp(psb,&hpg[i1],fIgnoreCase))
|
|
{ /* If symbols match */
|
|
#if LIBDEBUG
|
|
fprintf(stderr,"Match found in slot %d\r\n",i2 >> 1);
|
|
#endif
|
|
i1 += (WORD) (B2W(hpg[i1]) + 1); /* Skip over name */
|
|
i1 = getfarword(&hpg[i1]);
|
|
/* Get page number of module */
|
|
return(i1); /* Return page number of module */
|
|
}
|
|
if((pslot += dpslot) >= CSLOTMAX) pslot -= CSLOTMAX;
|
|
/* Try next slot */
|
|
}
|
|
if((pn += dpn) >= libcpnHash) pn -= libcpnHash;
|
|
/* Try next page */
|
|
if (ipn == pn) return(0); /* Once around without finding it */
|
|
}
|
|
}
|
|
#endif /*NOASM*/
|
|
/*
|
|
* FreeDictionary : free space allocated for dictionaries
|
|
*/
|
|
LOCAL void NEAR FreeDictionary ()
|
|
{
|
|
WORD i;
|
|
|
|
for (i = 0; i < ifhLibMac; ++i)
|
|
if (mpifhDict[i])
|
|
FFREE(mpifhDict[i]);
|
|
}
|
|
|
|
#if CPU8086 OR CPU286
|
|
/*
|
|
* readfar : read() with a far buffer
|
|
*
|
|
* Emulate read() except use a far buffer. Call the system
|
|
* directly.
|
|
*
|
|
* Returns:
|
|
* 0 if error, else number of bytes read.
|
|
*/
|
|
LOCAL WORD NEAR readfar (fh, buf, n)
|
|
int fh; /* File handle */
|
|
char FAR *buf; /* Buffer to store bytes in */
|
|
int n; /* # bytes to read */
|
|
{
|
|
#if OSMSDOS
|
|
unsigned bytesread; /* Number of bytes read */
|
|
|
|
#if CPU8086
|
|
if (_dos_read(fh, buf, n, &bytesread))
|
|
return(0);
|
|
return(bytesread);
|
|
#else
|
|
|
|
if(DosRead(fh,buf,n,(unsigned FAR *) &bytesread))
|
|
return(0);
|
|
return(bytesread);
|
|
#endif
|
|
#endif /* OSMSDOS */
|
|
#if OSXENIX
|
|
char mybuf[PAGLEN];
|
|
int cppage;
|
|
char *p;
|
|
|
|
while(n > 0)
|
|
{
|
|
cppage = n > PAGLEN ? PAGLEN : n;
|
|
if(read(fh,mybuf,cppage) != cppage)
|
|
return(0);
|
|
n -= cppage;
|
|
for(p = mybuf; p < mybuf[cppage]; *buf++ = *p++);
|
|
}
|
|
#endif
|
|
}
|
|
#endif
|
|
|
|
LOCAL void NEAR GetDictionary ()
|
|
{
|
|
unsigned cb;
|
|
|
|
|
|
#if CPU8086 OR CPU286
|
|
// If there is more than 128 pages in dictionary return,
|
|
// because the dictionary is bigger than 64k
|
|
|
|
if (libcpnHash >= 128)
|
|
return;
|
|
#endif
|
|
|
|
cb = libcpnHash << LG2PAG;
|
|
mpifhDict[ifhLibCur] = GetMem(cb);
|
|
|
|
// Go to the dictionary and read it in a single call
|
|
|
|
#if defined(M_I386) || defined( _WIN32 )
|
|
if (fseek(bsInput, libHTAddr, 0))
|
|
Fatal(ER_badlib,1 + GetPropName(FetchSym(mpifhrhte[ifhLibCur],FALSE)));
|
|
if (fread(mpifhDict[ifhLibCur], 1, cb, bsInput) != (int) cb)
|
|
Fatal(ER_badlib,1 + GetPropName(FetchSym(mpifhrhte[ifhLibCur],FALSE)));
|
|
#else
|
|
_lseek(fileno(bsInput), libHTAddr, 0);
|
|
if (readfar(fileno(bsInput), mpifhDict[ifhLibCur], cb) != cb)
|
|
Fatal(ER_badlib,1 + GetPropName(FetchSym(mpifhrhte[ifhLibCur],FALSE)));
|
|
#endif
|
|
}
|
|
|
|
#pragma check_stack(on)
|
|
|
|
LOCAL WORD NEAR GetLib(void) /* Open the next library in list */
|
|
{
|
|
AHTEPTR pahteLib; /* Pointer to library name */
|
|
#if OSMSDOS
|
|
SBTYPE sbLib; /* Library name */
|
|
SBTYPE sbNew; /* New parts to library name */
|
|
#endif
|
|
|
|
if(mpifhrhte[ifhLibCur] == RHTENIL) /* If this library is to be skipped */
|
|
{
|
|
return(FALSE); /* No library opened */
|
|
}
|
|
for(;;) /* Loop to open library */
|
|
{
|
|
pahteLib = (AHTEPTR ) FetchSym(mpifhrhte[ifhLibCur],FALSE);
|
|
/* Get name from hash table */
|
|
if(OpenLibrary(GetFarSb(pahteLib->cch))) break;
|
|
/* Break if lib opened okay */
|
|
if(fNoprompt)
|
|
Fatal(ER_libopn,1 + GetFarSb(pahteLib->cch));
|
|
else
|
|
{
|
|
sbLib[0] = '\0'; /* No string yet */
|
|
UpdateFileParts(sbLib,GetFarSb(pahteLib->cch));
|
|
(*pfPrompt)(sbNew,ER_libopn, /* Prompt for new filespec */
|
|
(int) (__int64) (1 + GetFarSb(pahteLib->cch)),
|
|
P_EnterNewFileSpec, 0);
|
|
}
|
|
if(fNoprompt || !sbNew[0])
|
|
{
|
|
mpifhrhte[ifhLibCur] = RHTENIL;
|
|
/* Do not bother next time */
|
|
return(FALSE); /* Unsuccessful */
|
|
}
|
|
#if OSMSDOS
|
|
UpdateFileParts(sbLib,sbNew); /* Update file name with new parts */
|
|
PropSymLookup(sbLib,ATTRFIL,TRUE);
|
|
/* Add library to symbol table */
|
|
mpifhrhte[ifhLibCur] = vrhte; /* Save virtual address */
|
|
AddLibPath(ifhLibCur); /* Add default path spec, maybe */
|
|
#endif
|
|
}
|
|
vfLibOpen = (FTYPE) TRUE; /* A library is open */
|
|
libcpnHash = mpifhcpnHash[ifhLibCur];
|
|
libAlign = mpifhAlign[ifhLibCur];
|
|
if (mpifhDict[ifhLibCur] == NULL) /* If dictionary not allocated, do it */
|
|
GetDictionary();
|
|
return(TRUE); /* Success */
|
|
}
|
|
|
|
#pragma check_stack(off)
|
|
|
|
/****************************************************************
|
|
* *
|
|
* ProcessAnUndef: *
|
|
* *
|
|
* This function takes as its arguments two pointers, two *
|
|
* RBTYPEs, and a flag. It does not return a meaningful *
|
|
* value. Most of the parameters to this function are *
|
|
* dummies; this function's address is passed as a parameter, *
|
|
* and its parameter list must match those of all the *
|
|
* functions whose addresses can be passed as a parameter to *
|
|
* the same function to which ProcessAnUndef's address is *
|
|
* passed. Called by EnSyms. *
|
|
* *
|
|
****************************************************************/
|
|
|
|
LOCAL void ProcessAnUndef(APROPNAMEPTR papropUndef,
|
|
RBTYPE rhte,
|
|
RBTYPE rprop__NotUsed__,
|
|
WORD fNewHte__NotUsed__)
|
|
{
|
|
AHTEPTR pahte; /* Pointer to hash table entry */
|
|
WORD pn; /* Library page number */
|
|
APROPUNDEFPTR pUndef;
|
|
ATTRTYPE attr;
|
|
#if NOT NEWSYM
|
|
SBTYPE sb; /* Undefined symbol */
|
|
#endif
|
|
|
|
fUndefHit = FALSE;
|
|
|
|
pUndef = (APROPUNDEFPTR ) papropUndef;
|
|
|
|
attr = pUndef->au_flags;
|
|
|
|
// don't pull out any "weak" externs or unused aliased externals
|
|
if (((attr & WEAKEXT) && !(attr & UNDECIDED)) ||
|
|
((attr & SUBSTITUTE) && !(attr & SEARCH_LIB)))
|
|
{
|
|
fUndefHit = TRUE; // this item is effectively resolved...
|
|
return;
|
|
}
|
|
|
|
fUndefsSeen = (FTYPE) TRUE; /* Set flag */
|
|
if(!mpifhDict[ifhLibCur] && !vfLibOpen)
|
|
return; /* Return if unable to get library */
|
|
|
|
pahte = (AHTEPTR ) FetchSym(rhte,FALSE);
|
|
/* Fetch name from symbol table */
|
|
#if NOT NEWSYM
|
|
memcpy(sb,pahte->cch,B2W(pahte->cch[0]) + 1);
|
|
/* Copy name */
|
|
#endif
|
|
#if LIBDEBUG
|
|
fprintf(stdout,"Looking for '%s' - ", 1+GetFarSb(pahte->cch));
|
|
fflush(stdout);
|
|
#endif
|
|
#if NEWSYM
|
|
if(pn = LookupLibSym(GetFarSb(pahte->cch)))
|
|
#else
|
|
if(pn = LookupLibSym(sb)) /* If symbol defined in this library */
|
|
#endif
|
|
{
|
|
fUndefHit = TRUE;
|
|
#if LIBDEBUG
|
|
fprintf(stdout,"Symbol found at page %xH\r\n", pn);
|
|
fflush(stdout);
|
|
#endif
|
|
/* We now try to stuff the page number (pn) into a table that will
|
|
* be sorted later.
|
|
*/
|
|
if (ipnMac < PNSORTMAX)
|
|
{
|
|
pnSort[ipnMac++] = pn;
|
|
return;
|
|
}
|
|
/*
|
|
* No room to save the file offset so save file directly.
|
|
*/
|
|
pahte = (AHTEPTR ) FetchSym(mpifhrhte[ifhLibCur],FALSE);
|
|
/*
|
|
* If SaveInput returns 0, then module was seen before. Means
|
|
* that dictionary says symbol is defined in this module but
|
|
* for some reason, such as IMPDEF, the definition wasn't
|
|
* accepted. In this case, we return.
|
|
*/
|
|
if(!SaveInput(GetFarSb(pahte->cch), (long)pn << libAlign, ifhLibCur, 0))
|
|
return;
|
|
/*
|
|
* If first module extracted, save start of file list.
|
|
*/
|
|
if(!fFileExtracted)
|
|
{
|
|
vrpNewList = vrpropTailFile;
|
|
fFileExtracted = (FTYPE) TRUE;
|
|
}
|
|
}
|
|
#if LIBDEBUG
|
|
else
|
|
{
|
|
fprintf(stdout, "Symbol NOT found\r\n"); /* Debug message */
|
|
fflush(stdout);
|
|
}
|
|
#endif
|
|
}
|
|
|
|
#if NEW_LIB_SEARCH
|
|
|
|
void StoreUndef(APROPNAMEPTR papropUndef, RBTYPE rhte,
|
|
RBTYPE rprop, WORD fNewHte)
|
|
{
|
|
UND * pund;
|
|
APROPUNDEFPTR pUndef;
|
|
ATTRTYPE attr;
|
|
|
|
pUndef = (APROPUNDEFPTR ) papropUndef;
|
|
|
|
attr = pUndef->au_flags;
|
|
|
|
// don't pull out any "weak" externs or unused aliased externals
|
|
if (((attr & WEAKEXT) && !(attr & UNDECIDED)) ||
|
|
((attr & SUBSTITUTE) && !(attr & SEARCH_LIB)))
|
|
return;
|
|
|
|
#ifdef LIBDEBUG
|
|
{
|
|
AHTEPTR pahte;
|
|
pahte = (AHTEPTR) FetchSym(rhte,FALSE);
|
|
fprintf(stdout,"Adding '%s'\r\n", 1+GetFarSb(pahte->cch));
|
|
fflush(stdout);
|
|
}
|
|
#endif
|
|
|
|
if (pundFree) // check free list
|
|
{
|
|
pund = pundFree;
|
|
pundFree = pundFree->pNext;
|
|
}
|
|
else if (iundPool < C_UNDS_POOL) // check pool
|
|
{
|
|
pund = &pundpoolCur->und[iundPool];
|
|
iundPool++;
|
|
}
|
|
else
|
|
{
|
|
// allocate new pool...
|
|
|
|
pundpoolCur = (UNDPOOL *)GetMem(sizeof(UNDPOOL));
|
|
|
|
pundpoolCur->pNext = pundpoolHead;
|
|
pundpoolHead = pundpoolCur;
|
|
pund = &pundpoolCur->und[0];
|
|
iundPool = 1; // entry zero is already used up
|
|
}
|
|
|
|
pund->dwLibMask = 0;
|
|
pund->pNext = pundListHead;
|
|
pund->papropUndef = papropUndef;
|
|
pund->rhte = rhte;
|
|
pundListHead = pund;
|
|
}
|
|
|
|
#endif
|
|
|
|
/*
|
|
* Greater-than comparator to be used by Sort routine.
|
|
*/
|
|
|
|
LOCAL int cdecl FGtNum(const WORD *pn1, const WORD *pn2)
|
|
{
|
|
if (*pn1 < *pn2)
|
|
return(-1);
|
|
if (*pn1 > *pn2)
|
|
return(1);
|
|
return(0);
|
|
}
|
|
|
|
/************************************************************************
|
|
* Extended Dictionary
|
|
*
|
|
* The extended dictionary occurs at the end of the regular dictionary
|
|
* and contains a first-level dependency tree for all the modules
|
|
* in the library.
|
|
************************************************************************/
|
|
|
|
#define LIBEXD 0xf2 /* Library EXtended Dictionary */
|
|
|
|
|
|
|
|
|
|
/****************************************************************
|
|
* *
|
|
* Extended Dictionary Format: *
|
|
* *
|
|
* *
|
|
* BYTE =0xF2 Extended Dictionary header *
|
|
* WORD length of extended dictionary in bytes *
|
|
* excluding 1st 3 bytes *
|
|
* *
|
|
* Start of ext. dictionary: *
|
|
* *
|
|
* WORD number of modules in library = N *
|
|
* *
|
|
* Module table, indexed by module number, with N + 1 fixed- *
|
|
* length entries: *
|
|
* *
|
|
* WORD module page number *
|
|
* WORD offset from start of ext. dictionary to list *
|
|
* of required modules *
|
|
* *
|
|
* Last entry is null. *
|
|
* *
|
|
* Module dependency lists, N variable-length lists: *
|
|
* *
|
|
* WORD list length (number of required modules) *
|
|
* WORD module index, 0-based; this is index to module *
|
|
* . . . table at the begin of ext. dictionary. *
|
|
* . . . *
|
|
* *
|
|
* *
|
|
****************************************************************/
|
|
|
|
|
|
/*
|
|
* LookMod : look up a module by index in the extended dictionary
|
|
*
|
|
* Get the list of modules required by the given module. If not
|
|
* already marked, save index in sorting table (which will be
|
|
* converted to page number later) and mark the entry in the
|
|
* module table as seen by setting the low bit of the list offset.
|
|
*
|
|
* Parameters:
|
|
* modtab: Pointer to module table
|
|
* iMod: Index into table, 0-based
|
|
*/
|
|
|
|
LOCAL void NEAR LookMod (edmt *modtab, WORD iMod)
|
|
{
|
|
WORD *pw; /* Pointer to list of indexes */
|
|
WORD n; /* List counter */
|
|
|
|
/*
|
|
* Get the pointer to the list. Mask off low bit since it is used
|
|
* as a marker.
|
|
*/
|
|
pw = (WORD *) ((char *) modtab + (modtab[iMod].list & ~1));
|
|
/*
|
|
* For every entry in the list, if the corresponding entry in the
|
|
* module table is not marked, save the index in pnSort and mark
|
|
* the entry in the module table.
|
|
*/
|
|
for(n = *pw++; n--; pw++)
|
|
{
|
|
if(!(modtab[*pw].list & 1))
|
|
{
|
|
/*
|
|
* Check for table overflow.
|
|
*/
|
|
if(ipnMac == PNSORTMAX)
|
|
return;
|
|
pnSort[ipnMac++] = *pw;
|
|
modtab[*pw].list |= 1;
|
|
}
|
|
}
|
|
}
|
|
|
|
/*
|
|
* LookPage : Look up a module in the module table by page number
|
|
*
|
|
* Use binary search. If page is found, call LookMod() on the
|
|
* matching entry.
|
|
*
|
|
* Parameters:
|
|
* modtab: Pointer to module table
|
|
* cMod: Number of entries in table
|
|
* page: Page number
|
|
* ASSUMES:
|
|
* The highest entry in the table has a page number of 0xffff.
|
|
*/
|
|
|
|
LOCAL void NEAR LookPage (edmt *modtab, WORD cMod, WORD page)
|
|
{
|
|
WORD mid; /* Current mid point */
|
|
WORD lo, hi; /* Current low and high points */
|
|
|
|
lo = 0; /* Table is 0-based. */
|
|
hi = (WORD) (cMod - 1);
|
|
while(lo <= hi)
|
|
{
|
|
if(modtab[mid = (WORD) ((lo + hi) >> 1)].page == page)
|
|
{
|
|
modtab[mid].list |= 1;
|
|
LookMod(modtab,mid);
|
|
return;
|
|
}
|
|
else if(modtab[mid].page < page)
|
|
lo = (WORD) (mid + 1);
|
|
else
|
|
hi = (WORD) (mid - 1);
|
|
}
|
|
}
|
|
|
|
/*
|
|
* ProcExtDic : Process Extended Dictionary
|
|
*
|
|
* Store in pnSort all the secondary modules required by
|
|
* the modules obtained from the regular dictionary lookup.
|
|
*
|
|
* Parameters:
|
|
* pExtDic: Pointer to extended dictionary
|
|
*/
|
|
|
|
LOCAL void NEAR ProcExtDic (pExtDic)
|
|
char *pExtDic;
|
|
{
|
|
WORD *p;
|
|
WORD *pEnd;
|
|
WORD cMod;
|
|
edmt *modtab;
|
|
|
|
cMod = getword(pExtDic);
|
|
modtab = (edmt *) (pExtDic + 2);
|
|
|
|
/* For the binary search algorithm, we make an artifical last entry
|
|
* with a page # at least as high as anything else.
|
|
*/
|
|
|
|
modtab[cMod].page = 0xffff;
|
|
|
|
/* Process by page numbers */
|
|
|
|
for(p = pnSort, pEnd = &pnSort[ipnMac]; p < pEnd; ++p)
|
|
LookPage(modtab, cMod, *p);
|
|
|
|
/* Now pnSort from pEnd to lfaSort[ipnMac] contains module
|
|
* index numbers. Process by index number and convert to page.
|
|
*/
|
|
|
|
for( ; p < &pnSort[ipnMac]; ++p)
|
|
{
|
|
LookMod(modtab,*p);
|
|
*p = modtab[*p].page;
|
|
}
|
|
}
|
|
|
|
/*
|
|
* GetExtDic - Get Extended Dictionary
|
|
*/
|
|
|
|
LOCAL char * NEAR GetExtDic ()
|
|
{
|
|
char *p;
|
|
int length;
|
|
|
|
if(!vfLibOpen)
|
|
if(!GetLib())
|
|
return(NULL);
|
|
/* WARNING: we must just have read dictionary for this to work,
|
|
* otherwise an fseek() is required here.
|
|
*/
|
|
if (!mpifhDict[ifhLibCur])
|
|
{
|
|
fflush(bsInput);
|
|
if (fseek(bsInput, libHTAddr + (libcpnHash << LG2PAG), 0))
|
|
Fatal(ER_badlib,1 + GetPropName(FetchSym(mpifhrhte[ifhLibCur],FALSE)));
|
|
}
|
|
if(getc(bsInput) != LIBEXD)
|
|
return(NULL);
|
|
if((p = GetMem(length = WSGets())) != NULL)
|
|
if(fread(p,1,length,bsInput) != length)
|
|
{
|
|
FreeMem(p);
|
|
p = NULL;
|
|
}
|
|
return(p);
|
|
}
|
|
|
|
|
|
char *pExtDic = NULL; /* Pointer to extended dictionary */
|
|
|
|
/****************************************************************
|
|
* *
|
|
* LibrarySearch: *
|
|
* *
|
|
* This function takes no arguments. It searches all open *
|
|
* libraries to resolve undefined externals. It does not *
|
|
* return a meaningful value. *
|
|
* *
|
|
****************************************************************/
|
|
|
|
void NEAR LibrarySearch(void)
|
|
{
|
|
RBTYPE vrpTmpFileFirst;
|
|
WORD ifhLibMacInit; /* Initial number of libs to search */
|
|
FTYPE searchMore; /* Search continue flag */
|
|
WORD bufpnSort[PNSORTMAX];
|
|
/* Actual space for pnSort */
|
|
SBTYPE sbLibname; /* Name of current library */
|
|
AHTEPTR pahte; /* Pointer to hash table entry */
|
|
REGISTER WORD i;
|
|
FTYPE fLibPass1 = (FTYPE) TRUE;
|
|
/* True if on 1st pass thru libs */
|
|
FTYPE *fUsedInPass1; /* True if lib used in 1st pass thru libs */
|
|
FTYPE fFirstTime; /* True if lib seen for the first time */
|
|
extern FTYPE fNoExtDic; /* True if /NOEXTDICTIONARY */
|
|
|
|
#if NEW_LIB_SEARCH
|
|
UND *pund; /* pointer in undef lookaside list */
|
|
UND *pundPrev; /* pointer to previous undef entry */
|
|
UND *pundNext; /* pointer to next undef entry */
|
|
#endif
|
|
|
|
|
|
fUndefsSeen = (FTYPE) TRUE; /* There are undefined externals */
|
|
vfLibOpen = FALSE; /* No libraries open yet */
|
|
pnSort = bufpnSort; /* Initialize sort table pointer */
|
|
ifhLibMacInit = ifhLibMac;
|
|
fUsedInPass1 = (FTYPE *) GetMem(ifhLibMac * sizeof(FTYPE));
|
|
if (fUsedInPass1 != NULL)
|
|
memset(fUsedInPass1, TRUE, ifhLibMac);
|
|
|
|
|
|
#if NEW_LIB_SEARCH
|
|
// build up the the lookaside list
|
|
|
|
EnSyms(StoreUndef,ATTRUND);
|
|
|
|
fStoreUndefsInLookaside = TRUE;
|
|
#endif
|
|
|
|
do /* Loop to search libraries */
|
|
{
|
|
searchMore = FALSE; /* Assume on final pass */
|
|
for(ifhLibCur = 0; ifhLibCur < ifhLibMac && FUndefsLeft(); ++ifhLibCur)
|
|
{ /* While undefs and libraries */
|
|
#if NEW_LIB_SEARCH
|
|
DWORD libMask = (1<<ifhLibCur);
|
|
|
|
if (pundListHead->dwLibMask & libMask)
|
|
continue; // no need to search this library
|
|
// the first item in the list has already
|
|
// been searched...
|
|
#endif
|
|
|
|
if(!GetLib())
|
|
continue;
|
|
|
|
/*
|
|
* If this is first pass through the libraries and /NOEXT was
|
|
* not given, try to get the extended dictionary. We assume that
|
|
* if there is one then only one library pass is needed.
|
|
*/
|
|
if(fLibPass1 && !fNoExtDic)
|
|
pExtDic = GetExtDic();
|
|
else
|
|
pExtDic = NULL;
|
|
/* If no extended dictionary, reduce buffer size because more
|
|
* seeking will be done. This will affect remaining libraries
|
|
* in search; we don't care about mixed extended and non-
|
|
* extended libraries.
|
|
*/
|
|
if(!pExtDic)
|
|
setvbuf(bsInput,bsInput->_base,_IOFBF,1024);
|
|
pahte = (AHTEPTR ) FetchSym(mpifhrhte[ifhLibCur],FALSE);
|
|
/* Get library name */
|
|
memcpy(sbLibname,GetFarSb(pahte->cch),B2W(pahte->cch[0])+1);
|
|
#if WIN_3 OR C8_IDE
|
|
sbLibname[B2W(*sbLibname)+1] = '\0';
|
|
#endif
|
|
#if WIN_3
|
|
StatMsgWin( "%s\r\n", sbLibname+1);
|
|
#endif
|
|
#if C8_IDE
|
|
if(fC8IDE)
|
|
{
|
|
sprintf(msgBuf, "@I4%s\r\n", sbLibname+1);
|
|
_write(fileno(stderr), msgBuf, strlen(msgBuf));
|
|
}
|
|
#endif
|
|
fFirstTime = (FTYPE) TRUE;
|
|
while(FUndefsLeft()) /* While there are undefs seen */
|
|
{
|
|
fFileExtracted = FALSE; /* Assume we won't take anything */
|
|
fUndefsSeen = FALSE; /* Assume no more undefs */
|
|
ipnMac = 0; /* Initialize sort table count */
|
|
|
|
#if NOT NEW_LIB_SEARCH
|
|
EnSyms(ProcessAnUndef,ATTRUND);
|
|
#else
|
|
|
|
pund = pundListHead;
|
|
pundPrev = NULL;
|
|
|
|
while (pund)
|
|
{
|
|
if (pund->dwLibMask & libMask)
|
|
{
|
|
break; // since items are added to the head,
|
|
// as soon as we find one item that has
|
|
// already been searched, the rest have
|
|
// also already been searched...
|
|
|
|
// pundPrev = pund;
|
|
// pund = pund->pNext;
|
|
// continue;
|
|
}
|
|
|
|
pundNext = pund->pNext;
|
|
|
|
if (pund->papropUndef->an_attr == ATTRUND)
|
|
ProcessAnUndef(pund->papropUndef, pund->rhte, 0, 0);
|
|
else
|
|
fUndefHit = TRUE; // no longer undefined -- remove
|
|
|
|
if (fUndefHit)
|
|
{
|
|
// remove this item from the undef list...
|
|
if (pundPrev)
|
|
pundPrev->pNext = pundNext;
|
|
else
|
|
pundListHead = pundNext;
|
|
|
|
pund->pNext = pundFree;
|
|
pundFree = pund;
|
|
|
|
}
|
|
else
|
|
{
|
|
pund->dwLibMask |= libMask;
|
|
pundPrev = pund;
|
|
}
|
|
|
|
pund = pundNext;
|
|
}
|
|
#endif
|
|
|
|
/* Try to resolve references */
|
|
/* If no modules obtained, exit loop. */
|
|
|
|
if(!ipnMac)
|
|
{
|
|
#if NEWIO
|
|
if (fLibPass1)
|
|
{
|
|
/*
|
|
* If this library is seen for the first time in
|
|
* the first pass thru libraries and we don't
|
|
* pull out any modules from it, then close this
|
|
* library, because there are big chances this
|
|
* library is not needed.
|
|
*/
|
|
|
|
if (fFirstTime)
|
|
{
|
|
_close(mpifhfh[ifhLibCur]);
|
|
mpifhfh[ifhLibCur] = 0;
|
|
/*
|
|
* Mark it also as not used in pass 1
|
|
* so, we can closed it also in the
|
|
* next passes thru libs.
|
|
*/
|
|
if (fUsedInPass1)
|
|
fUsedInPass1[ifhLibCur] = FALSE;
|
|
}
|
|
}
|
|
else if (fUsedInPass1 && !fUsedInPass1[ifhLibCur])
|
|
{
|
|
/*
|
|
* In pass "n" thru libs close libraries
|
|
* not used in pass 1.
|
|
*/
|
|
_close(mpifhfh[ifhLibCur]);
|
|
mpifhfh[ifhLibCur] = 0;
|
|
}
|
|
#endif
|
|
break;
|
|
}
|
|
fFirstTime = FALSE; /* No longer first time seen */
|
|
/* If extended dictionary present, process it. */
|
|
if(pExtDic)
|
|
ProcExtDic(pExtDic);
|
|
/* Sort modules by page offset. */
|
|
qsort(pnSort, ipnMac, sizeof(WORD),
|
|
(int (__cdecl *)(const void *, const void *)) FGtNum);
|
|
/*
|
|
* Save each module represented in the table.
|
|
*/
|
|
for (i = 0; i < ipnMac; i++)
|
|
{
|
|
/*
|
|
* If SaveInput returns 0, the module was already seen. See
|
|
* above comment in ProcessAnUndef().
|
|
*/
|
|
if(!SaveInput(sbLibname, (long)pnSort[i] << libAlign, ifhLibCur, 0))
|
|
continue;
|
|
if(!fFileExtracted) /* If no files extracted yet */
|
|
{
|
|
vrpNewList = vrpropTailFile;
|
|
/* Save start of file list */
|
|
fFileExtracted = (FTYPE) TRUE;
|
|
/* We have extracted a file */
|
|
}
|
|
}
|
|
if(!fFileExtracted)
|
|
break; /* If we didn't take anything, break */
|
|
/* Library might not be open because we may have searched
|
|
* an already-loaded dictionary. If necessary, re-open
|
|
* library.
|
|
*/
|
|
if(!vfLibOpen)
|
|
GetLib();
|
|
searchMore = (FTYPE) TRUE; /* Otherwise it's worth another pass */
|
|
vrpTmpFileFirst = rprop1stFile;
|
|
/* Save head of module list */
|
|
rprop1stFile = vrpNewList;
|
|
/* Put new modules at head of list */
|
|
fLibPass = (FTYPE) TRUE; /* Processing object from library */
|
|
DrivePass(ProcP1); /* Do pass 1 on object from library */
|
|
fLibPass = FALSE; /* No longer processing lib. object */
|
|
rprop1stFile = vrpTmpFileFirst;
|
|
/* Restore original head of list */
|
|
if (fUsedInPass1 && ifhLibMacInit < ifhLibMac)
|
|
{
|
|
/* DrivePass added more libraries to search */
|
|
/* Reallocate fUsedInPass1 */
|
|
|
|
FTYPE *p; /* Temporary pointer */
|
|
|
|
p = (FTYPE *) GetMem(ifhLibMac * sizeof(FTYPE));
|
|
if (p == NULL)
|
|
{
|
|
FFREE(fUsedInPass1);
|
|
fUsedInPass1 = NULL;
|
|
}
|
|
else
|
|
{
|
|
memset(p, TRUE, ifhLibMac);
|
|
memcpy(p, fUsedInPass1, ifhLibMacInit);
|
|
FFREE(fUsedInPass1);
|
|
fUsedInPass1 = p;
|
|
}
|
|
ifhLibMacInit = ifhLibMac;
|
|
}
|
|
}
|
|
/* Free space for extended dictionary if present */
|
|
if(pExtDic)
|
|
FFREE(pExtDic);
|
|
if(vfLibOpen)
|
|
{
|
|
#if NOT NEWIO
|
|
fclose(bsInput); /* Close the library */
|
|
#endif
|
|
vfLibOpen = FALSE; /* No library open */
|
|
}
|
|
}
|
|
/* No longer on 1st pass thru libraries. */
|
|
fLibPass1 = FALSE;
|
|
}
|
|
while(searchMore && FUndefsLeft()); /* Do until search done */
|
|
FreeMem(fUsedInPass1);
|
|
FreeDictionary(); /* Free dictionary space */
|
|
/*
|
|
* Restore large buffer size in case it was reduced.
|
|
*/
|
|
setvbuf(bsInput,bsInput->_base,_IOFBF,LBUFSIZ);
|
|
|
|
#if NEW_LIB_SEARCH
|
|
|
|
fStoreUndefsInLookaside = FALSE;
|
|
|
|
while (pundpoolHead)
|
|
{
|
|
pundpoolCur = pundpoolHead->pNext;
|
|
FFREE(pundpoolHead);
|
|
pundpoolHead = pundpoolCur;
|
|
}
|
|
|
|
#endif
|
|
}
|
|
|
|
#if CMDMSDOS
|
|
/*
|
|
* GetLibAll:
|
|
*
|
|
* Process all the modules in a given library in Pass 1.
|
|
* Create property cells for them and insert into the file list.
|
|
*/
|
|
|
|
void NEAR GetLibAll(sbLib)
|
|
BYTE *sbLib;
|
|
{
|
|
WORD ifh; /* (fake) library index */
|
|
long lfa; /* Current file offset */
|
|
IOVTYPE iov; /* Overlay number */
|
|
RBTYPE rbFileNext; /* Pointer to next file property */
|
|
RBTYPE rbFileNew; /* Pointer to new file property */
|
|
APROPFILEPTR apropFile, apropFilePrev;
|
|
BYTE *sbInput; /* Asciiz filename */
|
|
int fh; /* File handle */
|
|
|
|
fDrivePass = FALSE;
|
|
sbInput = sbLib + 1;
|
|
/* Get the ifh, iov, and pointer to the next file from the current
|
|
* file pointer.
|
|
*/
|
|
apropFile = (APROPFILEPTR ) FetchSym(vrpropFile,TRUE);
|
|
ifh = apropFile->af_ifh;
|
|
iov = apropFile->af_iov;
|
|
rbFileNext = apropFile->af_FNxt;
|
|
#if NEWIO
|
|
fh = SmartOpen(sbInput,ifh);
|
|
if (fh <= 0 && lpszLIB != NULL)
|
|
fh = SearchPathLink(lpszLIB, sbInput, ifh, TRUE);
|
|
|
|
if (fh > 0)
|
|
{
|
|
fflush(bsInput);
|
|
bsInput->_file = (char) fh;
|
|
}
|
|
else
|
|
Fatal(ER_fileopn,sbInput);
|
|
|
|
#else
|
|
if((bsInput = fopen(sbInput,RDBIN)) == NULL)
|
|
Fatal(ER_fileopn,sbInput);
|
|
#endif
|
|
if(getc(bsInput) != LIBHDR) /* Check for valid record type */
|
|
Fatal(ER_badlib,sbInput);
|
|
cbRec = (WORD) (3 + WSGets()); /* Get record length */
|
|
for(libAlign = 15; libAlign && !(cbRec & (1 << libAlign)); --libAlign);
|
|
/* Calculate alignment factor */
|
|
fDrivePass = (FTYPE) TRUE;
|
|
/* Reset current file's lfa from 0 to offset of 1st module */
|
|
apropFile->af_lfa = lfa = 1L << libAlign;
|
|
/* Go to the first module */
|
|
if (fseek(bsInput,lfa,0))
|
|
Fatal(ER_badlib,sbInput);
|
|
/* Process the library as follows: Process the current module.
|
|
* Go to the next module; if it starts with DICHDR then we're
|
|
* done. Else, create a new file property cell for the next
|
|
* module, insert it in the file list, and go to start of loop.
|
|
*/
|
|
|
|
rect = (WORD) getc(bsInput);
|
|
while (rect != DICHDR)
|
|
{
|
|
ungetc(rect, bsInput);
|
|
lfaLast = apropFile->af_lfa = ftell(bsInput);
|
|
ProcP1();
|
|
while (TYPEOF(rect) != MODEND)
|
|
{
|
|
rect = (WORD) getc(bsInput);
|
|
if (fseek(bsInput, (cbRec = WSGets()), 1))
|
|
Fatal(ER_badlib,sbInput);
|
|
}
|
|
|
|
do
|
|
{
|
|
rect = (WORD) getc(bsInput);
|
|
}
|
|
while (rect != THEADR && rect != DICHDR && rect != EOF);
|
|
if (rect == DICHDR)
|
|
{
|
|
if (rbFileNext == RHTENIL)
|
|
vrpropTailFile = vrpropFile;
|
|
#if NOT NEWIO
|
|
fclose(bsInput);
|
|
#else
|
|
rbFilePrev = vrpropFile;
|
|
#endif
|
|
return;
|
|
}
|
|
if (rect == EOF)
|
|
Fatal(ER_libeof);
|
|
|
|
// Make a new file property cell
|
|
|
|
apropFile = (APROPFILEPTR ) PropAdd(vrhteFile, ATTRFIL);
|
|
rbFileNew = vrprop;
|
|
#if ILINK
|
|
apropFile->af_imod = ++imodCur; // allocate a module number
|
|
apropFile->af_cont = 0;
|
|
apropFile->af_ientOnt = 0;
|
|
#endif
|
|
apropFile->af_rMod = 0;
|
|
apropFile->af_ifh = (char) ifh;
|
|
apropFile->af_iov = (IOVTYPE) iov;
|
|
apropFile->af_FNxt = rbFileNext;
|
|
#if SYMDEB
|
|
apropFile->af_publics = NULL;
|
|
apropFile->af_Src = NULL;
|
|
apropFile->af_SrcLast = NULL;
|
|
apropFile->af_cvInfo = NULL;
|
|
#endif
|
|
apropFile->af_ComDat = 0L;
|
|
apropFile->af_ComDatLast = 0L;
|
|
MARKVP();
|
|
|
|
// Get the just-processed property file cell
|
|
|
|
apropFilePrev = (APROPFILEPTR ) FetchSym(vrpropFile,TRUE);
|
|
apropFilePrev->af_FNxt = rbFileNew;
|
|
vrpropFile = rbFileNew;
|
|
};
|
|
|
|
// Remove an empty Lib from the chain of files
|
|
|
|
if (vrpropFile == rprop1stFile)
|
|
{
|
|
// If the empty lib is first on list
|
|
|
|
rprop1stFile = rbFileNext;
|
|
}
|
|
else
|
|
{
|
|
#if NEWIO
|
|
apropFilePrev = (APROPFILEPTR)FetchSym(rbFilePrev, TRUE);
|
|
apropFilePrev->af_FNxt = apropFile->af_FNxt;
|
|
#endif
|
|
}
|
|
#if NEWIO
|
|
if (rbFileNext == RHTENIL)
|
|
vrpropTailFile = rbFilePrev; // In case we removed the last file
|
|
_close(fileno(bsInput));
|
|
rbFilePrev = vrpropFile;
|
|
#endif
|
|
}
|
|
#endif /*CMDMSDOS*/
|