2020-09-30 16:53:55 +02:00

718 lines
21 KiB
C

/* %W% %E% */
/*
* Copyright Microsoft Corporation, 1983-1987
*
* This Module contains Proprietary Information of Microsoft
* Corporation and should be treated as Confidential.
*/
/****************************************************************
* *
* LINKER INITIALIZATION *
* *
****************************************************************/
#include <minlit.h> /* Types and constants */
#include <bndtrn.h> /* More of the same */
#include <bndrel.h> /* More of the same */
#include <lnkio.h> /* Linker I/O definitions */
#include <newexe.h> /* DOS & 286 .EXE format definitions */
#if EXE386
#include <exe386.h> /* 386 .EXE format definitions */
#endif
#include <signal.h> /* Signal definitions */
#if QCLINK
#include <stdlib.h>
#endif
#include <lnkmsg.h> /* Error messages */
#if OSMSDOS AND NOT (WIN_NT OR DOSEXTENDER OR DOSX32) AND NOT WIN_3
#define INCL_BASE
#define INCL_DOSMISC
#include <os2.h> /* OS/2 system calls */
#if defined(M_I86LM)
#undef NEAR
#define NEAR
#endif
#endif
#include <extern.h> /* External declarations */
#include <impexp.h>
#include <direct.h>
#if defined(DOSX32) OR defined(WIN_NT)
extern char FAR * _stdcall GetCommandLineA(void);
#endif
/*
* FUNCTION PROTOTYPES
*/
LOCAL void NEAR InitLeadByte(void);
LOCAL void NEAR SetupEnv(void);
LOCAL int NEAR IsPrefix(BYTE *pszPrefix, BYTE *pszString);
#if TCE
extern SYMBOLUSELIST aEntryPoints; // List of program entry points
#endif
#if ECS
/*
* InitLeadByte
*
* Initialize lead byte table structures.
* Returns no meaningful value.
*/
LOCAL void NEAR InitLeadByte ()
{
struct lbrange
{
unsigned char low; /* minimum */
unsigned char high; /* maximum */
};
static struct lbrange lbtab[5] = { { 0, 0 } };
struct lbrange *ptab;
WORD i; /* index */
COUNTRYCODE cc; /* country code */
cc.country = cc.codepage = 0;
if (DosGetDBCSEv(sizeof(lbtab), &cc, (char FAR *)lbtab))
return;
// For each range, set corresponding entries in fLeadByte
for (ptab = lbtab; ptab->low || ptab->high; ptab++)
if (ptab->low >= 0x80)
for (i = ptab->low; i <= ptab->high; i++)
fLeadByte[i-0x80] = (FTYPE) TRUE;
// Mark inclusive range true
}
#endif /* ECS */
#if NOT (WIN_NT OR DOSX32)
/*** _setenvp - stub for C run-time
*
* Purpose:
* Call stub instead of real function, we don't want C run-time to
* setup enviroment.
*
* Input:
* None;
*
* Output:
* None;
*
* Exceptions:
* None.
*
* Notes:
* None.
*
*************************************************************************/
void cdecl _setenvp(void)
{
return;
}
/*** IsPrefix - self-explanatory
*
* Purpose:
* Check if one string is a prefix of another.
*
* Input:
* pszPrefix - pointer to prefix string
* pszString - the string
*
* Output:
* The function returns TRUE if the first string is a prefix of the
* second; otherwise it returns FALSE.
*
* Exceptions:
* None.
*
* Notes:
* None.
*
*************************************************************************/
LOCAL int NEAR IsPrefix(BYTE *pszPrefix, BYTE *pszString)
{
while(*pszPrefix) // While not at end of prefix
{
if (*pszPrefix != *pszString) return(FALSE);
// Return zero if mismatch
++pszPrefix; // Increment pointer
++pszString; // Increment pointer
}
return(TRUE); // We have a prefix
}
#endif
/*** SetupEnv - set up pointer to linker evironment variables
*
* Purpose:
* Every byte in the DGROUP is to valuable to waste it to hold
* information available elswere in the memory linker is not using
* C run-time GETENV function, which accesses copy of the entire
* environemt in the DGROUP placed there by the startup code.
* Insted this function scans enviroment and set up pointers to
* appropriate strings. Because initially enviroment is in the FAR
* memory no space in DGROUP is used.
*
* Input:
* No explicit parameters are passed.
*
* Output:
* Four global pointer set to appropriate enviroment strings
*
* lpszLink - the LINK
* lpszPath - the PATH
* lpszTMP - the TMP
* lpszLIB - the LIB
* lpszQH - the QH for QuickHelp
*
* Exceptions:
* None.
*
* Notes:
* None.
*
*************************************************************************/
LOCAL void NEAR SetupEnv(void)
{
#if WIN_NT OR DOSX32
#if C8_IDE
char * pIDE = getenv("_MSC_IDE_FLAGS");
if(pIDE)
{
if(strstr(pIDE, "FEEDBACK"))
{
fC8IDE = TRUE;
#if DEBUG_IDE
fprintf(stdout, "\r\nIDE ACTIVE - FEEDBACK is ON");
#endif
}
if(strstr(pIDE, "BATCH"))
{
// NOTE: The link response file will still be echoed in this case!
// NOTE: this is different than if you specify /BATCH on the
// NOTE: command line -- also, the banner is still displayed
// NOTE: this is intentional as the IDE wants to BATCH to cause
// NOTE: the linker not to prompt, but it does want the banner
// NOTE: and response file echoed unless /NOLOGO is also specified
// NOTE: see CAVIAR 2378 [rm]
fNoprompt = (FTYPE) TRUE;
fPauseRun = FALSE; /* Disable /PAUSE */
#if DEBUG_IDE
fprintf(stdout, "\r\nIDE ACTIVE - BATCH is ON");
#endif
}
if(strstr(pIDE, "NOLOGO"))
{
fNoBanner = (FTYPE) TRUE;
#if DEBUG_IDE
fprintf(stdout, "\r\nIDE ACTIVE - LOGO is OFF");
#endif
}
}
#if DEBUG_IDE
else
fprintf(stdout, "\r\nIDE NOT ACTIVE");
fflush(stdout);
#endif
#endif // C8_IDE
lpszPath = getenv("PATH");
lpszLink = getenv("LINK");
lpszTMP = getenv("TMP");
lpszLIB = getenv("LIB");
lpszQH = getenv("QH");
lpszHELPFILES = getenv("HELPFILES");
lpszCmdLine = GetCommandLineA();
while (*lpszCmdLine != ' ')
lpszCmdLine++;
#else
WORD selEnv;
WORD cmdOffset;
register WORD offMac;
char FAR *lpszEnv;
char FAR *lpch;
SBTYPE buf;
register WORD ich;
WORD fEOS;
#if QCLINK OR CPU8086 OR DOSEXTENDER
// Get the segment address of the environment block
// and set the command line offset to infinity. We
// stop scanning environment block at NULL string.
lpszEnv = (char FAR *)
(((long) _psp << 16)
+ 0x2c);
selEnv = *((WORD FAR *) lpszEnv);
lpszCmdLine = (char FAR *)(((long) _psp << 16) + 0x80);
lpszCmdLine[lpszCmdLine[0] + 1] = '\0';
lpszCmdLine++;
cmdOffset = 0xffff;
#else
if (DosGetEnv((unsigned FAR *) &selEnv, (unsigned FAR *) &cmdOffset))
return;
#endif
lpszEnv = (char FAR *)((long) selEnv << 16);
#if NOT (QCLINK OR CPU8086 OR DOSEXTENDER)
lpszCmdLine = lpszEnv + cmdOffset;
// Skip LINK
lpszCmdLine += _fstrlen(lpszCmdLine) + 1;
#endif
// Skip leading spaces in command line
while (*lpszCmdLine == ' ')
lpszCmdLine++;
lpch = lpszEnv;
for (offMac = 0; offMac < cmdOffset && *lpszEnv; )
{
// Copy the enviroment variable string into near buffer
ich = 0;
while (*lpch && ich < sizeof(buf) - 1)
buf[ich++] = *lpch++;
if (*lpch == '\0')
{
// Skip over terminating zero
lpch++;
fEOS = TRUE;
}
else
fEOS = FALSE;
buf[ich] = '\0';
// Check what it is and setup appropriate pointer
if (lpszPath == NULL && IsPrefix((BYTE *) "PATH=", buf))
lpszPath = lpszEnv + 5;
else if (lpszLink == NULL && IsPrefix((BYTE *) "LINK=", buf))
lpszLink = lpszEnv + 5;
else if (lpszTMP == NULL && IsPrefix((BYTE *) "TMP=", buf))
lpszTMP = lpszEnv + 4;
else if (lpszLIB == NULL && IsPrefix((BYTE *) "LIB=", buf))
lpszLIB = lpszEnv + 4;
else if (lpszQH == NULL && IsPrefix((BYTE *) "QH=", buf))
lpszQH = lpszEnv + 3;
else if (lpszHELPFILES == NULL && IsPrefix((BYTE *) "HELPFILES=", buf))
lpszHELPFILES = lpszEnv + 10;
// If everything setup don't bother to look father
if (lpszPath && lpszLink && lpszTMP && lpszLIB && lpszQH && lpszHELPFILES)
break;
// Update enviroment pointer and offset in enviroment segment
offMac += ich;
if (!fEOS)
{
// Oops ! - enviroment variable longer then buffer
// skip to its end
while (*lpch && offMac < cmdOffset)
{
lpch++;
offMac++;
}
// Skip over terminating zero
lpch++;
offMac++;
}
lpszEnv = lpch;
}
#endif
}
#if FALSE
/*** Dos3SetMaxFH - set max file handle count for DOS
*
* Purpose:
* Sets the maximum number of files that may be opened
* simultaneously using handles by the linker.
*
* Input:
* cFH - number of desired handles
*
* Output:
* No explicit value is returned.
*
* Exceptions:
* None.
*
* Notes:
* This function uses the int 21h function 67h which available on
* DOS 3.3 and higher. The function fails if the requested number of
* handles is greater then 20 and there is not sufficient free memory
* in the system to allocate a new block to hold the enlarged table.
*
* If the number of handles requested is larger the available
* entries in the system's global table for file handles (controlled
* by the FILES entry in CONFIG.SYS), no error is returned.
* However, a subsequent attempt to open a file or create a new
* file will fail if all entries in the system's global file table
* are in use, even if the requesting process has not used up all
* of its own handles
*
* We don't check for error, because we can't do much about it.
* Linker will try to run with what is available.
*
*************************************************************************/
LOCAL void NEAR Dos3SetMaxFH(WORD cFH)
{
if ((_osmajor >= 3) && (_osminor >= 30))
{
_asm
{
mov ax, 0x6700
mov bx, cFH
int 0x21
}
}
}
#endif
/****************************************************************
* *
* InitializeWorld: *
* *
* This function takes no arguments and returns no meaningful *
* value. It sets up virtual memory, the symbol table *
* Handlers, and it initializes segment structures. *
* *
****************************************************************/
void InitializeWorld(void)
{
#if OSMSDOS
BYTE buf[512]; /* Temporary buffer */
char FAR *lpch; /* Temporary pointer */
int i; /* Temporary index */
#endif
#if NOT (FIXEDSTACK OR CPU386)
InitStack(); /* Initialize stack */
#endif
#if OSMSDOS
DskCur = (BYTE) (_getdrive() - 1); /* Get current (default) disk drive */
#if FALSE
if(!isatty(fileno(stderr))) /* No prompts if output not console */
fNoprompt = TRUE;
#endif
#if CRLF
/* Default mode of stdout, stdin, stderr is text, change to binary. */
_setmode(fileno(stdout),O_BINARY);
if(stderr != stdout)
_setmode(fileno(stderr),O_BINARY);
_setmode(fileno(stdin),O_BINARY);
#endif
#endif
InitSym(); /* Initialize symbol table handler */
DeclareStdIds();
// Install CTRL-C handler
#if OSMSDOS AND NOT WIN_NT
signal(SIGINT, (void (__cdecl *)(int)) UserKill);
#endif /* OSMSDOS */
#if OSXENIX
if(signal(SIGINT,UserKill) == SIG_IGN) signal(SIGINT,SIG_IGN);
/* Trap user interrupts */
if(signal(SIGHUP,UserKill) == SIG_IGN) signal(SIGHUP,SIG_IGN);
/* Trap hangup signal */
if(signal(SIGTERM,UserKill) == SIG_IGN) signal(SIGTERM,SIG_IGN);
/* Trap software termination */
#endif
#if SYMDEB
InitDbRhte();
#endif
#if ECS
InitLeadByte(); /* Initialize lead byte table */
#endif
#if OSMSDOS
// Initialize LINK environment.
// Do it yourself to save the memory.
SetupEnv();
/* Process switches from LINK environment variable */
if (lpszLink != NULL)
{
lpch = lpszLink;
/* Skip leading whitespace. */
while(*lpch == ' ' || *lpch == '\t')
lpch++;
if(*lpch++ == CHSWITCH)
{
// If string begins with switchr
// Copy string to buf, removing whitespace
for (i = 1; *lpch && i < sizeof(buf); lpch++)
if (*lpch != ' ' && *lpch != '\t')
buf[i++] = *lpch;
buf[0] = (BYTE) (i - 1); /* Set the length of buf */
if(buf[0]) /* If any switches, process them */
BreakLine(buf,ProcFlag,CHSWITCH);
}
}
#endif
#if CPU286
if (_osmode == OS2_MODE)
{
DosSetMaxFH(128); /* This is the same as _NFILE in crt0dat.asm */
DosError(EXCEPTION_DISABLE);
}
#if FALSE
else
Dos3SetMaxFH(40);
#endif
#endif
#if FALSE AND CPU8086
Dos3SetMaxFH(40);
#endif
// Initialize import/export tables
InitByteArray(&ResidentName);
InitByteArray(&NonResidentName);
InitByteArray(&ImportedName);
ImportedName.byteMac++; // Ensure non-zero offsets to imported names
InitWordArray(&ModuleRefTable);
InitByteArray(&EntryTable);
#if TCE
aEntryPoints.cMaxEntries = 64;
aEntryPoints.pEntries = (RBTYPE*)GetMem(aEntryPoints.cMaxEntries * sizeof(RBTYPE*));
#endif
}
#if (OSXENIX OR OSMSDOS OR OSPCDOS) AND NOT WIN_NT
/****************************************************************
* *
* UserKill: *
* *
* Clean up if linker killed by user. *
* *
****************************************************************/
void cdecl UserKill()
{
signal(SIGINT, SIG_IGN); /* Disallow ctrl-c during handler */
CtrlC();
}
#endif
/*
* InitTabs:
*
* Initialize tables required in Pass 1.
*/
void InitTabs(void)
{
#if NOT FAR_SEG_TABLES
char *tabs; /* Pointer to table space */
unsigned cbtabs;
#endif
/* Initialize the following tables:
*
* NAME TYPE
* ---- ----
* mpsegraFirst RATYPE
* mpgsnfCod FTYPE
* mpgsndra RATYPE
* mpgsnrprop RBTYPE
* mplnamerhte RBTYPE
*/
#if FAR_SEG_TABLES
mplnamerhte = (RBTYPE FAR *) GetMem(lnameMax * sizeof(RBTYPE));
mpsegraFirst = (RATYPE FAR *) GetMem(gsnMax * sizeof(RATYPE));
mpgsnfCod = (FTYPE FAR *) mpsegraFirst; /* Use same space twice */
mpgsndra = (RATYPE FAR *) GetMem(gsnMax * sizeof(RATYPE));
mpgsnrprop = (RBTYPE FAR *) GetMem(gsnMax * sizeof(RBTYPE));
#else
mplnamerhte = (RBTYPE *) malloc(lnameMax * sizeof(RBTYPE));
if (mplnamerhte == NULL)
Fatal(ER_seglim);
memset(mplnamerhte, 0, lnameMax * sizeof(RATYPE));
cbtabs = gsnMax * (sizeof(RATYPE) + sizeof(RATYPE) + sizeof(RBTYPE));
if((tabs = malloc(cbtabs)) == NULL)
Fatal(ER_seglim);
memset(tabs,0,cbtabs); /* Clear everything */
mpsegraFirst = (RATYPE *) tabs; /* Initialize base */
mpgsnfCod = (FTYPE *) mpsegraFirst; /* Use same space twice */
mpgsndra = (RATYPE *) &mpsegraFirst[gsnMax];
mpgsnrprop = (RBTYPE *) &mpgsndra[gsnMax];
#endif
}
/*
* InitP2Tabs:
*
* Initialize tables not needed until Pass 2.
*/
void InitP2Tabs (void)
{
char FAR *tabs; /* Pointer to table space */
unsigned cbtabs; /* Size of table space */
unsigned TabSize;
TabSize = gsnMac + iovMac + 1;
/* Tables required regardless of exe format generated:
* mpsegsa SATYPE
* mpgsnseg SEGTYPE
*/
#if FAR_SEG_TABLES
cbtabs = 0;
mpsegsa = (SATYPE FAR *) GetMem(TabSize * sizeof(SATYPE));
mpgsnseg = (SEGTYPE FAR *) GetMem(TabSize * sizeof(SEGTYPE));
mpseggsn = (SNTYPE FAR *) GetMem(TabSize * sizeof(SNTYPE));
#else
cbtabs = TabSize * (sizeof(RATYPE) + sizeof(SATYPE));
#endif
/* Tables required according to exe format generated:
*
* DOS 3:
* mpsegcb[TabSize] long
* mpsegFlags[TabSize] FTYPE
* mpsegalign[TabSize] ALIGNTYPE
* mpsegiov[TabSize] IOVTYPE
* mpiovRlc[iovMac] RUNRLC
* Seg. exe:
* mpsacb[SAMAX] long
* mpsadraDP[SAMAX] long; for O68K
* mpsacbinit[SAMAX] long
* mpsaRlc[SAMAX] HASHRLC FAR *
* mpsaflags[SAMAX] WORD; DWORD for EXE386
* htsaraep[HEPLEN] EPTYPE FAR *
* X.out:
* mpsegcb[TabSize] long
* mpsegFlags[TabSize] FTYPE
* mpstsa[TabSize] SATYPE
*/
#if EXE386
if(fNewExe)
cbtabs += (SAMAX*(sizeof(long)+sizeof(long)+sizeof(DWORD)+
sizeof(DWORD))) + (HEPLEN * sizeof(WORD));
else
#else
if(fNewExe)
#if O68K
cbtabs += (SAMAX*(sizeof(long)+sizeof(long)+sizeof(long)+sizeof(WORD)+
sizeof(WORD))) + (HEPLEN * sizeof(WORD));
#else
cbtabs += (SAMAX*(sizeof(long)+sizeof(long)+sizeof(RLCHASH FAR *) +
sizeof(WORD))) + (HEPLEN * sizeof(EPTYPE FAR *));
#endif
else
#endif
#if OEXE
cbtabs += TabSize * (sizeof(long) + sizeof(FTYPE) + sizeof(ALIGNTYPE));
#else
cbtabs += TabSize * (sizeof(long) + sizeof(FTYPE) + sizeof(SATYPE));
#endif
cbtabs += sizeof(WORD);
tabs = GetMem(cbtabs);
#if NOT FAR_SEG_TABLES
mpgsnseg = (SEGTYPE *)tabs;
mpsegsa = (SATYPE *)&mpgsnseg[TabSize];
tabs = (char *)&mpsegsa[TabSize];
#endif
#if OSEGEXE
if(fNewExe)
{
mpsacb = (DWORD FAR *) tabs;
#if O68K
mpsadraDP = (long *)&mpsacb[SAMAX];
mpsacbinit = (long *)&mpsadraDP[SAMAX];
#else
mpsacbinit = (DWORD FAR *)&mpsacb[SAMAX];
#endif
#if EXE386
mpsacrlc = (DWORD *)&mpsacbinit[SAMAX];
mpsaflags = (DWORD *)&mpsacrlc[SAMAX];
#else
mpsaRlc = (RLCHASH FAR * FAR *) &mpsacbinit[SAMAX];
mpsaflags = (WORD FAR *) &mpsaRlc[SAMAX];
#endif
htsaraep = (EPTYPE FAR * FAR *)&mpsaflags[SAMAX];
}
else
#endif
{
#if ODOS3EXE OR OIAPX286
mpsegcb = (long FAR *) tabs;
mpsegFlags = (FTYPE FAR *)&mpsegcb[TabSize];
#if OEXE
mpsegalign = (ALIGNTYPE FAR *)&mpsegFlags[TabSize];
#if OVERLAYS
cbtabs = iovMac * sizeof(RUNRLC) + TabSize * sizeof(IOVTYPE) +
(sizeof(DWORD) - 1); // leave room to align mpiovRlc
mpsegiov = (IOVTYPE FAR*) GetMem(cbtabs);
// align mpiovRlc on a DWORD, the alignment needed by struct _RUNRLC
mpiovRlc = (RUNRLC FAR*) ( ( (__int64)&mpsegiov[TabSize] +
(sizeof(DWORD) - 1)
) & ~(sizeof(DWORD) - 1)
);
#endif
#endif
#if OIAPX286
mpstsa = (SATYPE *)&mpsegFlags[TabSize];
#endif
#endif /* ODOS3EXE OR OIAPX286 */
}
/* Attempt to allocate space for mpextprop. */
cbtabs = extMax * sizeof(RBTYPE);
mpextprop = (RBTYPE FAR *) GetMem(cbtabs);
}