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

1804 lines
53 KiB
C

#include <dos.h>
#include <stdio.h>
#include <sys\types.h>
#include <sys\stat.h>
#include <io.h>
#include <fcntl.h>
#include <string.h>
#include <ctype.h>
#include <malloc.h>
//#include <tchar.h>
#ifdef RLDOS
#include "dosdefs.h"
#else
#include <windows.h>
#include "windefs.h"
#endif
#include "restok.h"
#include "exe2res.h"
#include "newexe.h"
/* ----- Function prototypes ----- */
static void PrepareFiles( PSTR, PSTR, PSTR);
static void ReadSegmentTable( void);
static void ComputeResTableSize( void);
static void ComputeStringOffsets( void);
static void BuildResTable( void);
static void SegsWrite( WORD);
static DWORD RelocCopy( WORD);
static void ResWrite( WORD);
static void SetEXEHeaderFlags( void);
static void RewriteTables( void);
static void CopyCodeViewInfo( FILE *, FILE *);
static void OutPutError( char *);
static void ResTableBufferInit( WORD);
static void ResTableBufferFree( void);
static WORD GetAlign( DWORD, WORD);
static LONG MoveFilePos( FILE *, WORD, WORD);
static WORD RoundUp( LONG, WORD);
static WORD AlignFilePos( FILE *, WORD, BOOL);
static WORD ReadExeOldHeader( FILE *, LONG, LONG *) ;
static WORD ReadExeNewHeader( FILE *, LONG, LONG, LONG *);
static WORD ExtractExeResources( FILE *, FILE *, LONG , LONG);
static void ExtractString( FILE *, FILE *, LONG);
static WORD WriteResFromExe( FILE *,
FILE *,
LONG,
RESNAMEINFO,
RESTYPEINFO,
WORD);
static TYPINFO *AddResType( CHAR *, WORD);
static PSTR MyMakeStr( PSTR);
static SHORT MyRead( FILE *, PSTR, WORD);
static SHORT MyWrite( FILE *, PSTR, WORD);
static LONG MySeek( FILE *, LONG, WORD);
static void MyCopy( FILE *, FILE *, DWORD);
static int ProcessBinFile( void);
static PSTR RcAlloc( WORD);
static void AddResToResFile( TYPINFO *, RESINFO *);
static void AddDefaultTypes( void);
static void GetOrdOrName( unsigned int *, unsigned char *);
/* ----- Version functions (added for 3.1) ----- */
static void RcPutWord( unsigned int);
static int RcPutString( char *);
/* ----- Module variables ----- */
static struct exe_hdr OldExe;
static struct new_exe NewExe;
static struct new_seg *pSegTable;
static PSTR pResTable;
static PSTR pResNext;
static FILE * fhInput;
static FILE * fhOutput;
static FILE * fhBin;
static DWORD dwMaxFilePos;
static DWORD dwNewExe;
static WORD wPreloadOffset;
static WORD wPreloadLength;
static WORD wResTableOffset;
static WORD wSegTableLen;
static WORD wResTableLen;
static BYTE zeros[ NUMZEROS] = "";
static DWORD dwExeEndFile;
static WORD fMultipleDataSegs;
//.........................................................................
int ExtractResFromExe16A( CHAR *szInputExe, CHAR *szOutputRes, WORD wFilter)
{
QuitT( IDS_NO16RESWINYET, NULL, NULL);
#ifdef 0
WORD wResult = (WORD)-1;
LONG lPosNewHdr;
LONG lPosResourceTable;
LONG lFileLen;
FILE *fExeFile;
FILE *fResFile;
struct _stat ExeStats;
/* initialize */
wResult = IDERR_SUCCESS;
/* open file for reading */
if ((fExeFile = FOPEN(szInputExe, "rb" )) == NULL ) {
wResult = IDERR_OPENFAIL;
}
if ((fResFile = FOPEN(szOutputRes, "wb" )) == NULL ) {
wResult = IDERR_OPENFAIL;
}
/* get file length */
if (wResult == IDERR_SUCCESS) {
_stat(szInputExe , &ExeStats );
lFileLen = ExeStats.st_size;
}
/* read old header, verify contents, and get positon of new header */
if (wResult == IDERR_SUCCESS) {
wResult = ReadExeOldHeader( fExeFile, lFileLen, &lPosNewHdr );
}
/* read new header, verify contents, & get position of resource table */
if (wResult == IDERR_SUCCESS)
wResult = ReadExeNewHeader(
fExeFile,
lFileLen,
lPosNewHdr,
&lPosResourceTable
);
wResult = ExtractExeResources( fExeFile , fResFile, lPosResourceTable , lFileLen);
return ( wResult);
#endif // 0
}
//....................................................................
int BuildExeFromRes16A(
CHAR *pstrDest,
CHAR *pstrRes,
CHAR *pstrSource )
{
QuitT( IDS_NO16WINRESYET, NULL, NULL);
#ifdef 0
SHORT nResTableDelta;
fSortSegments = TRUE;
/* Get a memory block to use for MyCopy\(\) */
PrepareFiles(pstrSource, pstrDest, pstrRes);
ProcessBinFile();
/* Read the segment table */
ReadSegmentTable();
/* Compute the length of the resource table */
ComputeResTableSize();
/* Compute string offsets for non-ordinal type and resource names */
ComputeStringOffsets();
/* Build the resource table */
BuildResTable();
/* Now go back to the beginning */
MySeek(fhInput, 0L, 0);
/* Copy from input to output up to the segment table */
MyCopy(fhInput, fhOutput, dwNewExe + (DWORD)NewExe.ne_segtab);
/* Copy the segment table */
MyCopy(fhInput, fhOutput, (long)wSegTableLen);
/* Save a pointer to the start of the resource table */
wResTableOffset = (unsigned)(MySeek(fhOutput, 0L, 1) - dwNewExe);
/* Write our resource table out */
if (wResTableLen) {
MyWrite(fhOutput, pResTable, wResTableLen);
}
/* Now we\'re looking at the beginning of the resident name table */
MySeek(fhInput, (LONG)NewExe.ne_restab + dwNewExe, 0);
/* Copy all the other tables \(they must fall between the resident
* names table and the non-resident names table.
*/
MyCopy(fhInput, fhOutput,
NewExe.ne_nrestab - (LONG)NewExe.ne_restab - dwNewExe);
/* Copy the nonresident name table as well */
MyCopy(fhInput, fhOutput, (LONG)NewExe.ne_cbnrestab);
/* Compute new pointers in new exe header */
NewExe.ne_rsrctab = wResTableOffset;
nResTableDelta = wResTableOffset + wResTableLen - NewExe.ne_restab;
NewExe.ne_restab += nResTableDelta;
NewExe.ne_modtab += nResTableDelta;
NewExe.ne_imptab += nResTableDelta;
NewExe.ne_enttab += nResTableDelta;
NewExe.ne_nrestab += nResTableDelta;
#ifdef VERBOSE
/* Tell the user what we\'re doing */
if (fVerbose && fSortSegments) {
fprintf(errfh, "Sorting preload segments and"
" resources into fast-load section\n");
if (fBootModule)
fprintf(errfh,"This is a boot module; the .DEF file"
" is assumed to be correct!\n");
}
#endif
/* If we\'re sorting segments, write preload segments and resources
* into a section separate from the load on call stuff.
*/
if (fSortSegments) {
/* Save the start of the preload section */
wPreloadOffset = AlignFilePos(fhOutput, NewExe.ne_align, TRUE);
/* Write PRELOAD segments and resources */
SegsWrite(DO_PRELOAD);
ResWrite(DO_PRELOAD);
/* Compute the properly aligned length of the preload section */
wPreloadLength = AlignFilePos(fhOutput, NewExe.ne_align, TRUE) -
wPreloadOffset;
/* Now do the LOADONCALL segs and resources */
SegsWrite(DO_LOADONCALL);
ResWrite(DO_LOADONCALL);
}
/* If we\'re not sorting segments, just write them into a common block */
else {
/* Write the segs and resources */
SegsWrite(DO_PRELOAD | DO_LOADONCALL);
ResWrite(DO_PRELOAD | DO_LOADONCALL);
}
#ifdef SETEXEFLAGS
/* Set flags and other values in the EXE header */
SetEXEHeaderFlags();
#endif
/* Rewrite the new exe header, segment table and resource table */
RewriteTables();
ResTableBufferFree();
/* Handle CodeView info */
CopyCodeViewInfo(fhInput, fhOutput);
/* Seek to end of output file and issue truncating write */
MySeek(fhOutput, 0L, 2);
MyWrite(fhOutput, zeros, 0);
fclose(fhInput);
fclose(fhOutput);
fclose(fhBin);
FreePTypInfo(pTypInfo);
pTypInfo=NULL;
MyFree(pSegTable);
pSegTable=NULL;
return TRUE;
#endif // 0
}
/*
*
* ReadExeOldHeader\( fExeFile, lFileLen, plPosNewHdr \) : WORD;
*
* fExeFile file handle of .exe file being read
* lFileLen length of file
* plPosNewHdr pointer to file position of new header
*
* This function reads the old header from an executable file, checks to be
* sure that it is a valid header, and saves the position of the file\'s
* new header.
*
* This function returns IDERR_SUCCESS if there are no errors, or a non-zero
* error code if there are.
*
*/
static WORD ReadExeOldHeader(
FILE *fExeFile,
LONG lFileLen,
LONG *plPosNewHdr )
{
LONG lPos;
WORD cb;
EXEHDR ehOldHeader;
WORD wResult;
/* initialize */
wResult = IDERR_SUCCESS;
lPos = fseek( fExeFile, 0L, SEEK_SET );
if (lPos != 0L)
wResult = IDERR_READFAIL;
if (wResult == IDERR_SUCCESS) {
cb = fread( (void *) &ehOldHeader, sizeof( EXEHDR) , 1, fExeFile );
if (cb != 1 ) {
wResult = IDERR_READFAIL;
} else if (ehOldHeader.ehSignature != OLDEXESIGNATURE) {
wResult = IDERR_FILETYPEBAD;
} else if (ehOldHeader.ehPosNewHdr < sizeof(EXEHDR)) {
wResult = IDERR_EXETYPEBAD;
} else if ( ehOldHeader.ehPosNewHdr > (LONG)(lFileLen - sizeof(NEWHDR)) ) {
wResult = IDERR_EXETYPEBAD;
} else {
*plPosNewHdr = ehOldHeader.ehPosNewHdr;
}
}
return wResult;
}
/*
*
* ReadExeNewHeader\( fExeFile, lFileLen, lPosNewHdr, plPosResourceTable \) : WORD;
*
* fExeFile file handle of .exe file being read
* lFileLen length of file
* lPosNewHdr file position of new header
* plPosResourceTable pointer to file position of resource table
*
* This function reads the new header from an executable file, checks to be
* sure that it is a valid header, and saves the position of the file\'s
* resource table.
*
* This function returns IDERR_SUCCESS if there are no errors, or a non-zero
* error code if there are.
*
*/
static WORD ReadExeNewHeader(
FILE *fExeFile,
LONG lFileLen,
long lPosNewHdr,
LONG *plPosResourceTable )
{
WORD wResult;
WORD cb;
LONG lPos;
NEWHDR nhNewHeader;
/* initialize */
wResult = IDERR_SUCCESS;
fseek( fExeFile, lPosNewHdr, SEEK_SET );
lPos = ftell( fExeFile );
if (lPos == (long) -1 || lPos > lFileLen || lPos != lPosNewHdr) {
wResult = IDERR_READFAIL;
} else {
cb = fread( ( void *)&nhNewHeader, sizeof(nhNewHeader) , 1, fExeFile );
if (cb != 1 ) {
wResult = IDERR_READFAIL;
} else if (nhNewHeader.nhSignature != NEWEXESIGNATURE) {
wResult = IDERR_FILETYPEBAD;
} else if (nhNewHeader.nhExeType != WINDOWSEXE) {
wResult = IDERR_EXETYPEBAD;
} else if (nhNewHeader.nhExpVer < 0x0300) {
wResult = IDERR_WINVERSIONBAD;
} else if (nhNewHeader.nhoffResourceTable == 0) {
wResult = IDERR_RESTABLEBAD;
} else {
*plPosResourceTable = lPosNewHdr + nhNewHeader.nhoffResourceTable;
}
}
return wResult;
}
/*
*
* ReadExeTable\( fExeFile , lPosResourcTable \) : WORD;
*
* fExeFile file handle of .exe file being read
*
* This function reads through the entries in an .exe file\'s resource table,
* identifies any icons in that table, and saves the file offsets of the data
* for those icons. This function expects the initial file position to point
* to the first entry in the resource table.
*
* This function returns IDERR_SUCCESS if there are no errors, or a non-zero
* error code if there are.
*
*/
static WORD ExtractExeResources(
FILE *fExeFile,
FILE *fResFile,
LONG lPosResourceTable,
LONG lFileLen )
{
BOOL fLoop;
WORD wResult;
WORD cb;
LONG lPos;
WORD wShiftCount;
wResult = IDERR_SUCCESS;
// posistion file pointer at resource table
fseek( fExeFile, lPosResourceTable, SEEK_SET );
lPos = ftell(fExeFile);
if (lPos == (LONG) -1 || lPos > lFileLen || lPos != lPosResourceTable) {
return IDERR_READFAIL ;
}
if (wResult == IDERR_SUCCESS) {
cb = fread( (void *) &wShiftCount, sizeof(wShiftCount), 1 , fExeFile );
}
if (cb != 1 ) {
return IDERR_READFAIL;
}
if (wShiftCount > 16) {
return IDERR_RESTABLEBAD;
}
/* initialize */
wResult = IDERR_SUCCESS;
fLoop = TRUE;
/* loop through entries in resource table */
while (fLoop == TRUE) {
WORD cb;
WORD iFile;
RESTYPEINFO rt;
/* read RESTYPEINFO */
cb = fread( (void *)&rt, sizeof(rt), 1, fExeFile );
if (cb != 1 ) {
wResult = IDERR_READFAIL;
break;
}
if ( rt.rtType == 0 )
break;
// now get all the resource of this type
for (
iFile = 0;
iFile<rt.rtCount && wResult==IDERR_SUCCESS;
iFile++
) {
RESNAMEINFO rn;
cb = fread( (void *) &rn, sizeof(rn) , 1 , fExeFile );
if (cb != 1 ) {
wResult = IDERR_READFAIL;
}
WriteResFromExe( fExeFile, fResFile,lPos, rn, rt, wShiftCount );
}
fLoop = (rt.rtType != 0) && (wResult == IDERR_SUCCESS);
}
FCLOSE(fExeFile);
FCLOSE(fResFile);
return wResult;
}
//.................................................................
static WORD WriteResFromExe(
FILE *fExeFile,
FILE *fResFile,
LONG lPos,
RESNAMEINFO ResNameInfo,
RESTYPEINFO ResTypeInfo,
WORD wShiftCount )
{
LONG lCurPos;
LONG lResPos;
LONG wLength;
LONG wTmpLength;
wLength = (LONG) ResNameInfo.rnLength << wShiftCount;
lCurPos = ftell( fExeFile );
// position file pointer at resouce location
lResPos = (LONG) ResNameInfo.rnOffset << wShiftCount;
fseek( fExeFile, lResPos, SEEK_SET );
if ( ResTypeInfo.rtType & 0x8000) {
PutByte( fResFile, (BYTE) 0xff, NULL );
PutWord( fResFile, (WORD)(ResTypeInfo.rtType & 0x7FFF), NULL);
} else {
ExtractString(fExeFile,fResFile,lPos+ResTypeInfo.rtType);
}
if ( ResNameInfo.rnID & 0x8000 ) {
PutByte( fResFile, (BYTE) 0xff, NULL );
PutWord( fResFile, (WORD)(ResNameInfo.rnID & 0x7fFF), NULL);
} else {
ExtractString(fExeFile,fResFile,lPos+ResNameInfo.rnID);
}
PutWord( fResFile, ResNameInfo.rnFlags , NULL );
PutdWord( fResFile, (LONG) wLength, NULL );
wTmpLength = wLength;
// now write the actual data
fseek( fExeFile, lResPos, SEEK_SET );
ReadInRes( fExeFile, fResFile, &wTmpLength );
fseek( fExeFile, lCurPos, SEEK_SET );
return 0;
}
//..................................................................
static void ExtractString( FILE *fExeFile, FILE *fResFile, LONG lPos)
{
BYTE n,b;
fseek(fExeFile, lPos, SEEK_SET);
n=GetByte(fExeFile, NULL);
for (;n--; ) {
b=GetByte(fExeFile, NULL);
PutByte(fResFile, (CHAR) b, NULL);
}
PutByte(fResFile, (CHAR) 0, NULL);
}
// Modifications for RLTOOLS
// Currently we dont support any dynamic flags
BOOL fBootModule = FALSE;
BOOL fSortSegments = TRUE;
TYPINFO *pTypInfo = NULL;
static void FreePTypInfo( TYPINFO *pTypInfo)
{
RESINFO * pRes, *pRTemp;
TYPINFO * pTItemp;
while (pTypInfo) {
pRes = pTypInfo->pres;
while (pRes) {
pRTemp = pRes->next;
MyFree(pRes->name);
MyFree(pRes);
pRes = pRTemp;
}
pTItemp = pTypInfo->next;
MyFree(pTypInfo->type);
MyFree(pTypInfo);
pTypInfo = pTItemp;
}
}
/* ----- Helper functions ----- */
/* PrepareFiles
* Prepares the EXE files \(new and old\) for writing and verifies
* that all is well.
* Exits on error, returns if processing should continue.
*/
static void PrepareFiles(
PSTR pstrSource,
PSTR pstrDest,
PSTR pstrRes )
{
/* Open the .EXE file the linker gave us */
if ( (fhInput = FOPEN(pstrSource, "rb" )) == NULL ) {
OutPutError("Unable to open Exe Source File");
}
if ((fhBin = FOPEN(pstrRes, "rb")) == NULL ) {
OutPutError("Unable to open Resource File");
}
/* Read the old format EXE header */
MyRead(fhInput, (PSTR)&OldExe, sizeof (OldExe));
/* Make sure its really an EXE file */
if (OldExe.e_magic != EMAGIC) {
OutPutError("Invalid .EXE file" );
}
/* Make sure theres a new EXE header floating around somewhere */
if (!(dwNewExe = OldExe.e_lfanew)) {
OutPutError("Not a Microsoft Windows format .EXE file");
}
/* Go find the new .EXE header */
MySeek(fhInput, dwNewExe, 0);
MyRead(fhInput, (PSTR)&NewExe, sizeof (NewExe));
/* Check version numbers */
if (NewExe.ne_ver < 4) {
OutPutError("File not created by LINK");
}
/* Were there linker errors? */
if (NewExe.ne_flags & NEIERR) {
OutPutError("Errors occurred when linking file.");
}
/* Make sure that this program\'s EXETYPE is WINDOWS \(2\) not OS/2 \(1\) */
if (NewExe.ne_exetyp != 2)
OutPutError("The EXETYPE of the program is not WINDOWS.\n"
"(Make sure the .DEF file is correct.");
#ifdef VERBOSE
if (fVerbose) {
fprintf(errfh, "\n");
}
#endif
/* Open the all new executable file */
if ( (fhOutput = FOPEN( pstrDest, "wb")) == NULL ) {
OutPutError("Unable to create destination");
}
}
/* ReadSegmentTable
* Reads the segment table from the file.
*/
static void ReadSegmentTable( void)
{
struct new_seg* pSeg;
WORD i;
MySeek(fhInput, (LONG)NewExe.ne_segtab + dwNewExe, 0);
if ((wSegTableLen = NewExe.ne_cseg * sizeof (struct new_seg)) > 0) {
pSegTable = (struct new_seg *)RcAlloc (wSegTableLen);
MyRead(fhInput, (PSTR)pSegTable, wSegTableLen);
/* See if we have more than one data segment */
fMultipleDataSegs = 0;
for (pSeg = pSegTable, i = NewExe.ne_cseg ; i ; --i, ++pSeg) {
if ((pSeg->ns_flags & NSTYPE) == NSDATA) {
++fMultipleDataSegs;
}
}
if (fMultipleDataSegs) {
--fMultipleDataSegs;
}
} else {
pSegTable = NULL;
}
}
/* ComputeResTableSize
* Computes the size of the resource table by enumerating all the
* resources currently in the linked lists.
*/
static void ComputeResTableSize( void)
{
TYPINFO **pPrev;
TYPINFO *pType;
RESINFO *pRes;
/* Start with the minimum overhead size of the resource table. This
* is the resource alignment count and the zero WORD terminating the
* table. This is necessary so that we put the correct file offset
* in for the string offsets to named resources.
*/
wResTableLen = RESTABLEHEADER;
/* Loop over type table, computing the fixed length of the
* resource table, removing unused type entries.
*/
pPrev = &pTypInfo;
dwMaxFilePos = 0L;
while (pType = *pPrev) {
if (pRes = pType->pres) {
/* Size of type entry */
wResTableLen += sizeof (struct rsrc_typeinfo);
while (pRes) {
/* Size of resource entry */
wResTableLen += sizeof (struct rsrc_nameinfo);
if (pType->next || pRes->next) {
dwMaxFilePos += pRes->size;
}
pRes = pRes->next;
}
pPrev = &pType->next;
} else {
*pPrev = pType->next;
MyFree(pType->type);
MyFree(pType);
}
}
}
/* ComputeStringOffsets
* Computes offsets to strings from named resource and types.
*/
static void ComputeStringOffsets( void)
{
TYPINFO *pType;
RESINFO *pRes;
/* Loop over type table, computing string offsets for non-ordinal
* type and resource names.
*/
pType = pTypInfo;
while (pType) {
pRes = pType->pres;
/* Is there an ordinal? */
if (pType->typeord) {
/* Mark the ordinal */
pType->typeord |= RSORDID;
/* Flush the string name */
MyFree(pType->type);
pType->type = NULL;
} else if (pType->type) { /* is there a type string? */
/* Yes, compute location of the type string */
pType->typeord = wResTableLen;
wResTableLen += strlen(pType->type) + 1;
}
while (pRes) {
/* Is there an ordinal? */
if (pRes->nameord) {
/* Mark the ordinal */
pRes->nameord |= RSORDID;
/* Flush the string name */
MyFree(pRes->name);
pRes->name = NULL;
}
/* Is there a resource name? */
else if (pRes->name) {
/* Yes, compute location of the resource string */
pRes->nameord = wResTableLen;
wResTableLen += strlen(pRes->name) + 1;
}
pRes = pRes->next;
}
pType = pType->next;
}
}
/* BuildResTable
* Builds the local memory image of the resource table.
*/
static void BuildResTable( void)
{
TYPINFO *pType;
RESINFO *pRes;
/* Check to see if we have any resources. If not, just omit the table */
if (wResTableLen > RESTABLEHEADER) {
/* Set up the temporary resource table buffer */
ResTableBufferInit(wResTableLen);
/* Alignment shift count
* \(we default here to the segment alignment count\)
*/
RcPutWord(NewExe.ne_align);
pType = pTypInfo;
while (pType) {
/* output the type and number of resources */
RcPutWord(pType->typeord); /* DW type id */
RcPutWord(pType->nres); /* DW #resources for this type */
RcPutWord(0); /* DD type procedure */
RcPutWord(0);
/* output flags and space for the file offset for each resource */
pRes = pType->pres;
while (pRes) {
pRes->poffset = (WORD *)pResNext;
RcPutWord(0); /* DW file offset */
RcPutWord(0); /* DW resource size */
pRes->flags |= NSDPL;
RcPutWord(pRes->flags ); /* DW flags */
RcPutWord(pRes->nameord ); /* DW name id */
RcPutWord(0); /* DW handle */
RcPutWord(0); /* DW usage or minalloc */
pRes = pRes->next;
}
pType = pType->next;
}
/* Null entry terminates table */
RcPutWord(0);
/* Output type and name strings for non-ordinal resource types
* and names */
pType = pTypInfo;
while (pType) {
/* Dump out any strings for this type */
if (pType->type && !(pType->typeord & RSORDID)) {
RcPutString(pType->type);
}
pRes = pType->pres;
while (pRes) {
if (pRes->name && !(pRes->nameord & RSORDID))
RcPutString(pRes->name);
pRes = pRes->next;
}
pType = pType->next;
}
} else
wResTableLen = 0;
}
/* SegsWrite
* Copies segments to the file. This routine will do only preload,
* only the load on call, or both types of segments depending on
* the flags.
*/
static void SegsWrite( WORD wFlags)
{
WORD wExtraPadding;
WORD i;
static struct new_seg *pSeg;
DWORD dwSegSize;
DWORD dwWriteSize;
WORD wTemp;
WORD wcbDebug;
/* We only need extra padding in the preload section.
* Note that when wFlags == DO_PRELOAD | DO_LOADONCALL, we DON\'T
* need extra padding because this is NOT a preload section.
* \(hence the \'==\' instead of an \'&\'\)
*/
wExtraPadding = (wFlags == DO_PRELOAD);
/* Copy segment data for each segment, fixed and preload only */
for (i = 1, pSeg = pSegTable; i <= NewExe.ne_cseg; i++, pSeg++) {
/* If there\'s no data in segment, skip it here */
if (!pSeg->ns_sector) {
continue;
}
/* Force some segments to be preload if doing preload resources */
if ((wFlags & DO_PRELOAD) && !fBootModule) {
char *reason = NULL;
/* Check various conditions that would force preloading */
if (i == (unsigned)(NewExe.ne_csip >> 16)) {
reason = "Entry point";
}
if (!(pSeg->ns_flags & NSMOVE)) {
reason = "Fixed";
}
if (pSeg->ns_flags & NSDATA) {
reason = "Data";
}
if (!(pSeg->ns_flags & NSDISCARD)) {
reason = "Non-discardable";
}
/* If this segment must be preload and the segment is not already
* marked as such, warn the user and set it.
*/
if (reason && !(pSeg->ns_flags & NSPRELOAD)) {
#ifdef VERBOSE
fprintf(errfh,
"RC: warning RW4002: %s segment %d set to PRELOAD\n",
reason, i);
#endif
pSeg->ns_flags |= NSPRELOAD;
}
}
/* Skip this segment if it doesn\'t match the current mode */
wTemp = pSeg->ns_flags & NSPRELOAD ? DO_PRELOAD : DO_LOADONCALL;
if (!(wTemp & wFlags)) {
continue;
}
/* Get the true segment length. A zero length implies 64K */
if (pSeg->ns_cbseg) {
dwSegSize = pSeg->ns_cbseg;
} else {
dwSegSize = 0x10000L;
}
#ifdef VERBOSE
if (fVerbose)
fprintf(errfh, "Copying segment %d (%lu bytes)\n", i, dwSegSize);
#endif
/* Align the segment correctly and pad the file to match */
MoveFilePos(fhInput, pSeg->ns_sector, NewExe.ne_align);
pSeg->ns_sector = AlignFilePos(fhOutput, NewExe.ne_align,
wExtraPadding);
/* Copy the segment */
MyCopy(fhInput, fhOutput, dwSegSize);
/* Pad out all segments in the preload area to their minimum
* memory allocation size so that KERNEL doesn\'t have to realloc
* the segment.
*/
if (wExtraPadding && pSeg->ns_cbseg != pSeg->ns_minalloc) {
/* A minalloc size of zero implies 64K */
if (!pSeg->ns_minalloc) {
dwWriteSize = 0x10000L - pSeg->ns_cbseg;
} else {
dwWriteSize = pSeg->ns_minalloc - pSeg->ns_cbseg;
}
/* Add in to total size of segment */
dwSegSize += dwWriteSize;
/* Set the segment table size to this new size */
pSeg->ns_cbseg = pSeg->ns_minalloc;
/* Pad the file */
while (dwWriteSize) {
dwWriteSize -= MyWrite(fhOutput,
zeros,
(WORD)(dwWriteSize > (DWORD) NUMZEROS
? NUMZEROS : dwWriteSize));
}
}
/* Copy the relocation information */
if (pSeg->ns_flags & NSRELOC) {
/* Copy the relocation stuff */
dwSegSize += RelocCopy(i);
/* Segment + padding + relocations can\'t be >64K for preload
* segments.
*/
if (fSortSegments && (pSeg->ns_flags & NSPRELOAD) &&
dwSegSize > 65536L) {
#ifdef VERBOSE
fprintf(errfh,
"RC : fatal error RW1031: Segment %d and its\n"
" relocation information is too large for load\n"
" optimization. Make the segment LOADONCALL or\n"
" rerun RC using the -K switch if the segment must\n"
" be preloaded.\n", i);
#endif
}
}
/* Copy any per-segment debug information */
if (pSeg->ns_flags & NSDEBUG) {
MyRead(fhInput, (PSTR)&wcbDebug, sizeof (WORD));
MyWrite(fhOutput, (PSTR)&wcbDebug, sizeof (WORD));
MyCopy(fhInput, fhOutput, (LONG)wcbDebug);
}
}
}
/* RelocCopy
* Copys all the relocation records for a given segment.
* Also checks for invalid fixups.
*/
static DWORD RelocCopy( WORD wSegNum)
{
WORD wNumReloc;
struct new_rlc RelocRec;
WORD i;
BYTE byFixupType;
BYTE byFixupFlags;
WORD wDGROUP;
/* Get the number of relocations */
MyRead(fhInput, (PSTR)&wNumReloc, sizeof (WORD));
MyWrite(fhOutput, (PSTR)&wNumReloc, sizeof (WORD));
/* Get the automatic data segment */
wDGROUP = NewExe.ne_autodata;
/* Copy and verify all relocations */
for (i = 0 ; i < wNumReloc ; ++i) {
/* Copy the record */
MyRead(fhInput, (PSTR)&RelocRec, sizeof (RelocRec));
MyWrite(fhOutput, (PSTR)&RelocRec, sizeof (RelocRec));
/* Validate it only if necessary */
if ((NewExe.ne_flags & (NENOTP | NESOLO)) ||
wSegNum == wDGROUP || fMultipleDataSegs) {
continue;
}
/* Bad fixups are fixups to DGROUP in code segments in apps
* that can be multi-instanced. Since we can\'t fix up locations
* that are different from instance to instance in shared code
* segments, we have to warn the user. We only warn because this
* may be allowable if the app only allows a single instance of
* itself to run.
*/
byFixupType = (BYTE) (RelocRec.nr_stype & NRSTYP);
byFixupFlags = (BYTE) (RelocRec.nr_flags & NRRTYP);
#ifdef VERBOSE
if ((byFixupType == NRSSEG || byFixupType == NRSOFF) &&
byFixupFlags == NRRINT &&
RelocRec.nr_union.nr_intref.nr_segno == wDGROUP)
fprintf(errfh,
"RC : warning RW4005: Segment %d (offset %04X) contains a\n"
" relocation record pointing to the automatic\n"
" data segment. This will cause the program to crash\n"
" if the instruction being fixed up is executed in a\n"
" multi-instance application. If this fixup is\n"
" necessary, the program should be restricted to run\n"
" only a single instance.\n", wSegNum, RelocRec.nr_soff);
#endif
}
return wNumReloc * sizeof (struct new_rlc);
}
/* ResWrite
* Copies resources to the file. This routine will do only the preload,
* only the load on call, or both types of resources depending on the
* flags.
*/
static void ResWrite( WORD wFlags)
{
WORD wExtraPadding;
WORD wTemp;
WORD wResAlign;
TYPINFO *pType;
RESINFO *pRes;
/* If we have no resource table, just ignore this */
if (!wResTableLen) {
return;
}
/* We only need extra padding in the preload section.
* Note that when wFlags == DO_PRELOAD | DO_LOADONCALL, we DON\'T
* need extra padding because this is NOT a preload section.
* \(hence the \'==\' instead of an \'&\'\)
*/
wExtraPadding = (wFlags == DO_PRELOAD);
/* Compute resource alignment. Note that the alignment is not the
* same as the segment alignment ONLY IF there is no segment sorting
* and some resources cannot be reached with the current segment
* align count.
*/
wResAlign = NewExe.ne_align;
if (!fSortSegments) {
/* Compute the needed alignment */
dwMaxFilePos += MySeek(fhOutput, 0L, 2);
wResAlign = GetAlign(dwMaxFilePos, NewExe.ne_align);
#ifdef VERBOSE
if (fVerbose)
fprintf(errfh, "Resources will be aligned on %d byte boundaries\n",
1 << wResAlign);
#endif
/* Point back to the start of the local memory resource table */
pResNext = pResTable;
RcPutWord(wResAlign);
}
/* Output contents associated with each resource */
for (pType = pTypInfo ; pType; pType = pType->next) {
for (pRes = pType->pres ; pRes ; pRes = pRes->next) {
/* Make sure this is the right kind of resource */
wTemp = pRes->flags & RNPRELOAD ? DO_PRELOAD : DO_LOADONCALL;
if (!(wTemp & wFlags)) {
continue;
}
/* Give some info to the user */
#ifdef VERBOSE
if (fVerbose) {
fprintf(errfh, "Writing resource ");
if (pRes->name && !(pRes->nameord & RSORDID)) {
fprintf(errfh, "%s", pRes->name);
} else {
fprintf(errfh, "%d", pRes->nameord & 0x7FFF);
}
if (pType->type && !(pType->typeord & RSORDID)) {
fprintf(errfh, ".%s", pType->type);
} else {
fprintf(errfh, ".%d", pType->typeord & 0x7FFF);
}
fprintf(errfh, " (%lu bytes)\n", pRes->size);
fflush(errfh);
}
#endif
/* Copy the resource from the RES file to the EXE file */
MySeek(fhBin, (long)pRes->BinOffset, 0);
*(pRes->poffset)++ =
AlignFilePos(fhOutput, wResAlign, wExtraPadding);
*(pRes->poffset) = RoundUp(pRes->size, wResAlign);
MyCopy(fhBin, fhOutput, pRes->size);
}
}
/* Compute the end of the EXE file thus far for the CV info */
dwExeEndFile = AlignFilePos(fhOutput, wResAlign, wExtraPadding);
}
#ifdef SETEXEFLAGS
/* SetEXEHeaderFlags
* Sets necessary flags and values in the EXE header.
*/
static void SetEXEHeaderFlags( void)
{
/* Tell loader we initialized previously unused fields */
if (NewExe.ne_ver == 4) {
NewExe.ne_rev = 2;
}
/* Set command line values into the header */
NewExe.ne_expver = expWinVer;
NewExe.ne_swaparea = swapArea;
/* Set the preload section values */
if (fSortSegments) {
/* Set the new fastload section values */
NewExe.ne_gangstart = wPreloadOffset;
NewExe.ne_ganglength = wPreloadLength;
#ifdef VERBOSE
if (fVerbose)
fprintf(errfh, "Fastload area is %ld bytes at offset 0x%lX.\n",
(LONG)wPreloadLength << NewExe.ne_align,
(LONG)wPreloadOffset << NewExe.ne_align);
}
#endif
/* Clear all the flags */
NewExe.ne_flags &=
~(NELIM32|NEMULTINST|NEEMSLIB|NEPRIVLIB|NEPRELOAD);
/* Set appropriate flags */
if (fLim32) {
NewExe.ne_flags |= NELIM32;
}
if (fMultInst) {
NewExe.ne_flags |= NEMULTINST;
}
if (fEmsLibrary) {
NewExe.ne_flags |= NEEMSLIB;
}
if (fPrivateLibrary) {
NewExe.ne_flags |= NEPRIVLIB;
}
if (fProtOnly) {
NewExe.ne_flags |= NEPROT;
}
if (fSortSegments && wPreloadLength) {
NewExe.ne_flagsother |= NEPRELOAD;
}
NewExe.ne_flags |= NEWINAPI;
}
#endif
/* RewriteTables
* Rewrites the EXE header and the resource and segment tables
* with their newly-updated information.
*/
static void RewriteTables( void)
{
/* Write the new EXE header */
MySeek(fhOutput, (LONG)dwNewExe, 0);
MyWrite(fhOutput, (PSTR)&NewExe, sizeof (NewExe));
/* Seek to the start of the segment table */
MySeek(fhOutput, dwNewExe + (LONG)NewExe.ne_segtab, 0);
MyWrite(fhOutput, (PSTR)pSegTable, wSegTableLen);
/* Seek to and write the resource table */
if (wResTableLen) {
MySeek(fhOutput, dwNewExe + (LONG)NewExe.ne_rsrctab, 0);
MyWrite(fhOutput, pResTable, wResTableLen);
}
}
/* CopyCodeViewInfo
* Copies CodeView info to the new EXE file and relocates it if
* necessary. This routine is designed to work with the
* DNRB-style info as well as NBxx info where x is a digit.
*/
static void CopyCodeViewInfo( FILE *fhInput, FILE *fhOutput)
{
unsigned long dwcb;
unsigned int i;
CVINFO cvinfo;
CVSECTBL cvsectbl;
/* See if old format \(DNRB\) symbols present at end of input file
* If they are, relocate the table to the new file position and
* fix up the file-position dependent offsets.
*/
dwcb = MySeek( fhInput, -(signed long)sizeof (CVINFO), 2);
MyRead( fhInput, (char *)&cvinfo, sizeof (cvinfo));
if (*(unsigned long *)cvinfo.signature == CV_OLD_SIG) {
dwcb -= cvinfo.secTblOffset;
MySeek( fhInput, cvinfo.secTblOffset, 0);
MyRead( fhInput, (char *)&cvsectbl, sizeof (cvsectbl));
dwcb -= sizeof (cvsectbl);
for (i = 0 ; i < 5 ; ++i) {
cvsectbl.secOffset[i] -= cvinfo.secTblOffset;
}
cvinfo.secTblOffset = dwExeEndFile;
for (i = 0 ; i < 5 ; ++i) {
cvsectbl.secOffset[i] += cvinfo.secTblOffset;
}
MySeek( fhOutput, cvinfo.secTblOffset, 0);
MyWrite( fhOutput, (char *)&cvsectbl, sizeof (cvsectbl));
MyCopy( fhInput, fhOutput, dwcb);
MyWrite( fhOutput, (char *)&cvinfo, sizeof (cvinfo));
}
/* Check for new format \(NBxx\) symbols. Since these symbols are
* file-position independent, just copy them over; no need to
* fix them up as with the old format symbols.
*/
else if (*(unsigned short int *)cvinfo.signature == CV_SIGNATURE &&
isdigit(cvinfo.signature[2]) && isdigit(cvinfo.signature[3])) {
MySeek( fhOutput, 0L, 2);
MySeek( fhInput, -cvinfo.secTblOffset, 2);
MyCopy( fhInput, fhOutput, cvinfo.secTblOffset);
}
}
/* OutPutError
* Outputs a fatal error message and exits.
*/
static void OutPutError( char *szMessage)
{
QuitA( 0, szMessage, NULL);
}
/* ResTableBufferInit
* Creates the resource table buffer and points global pointers
* to it. This table is written to so that we can modifiy it
* before writing it out to the EXE file.
*/
static void ResTableBufferInit( WORD wLen)
{
/* Allocate local storage for resource table */
pResTable = RcAlloc (wLen);
/* Point to the start of the table for the PutXXXX\(\) */
pResNext = pResTable;
}
/* ResTableBufferFree
* Frees the temporary storage for resource table
*/
static void ResTableBufferFree( void)
{
/* Nuke the table */
MyFree(pResTable);
}
/* GetAlign
* Computes the alignment value needed for the given maximum file
* position passed in. This is done by computing the number of
* bits to be shifted left in order to represent the maximum
* file position in 16 bits.
*/
static WORD GetAlign( DWORD dwMaxpos, WORD wAlign)
{
DWORD dwMask;
WORD i;
/* Compute the initial mask based on the input align value */
dwMask = 0xFFFFL;
for (i = 0; i < wAlign ; ++i) {
dwMask <<= 1;
dwMask |= 1;
}
/* See if we need to increase the default mask to reach the maximum
* file position.
*/
while (dwMaxpos > dwMask) {
dwMask <<= 1;
dwMask |= 1;
++wAlign;
}
/* Return the new alignment */
return wAlign;
}
/* MoveFilePos
* Moves the file pointer to the position indicated by wPos, using the
* align shift count wAlign. This converts the WORD value wPos
* into a LONG value by shifting left wAlign bits.
*/
static LONG MoveFilePos( FILE *fh, WORD wPos, WORD wAlign)
{
return MySeek(fh, ((LONG)wPos) << wAlign, 0);
}
/* RoundUp
* Computes the value that should go into a 16 bit entry in an EXE
* table by rounding up to the next boundary determined by the
* passed in alignment value.
*/
static WORD RoundUp( LONG lValue, WORD wAlign)
{
LONG lMask;
/* Get all the default mask of all ones except in the bits below the
* alignment value.
*/
lMask = -1L;
lMask <<= wAlign;
/* Now round up using this mask */
lValue += ~lMask;
lValue &= lMask;
/* Return as a 16 bit value */
return ((WORD) (lValue >> (LONG) wAlign));
}
/* AlignFilePos
* Computes a correctly aligned file position based on the current
* alignment.
*/
static WORD AlignFilePos( FILE *fh, WORD wAlign, BOOL fPreload)
{
LONG lCurPos;
LONG lNewPos;
LONG lMask;
WORD nbytes;
WORD wNewAlign;
/* If we\'re in the preload section, we have tougher alignment
* restrictions: We have to be at least 32-byte aligned and have
* at least 32 bytes between objects for arena headers. It turns
* out that this feature is not really used in KERNEL but could be
* implemented someday.
*/
if (fPreload && wAlign < PRELOAD_ALIGN) {
wNewAlign = PRELOAD_ALIGN;
} else {
wNewAlign = wAlign;
}
/* Get the current file position */
lCurPos = MySeek(fh, 0L, 1);
/* Compute the new position by rounding up to the align value */
lMask = -1L;
lMask <<= wNewAlign;
lNewPos = lCurPos + ~lMask;
lNewPos &= lMask;
/* We have to have at least 32 bytes between objects in the preload
* section.
*/
if (fPreload) {
while (lNewPos - lCurPos < PRELOAD_MINPADDING) {
lNewPos += 1 << wNewAlign;
}
}
/* Check to see if it\'s representable in 16 bits */
if (lNewPos >= (0x10000L << wAlign)) {
OutPutError(".EXE file too large; relink with higher /ALIGN value");
}
/* Write stuff out to file until new position reached */
if (lNewPos > lCurPos) {
/* Compute number of bytes to write out and write them out */
nbytes = (WORD) (lNewPos - lCurPos);
while (nbytes) {
nbytes -= MyWrite( fh,
zeros,
(WORD)(nbytes > NUMZEROS ? NUMZEROS : nbytes));
}
}
/* Seek to and return this new position */
return (WORD)(MySeek(fh, lNewPos, (WORD) 0) >> (LONG) wAlign);
}
/*--------------------------------------------------------------------------*/
/* */
/* AddResType\(\) - */
/* */
/*--------------------------------------------------------------------------*/
static TYPINFO *AddResType( CHAR *s, WORD n )
{
TYPINFO *pType;
if (pType = pTypInfo) {
while (TRUE) {
/* search for resource type, return if already exists */
if ((s && !strcmp(s, pType->type)) || (!s && n && pType->typeord == n)) {
return (pType);
} else if (!pType->next) {
break;
} else {
pType = pType->next;
}
}
/* if not in list, add space for it */
pType->next = (TYPINFO *) RcAlloc(sizeof(TYPINFO));
pType = pType->next;
} else {
/* allocate space for resource list */
pTypInfo = (TYPINFO *)RcAlloc (sizeof(TYPINFO));
pType = pTypInfo;
}
/* fill allocated space with name and ordinal, and clear the resources
of this type */
pType->type = MyMakeStr(s);
pType->typeord = n;
pType->nres = 0;
pType->pres = NULL;
pType->next = NULL;
return (pType);
}
/*--------------------------------------------------------------------------*/
/* */
/* GetOrdOrName\(\) - */
/* */
/*--------------------------------------------------------------------------*/
static void GetOrdOrName( unsigned int *pint, unsigned char *szstr)
{
unsigned char c1;
/* read the first character of the identifier */
MyRead(fhBin, &c1, sizeof(unsigned char));
/* if the first character is 0xff, the id is an ordinal, else a string */
if (c1 == 0xFF) {
MyRead(fhBin, (PSTR)pint, sizeof (int));
} else { /* string */
*pint = 0;
*szstr++ = c1;
do {
MyRead( fhBin, szstr, 1);
}
while (*szstr++ != 0);
}
}
/*--------------------------------------------------------------------------*/
/* */
/* AddDefaultTypes\(\) - */
/* */
/*--------------------------------------------------------------------------*/
static void AddDefaultTypes( void)
{
AddResType( "CURSOR", ID_RT_GROUP_CURSOR);
AddResType( "ICON", ID_RT_GROUP_ICON);
AddResType( "BITMAP", ID_RT_BITMAP);
AddResType( "MENU", ID_RT_MENU);
AddResType( "DIALOG", ID_RT_DIALOG);
AddResType( "STRINGTABLE", ID_RT_STRING);
AddResType( "FONTDIR", ID_RT_FONTDIR);
AddResType( "FONT", ID_RT_FONT);
AddResType( "ACCELERATORS", ID_RT_ACCELERATORS);
AddResType( "RCDATA", ID_RT_RCDATA);
AddResType( "VERSIONINFO", ID_RT_VERSION);
}
/*--------------------------------------------------------------------------*/
/* */
/* ProcessBinFile\(\) - */
/* */
/*--------------------------------------------------------------------------*/
static int ProcessBinFile( void)
{
unsigned int ord;
unsigned char tokstr[64];
RESINFO *pRes;
TYPINFO *pType;
long curloc;
long eofloc;
WORD wResType;
/* initialize for reading .RES file */
AddDefaultTypes();
eofloc = MySeek(fhBin, 0L, 2); /* get file size */
curloc = MySeek(fhBin, 0L, 0); /* go to beginning of file */
/* while there are more resources in the .RES file */
while (curloc < eofloc) {
#ifdef VERBOSE
if (fVerbose) {
fprintf(errfh, ".");
fflush(errfh);
}
#endif
/* find the resource type of the next resource */
GetOrdOrName(&ord, tokstr);
if (!ord) {
pType = AddResType(tokstr, 0);
} else {
pType = AddResType(NULL, (WORD)ord);
}
if (!pType) {
break;
}
/* Save the type number so we can see if we want to skip it later */
wResType = ord;
/* find the identifier \(name\) of the resource */
GetOrdOrName(&ord, tokstr);
pRes = (RESINFO *)RcAlloc (sizeof(RESINFO));
if (!ord) {
pRes->name = MyMakeStr(tokstr);
} else {
pRes->nameord = ord;
}
/* read the flag bits */
MyRead(fhBin, (PSTR)&pRes->flags, sizeof(int));
/* Clear the old DISCARD bits. */
pRes->flags &= 0x1FFF;
/* find the size of the resource */
MyRead(fhBin, (PSTR)&pRes->size, sizeof(long));
/* save the position of the resource for when we add it to the .EXE */
pRes->BinOffset = (long)MySeek(fhBin, 0L, 1);
/* skip the resource to the next resource header */
curloc = MySeek(fhBin, (long)pRes->size, 1);
/* add the resource to the resource lists. We don\'t add name
* tables. They are an unnecessary 3.0 artifact.
*/
if (wResType != ID_RT_NAMETABLE) {
AddResToResFile(pType, pRes);
} else {
MyFree(pRes->name);
MyFree(pRes);
}
}
return 1;
}
/*--------------------------------------------------------------------------*/
/* */
/* AddResToResFile\(pType, pRes\) */
/* */
/* Parameters: */
/* pType : Pointer to Res Type */
/* pRes : Pointer to resource */
/* */
/*--------------------------------------------------------------------------*/
static void AddResToResFile( TYPINFO *pType, RESINFO *pRes)
{
RESINFO *p;
p = pType->pres;
/* add resource to end of resource list for this type */
if (p) {
while (p->next) {
p = p->next;
}
p->next = pRes;
p->next->next = NULL;
} else {
pType->pres = pRes;
pType->pres->next = NULL;
}
/* keep track of number of resources and types */
pType->nres++;
}
/* MyMakeStr
* Makes a duplicate string from the string passed in. The new string
* should be freed when it is no longer useful.
*/
static PSTR MyMakeStr( PSTR s)
{
PSTR s1;
if (s) {
s1 = RcAlloc( (WORD)(strlen(s) + 1)); /* allocate buffer */
strcpy(s1, s); /* copy string */
} else {
s1 = s;
}
return s1;
}
static SHORT MyRead( FILE *fh, PSTR p, WORD n)
{
size_t n1;
if ( (n1 = fread( p, 1, n, fh)) != n )
; // quit\("RC : fatal error RW1021: I/O error reading file."\);
else
return ( n1);
}
/* MyWrite
* Replaces calls to write\(\) and does error checking.
*/
static SHORT MyWrite( FILE *fh, PSTR p, WORD n)
{
size_t n1;
if ( (n1 = fwrite( p, 1, n, fh)) != n )
; // quit\("RC : fatal error RW1022: I/O error writing file."\);
else
return ( n1);
}
/* MySeek
* Replaces calls to lseek\(\) and does error checking
*/
static LONG MySeek( FILE *fh, LONG pos, WORD cmd)
{
if ( (pos = fseek( fh, pos, cmd)) != 0 ) {
OutPutError ("RC : fatal error RW1023: I/O error seeking in file");
}
return ( pos);
}
/* MyCopy
* Copies dwSize bytes from source to dest in fixed size chunks.
*/
static void MyCopy( FILE *srcfh, FILE *dstfh, DWORD dwSize)
{
WORD n;
static char chCopyBuffer[ BUFSIZE];
while ( dwSize ) {
n = MyRead( srcfh, chCopyBuffer, sizeof( chCopyBuffer));
MyWrite( dstfh, chCopyBuffer, n);
dwSize -= n;
}
}
static void RcPutWord( unsigned int w)
{
*((WORD *)pResNext) = w;
pResNext++;
pResNext++;
}
/* PutStringWord
* Writes a string to the static resource buffer pointed to by pResNext.
* The string is stored in Pascal-format \(leading byte first\).
* Returns the number of characters written.
*/
static int RcPutString( char *pstr)
{
int i;
/* Make sure we have a valid string */
if (!pstr || !(i = strlen(pstr))) {
return 0;
}
/* Write the length byte */
*pResNext++ = (char) i;
/* Write all the characters */
while (*pstr) {
*pResNext++ = *pstr++;
}
/* Return the length */
return (i + 1);
}
static PSTR RcAlloc( WORD nbytes)
{
PSTR ps = NULL;
if ( ps = (PSTR)MyAlloc( nbytes)) {
return ( ps);
}
}