reactos/sdk/tools/cabman/dfp.cxx
2020-09-20 17:28:02 +02:00

1317 lines
32 KiB
C++

/*
* COPYRIGHT: See COPYING in the top level directory
* PROJECT: ReactOS cabinet manager
* FILE: tools/cabman/dfp.cxx
* PURPOSE: Directive file parser
* PROGRAMMERS: Casper S. Hornstrup (chorns@users.sourceforge.net)
* Colin Finck <mail@colinfinck.de>
* NOTES: The directive file format is similar to the
* directive file format used by Microsoft's MAKECAB (But not entirely compatible!)
* REVISIONS:
* CSH 21/03-2001 Created
* CSH 15/08-2003 Made it portable
* CF 04/05-2007 Made it compatible with 64-bit operating systems
*/
#include <stdlib.h>
#include <stdio.h>
#include "cabman.h"
#include "dfp.h"
/* CDFParser */
CDFParser::CDFParser()
/*
* FUNCTION: Default constructor
*/
{
InfFileOnly = false;
DontGenerateInf = false;
FileBuffer = NULL;
FileLoaded = false;
CurrentOffset = 0;
CurrentLine = 0;
CabinetCreated = false;
DiskCreated = false;
FolderCreated = false;
CabinetName = NULL;
DiskLabel = NULL;
MaxDiskSize = NULL;
MaxDiskSizeAllSet = false;
CabinetNameTemplateSet = false;
DiskLabelTemplateSet = false;
InfFileNameSet = false;
InfModeEnabled = false;
InfFileHandle = NULL;
}
CDFParser::~CDFParser()
/*
* FUNCTION: Default destructor
*/
{
PCABINET_NAME CNPrev;
PCABINET_NAME CNNext;
PDISK_NUMBER DNPrev;
PDISK_NUMBER DNNext;
if (FileBuffer)
free(FileBuffer);
CNNext = CabinetName;
while (CNNext != NULL)
{
CNPrev = CNNext->Next;
delete CNNext;
CNNext = CNPrev;
}
CNNext = DiskLabel;
while (CNNext != NULL)
{
CNPrev = CNNext->Next;
delete CNNext;
CNNext = CNPrev;
}
DNNext = MaxDiskSize;
while (DNNext != NULL)
{
DNPrev = DNNext->Next;
delete DNNext;
DNNext = DNPrev;
}
if (InfFileHandle != NULL)
fclose(InfFileHandle);
}
void CDFParser::WriteInfLine(char* InfLine)
{
char buf[PATH_MAX];
char eolbuf[2];
const char* destpath;
if (DontGenerateInf)
return;
if (InfFileHandle == NULL)
{
if (!InfFileNameSet)
/* FIXME: Use cabinet name with extension .inf */
return;
destpath = GetDestinationPath();
if (strlen(destpath) > 0)
{
strcpy(buf, destpath);
strcat(buf, InfFileName);
}
else
strcpy(buf, InfFileName);
/* Create .inf file, overwrite if it already exists */
InfFileHandle = fopen(buf, "wb");
if (InfFileHandle == NULL)
{
DPRINT(MID_TRACE, ("Error creating INF file.\n"));
return;
}
}
if (fwrite(InfLine, strlen(InfLine), 1, InfFileHandle) < 1)
{
DPRINT(MID_TRACE, ("Error writing INF file.\n"));
return;
}
eolbuf[0] = 0x0d;
eolbuf[1] = 0x0a;
if (fwrite(eolbuf, sizeof(eolbuf), 1, InfFileHandle) < 1)
{
DPRINT(MID_TRACE, ("Error writing INF file.\n"));
return;
}
}
ULONG CDFParser::Load(char* FileName)
/*
* FUNCTION: Loads a directive file into memory
* ARGUMENTS:
* FileName = Pointer to name of directive file
* RETURNS:
* Status of operation
*/
{
LONG FileSize;
if (FileLoaded)
return CAB_STATUS_SUCCESS;
/* Open the directive file */
FileHandle = fopen(FileName, "rb");
if (FileHandle == NULL)
{
return CAB_STATUS_CANNOT_OPEN;
}
FileSize = GetSizeOfFile(FileHandle);
if (FileSize == -1)
{
fclose(FileHandle);
return CAB_STATUS_CANNOT_OPEN;
}
FileBufferSize = (ULONG)FileSize;
FileBuffer = (char*)malloc(FileBufferSize);
if (!FileBuffer)
{
fclose(FileHandle);
return CAB_STATUS_NOMEMORY;
}
if ( fread(FileBuffer, FileBufferSize, 1, FileHandle) < 1 )
{
fclose(FileHandle);
free(FileBuffer);
FileBuffer = NULL;
return CAB_STATUS_CANNOT_READ;
}
fclose(FileHandle);
FileLoaded = true;
DPRINT(MAX_TRACE, ("File (%u bytes)\n", (UINT)FileBufferSize));
return CAB_STATUS_SUCCESS;
}
ULONG CDFParser::Parse()
/*
* FUNCTION: Parses a loaded directive file
* RETURNS:
* Status of operation
*/
{
bool Command;
ULONG Status;
if (!FileLoaded)
return CAB_STATUS_NOFILE;
while (ReadLine())
{
Command = false;
if (InfModeEnabled)
{
bool WriteLine = true;
while (CurrentToken != TokenEnd)
{
switch (CurrentToken)
{
case TokenIdentifier:
if (Command)
{
/* Command */
Status = PerformCommand();
if (Status == CAB_STATUS_FAILURE)
WriteLine = true;
else
if (!InfModeEnabled)
WriteLine = false;
CurrentToken = TokenEnd;
continue;
}
else
{
WriteLine = true;
CurrentToken = TokenEnd;
continue;
}
break;
case TokenSpace:
break;
case TokenPeriod:
Command = true;
break;
default:
WriteLine = true;
CurrentToken = TokenEnd;
continue;
}
NextToken();
}
if (WriteLine)
WriteInfLine(Line);
}
else
{
while (CurrentToken != TokenEnd)
{
switch (CurrentToken)
{
case TokenInteger:
sprintf(CurrentString, "%u", (UINT)CurrentInteger);
case TokenIdentifier:
case TokenString:
if (Command)
{
/* Command */
Status = PerformCommand();
if (Status == CAB_STATUS_FAILURE)
{
printf("ERROR: Directive file contains errors at line %u.\n", (UINT)CurrentLine);
DPRINT(MID_TRACE, ("Error while executing command.\n"));
}
if (Status != CAB_STATUS_SUCCESS)
return Status;
}
else
{
/* File copy */
Status = PerformFileCopy();
if (Status != CAB_STATUS_SUCCESS)
{
printf("ERROR: Directive file contains errors at line %u.\n", (UINT)CurrentLine);
DPRINT(MID_TRACE, ("Error while copying file.\n"));
return Status;
}
}
break;
case TokenSpace:
break;
case TokenSemi:
CurrentToken = TokenEnd;
continue;
case TokenPeriod:
Command = true;
break;
default:
printf("ERROR: Directive file contains errors at line %u.\n", (UINT)CurrentLine);
DPRINT(MID_TRACE, ("Token is (%u).\n", (UINT)CurrentToken));
return CAB_STATUS_SUCCESS;
}
NextToken();
}
}
}
if (!InfFileOnly)
{
OnVerboseMessage("Writing cabinet. This may take a while...\n");
if (DiskCreated)
{
Status = WriteDisk(false);
if (Status == CAB_STATUS_SUCCESS)
Status = CloseDisk();
if (Status != CAB_STATUS_SUCCESS)
{
DPRINT(MIN_TRACE, ("Cannot write disk (%u).\n", (UINT)Status));
return Status;
}
}
if (CabinetCreated)
{
Status = CloseCabinet();
if (Status != CAB_STATUS_SUCCESS)
{
DPRINT(MIN_TRACE, ("Cannot close cabinet (%u).\n", (UINT)Status));
return Status;
}
}
OnVerboseMessage("Done.\n");
}
return CAB_STATUS_SUCCESS;
}
void CDFParser::SetFileRelativePath(char* Path)
/*
* FUNCTION: Sets path where files in the .dff is assumed relative to
* ARGUMENTS:
* Path = Pointer to string with path
*/
{
FileRelativePath = Path;
ConvertPath(FileRelativePath);
NormalizePath(FileRelativePath);
}
bool CDFParser::OnDiskLabel(ULONG Number, char* Label)
/*
* FUNCTION: Called when a disk needs a label
* ARGUMENTS:
* Number = Cabinet number that needs a label
* Label = Pointer to buffer to place label of disk
* RETURNS:
* true if a disk label was returned, false if not
*/
{
char Buffer[20];
ULONG i;
int j;
char ch;
Number += 1;
DPRINT(MID_TRACE, ("Giving disk (%u) a label...\n", (UINT)Number));
if (GetDiskName(&DiskLabel, Number, Label))
return true;
if (DiskLabelTemplateSet)
{
j = 0;
strcpy(Label, "");
for (i = 0; i < strlen(DiskLabelTemplate); i++)
{
ch = DiskLabelTemplate[i];
if (ch == '*')
{
sprintf(Buffer, "%u", (UINT)Number);
strcat(Label, Buffer);
j += (LONG)strlen(Buffer);
}
else
{
Label[j] = ch;
j++;
}
Label[j] = '\0';
}
DPRINT(MID_TRACE, ("Giving disk (%s) as a label...\n", Label));
return true;
}
else
return false;
}
bool CDFParser::OnCabinetName(ULONG Number, char* Name)
/*
* FUNCTION: Called when a cabinet needs a name
* ARGUMENTS:
* Number = Disk number that needs a name
* Name = Pointer to buffer to place name of cabinet
* RETURNS:
* true if a cabinet name was returned, false if not
*/
{
char Buffer[PATH_MAX];
ULONG i;
int j;
char ch;
Number += 1;
DPRINT(MID_TRACE, ("Giving cabinet (%u) a name...\n", (UINT)Number));
if (GetDiskName(&CabinetName, Number, Buffer))
{
strcpy(Name, GetDestinationPath());
strcat(Name, Buffer);
return true;
}
if (CabinetNameTemplateSet)
{
strcpy(Name, GetDestinationPath());
j = (LONG)strlen(Name);
for (i = 0; i < strlen(CabinetNameTemplate); i++)
{
ch = CabinetNameTemplate[i];
if (ch == '*')
{
sprintf(Buffer, "%u", (UINT)Number);
strcat(Name, Buffer);
j += (LONG)strlen(Buffer);
}
else
{
Name[j] = ch;
j++;
}
Name[j] = '\0';
}
DPRINT(MID_TRACE, ("Giving cabinet (%s) as a name...\n", Name));
return true;
}
else
return false;
}
bool CDFParser::SetDiskName(PCABINET_NAME *List, ULONG Number, char* String)
/*
* FUNCTION: Sets an entry in a list
* ARGUMENTS:
* List = Address of pointer to list
* Number = Disk number
* String = Pointer to string
* RETURNS:
* false if there was not enough free memory available
*/
{
PCABINET_NAME CN;
CN = *List;
while (CN != NULL)
{
if (CN->DiskNumber == Number)
{
strcpy(CN->Name, String);
return true;
}
CN = CN->Next;
}
CN = new CABINET_NAME;
if (!CN)
return false;
CN->DiskNumber = Number;
strcpy(CN->Name, String);
CN->Next = *List;
*List = CN;
return true;
}
bool CDFParser::GetDiskName(PCABINET_NAME *List, ULONG Number, char* String)
/*
* FUNCTION: Returns an entry in a list
* ARGUMENTS:
* List = Address of pointer to list
* Number = Disk number
* String = Address of buffer to copy string to
* RETURNS:
* false if there was not enough free memory available
*/
{
PCABINET_NAME CN;
CN = *List;
while (CN != NULL)
{
if (CN->DiskNumber == Number)
{
strcpy(String, CN->Name);
return true;
}
CN = CN->Next;
}
return false;
}
bool CDFParser::SetDiskNumber(PDISK_NUMBER *List, ULONG Number, ULONG Value)
/*
* FUNCTION: Sets an entry in a list
* ARGUMENTS:
* List = Address of pointer to list
* Number = Disk number
* Value = Value to set
* RETURNS:
* false if there was not enough free memory available
*/
{
PDISK_NUMBER DN;
DN = *List;
while (DN != NULL)
{
if (DN->DiskNumber == Number)
{
DN->Number = Value;
return true;
}
DN = DN->Next;
}
DN = new DISK_NUMBER;
if (!DN)
return false;
DN->DiskNumber = Number;
DN->Number = Value;
DN->Next = *List;
*List = DN;
return true;
}
bool CDFParser::GetDiskNumber(PDISK_NUMBER *List, ULONG Number, PULONG Value)
/*
* FUNCTION: Returns an entry in a list
* ARGUMENTS:
* List = Address of pointer to list
* Number = Disk number
* Value = Address of buffer to place value
* RETURNS:
* true if the entry was found
*/
{
PDISK_NUMBER DN;
DN = *List;
while (DN != NULL)
{
if (DN->DiskNumber == Number)
{
*Value = DN->Number;
return true;
}
DN = DN->Next;
}
return false;
}
bool CDFParser::DoDiskLabel(ULONG Number, char* Label)
/*
* FUNCTION: Sets the label of a disk
* ARGUMENTS:
* Number = Disk number
* Label = Pointer to label of disk
* RETURNS:
* false if there was not enough free memory available
*/
{
DPRINT(MID_TRACE, ("Setting label of disk (%u) to '%s'\n", (UINT)Number, Label));
return SetDiskName(&DiskLabel, Number, Label);
}
void CDFParser::DoDiskLabelTemplate(char* Template)
/*
* FUNCTION: Sets a disk label template to use
* ARGUMENTS:
* Template = Pointer to disk label template
*/
{
DPRINT(MID_TRACE, ("Setting disk label template to '%s'\n", Template));
strcpy(DiskLabelTemplate, Template);
DiskLabelTemplateSet = true;
}
bool CDFParser::DoCabinetName(ULONG Number, char* Name)
/*
* FUNCTION: Sets the name of a cabinet
* ARGUMENTS:
* Number = Disk number
* Name = Pointer to name of cabinet
* RETURNS:
* false if there was not enough free memory available
*/
{
DPRINT(MID_TRACE, ("Setting name of cabinet (%u) to '%s'\n", (UINT)Number, Name));
return SetDiskName(&CabinetName, Number, Name);
}
void CDFParser::DoCabinetNameTemplate(char* Template)
/*
* FUNCTION: Sets a cabinet name template to use
* ARGUMENTS:
* Template = Pointer to cabinet name template
*/
{
DPRINT(MID_TRACE, ("Setting cabinet name template to '%s'\n", Template));
strcpy(CabinetNameTemplate, Template);
CabinetNameTemplateSet = true;
}
ULONG CDFParser::DoMaxDiskSize(bool NumberValid, ULONG Number)
/*
* FUNCTION: Sets the maximum disk size
* ARGUMENTS:
* NumberValid = true if disk number is valid
* Number = Disk number
* RETURNS:
* Status of operation
* NOTES:
* Standard sizes are 2.88M, 1.44M, 1.25M, 1.2M, 720K, 360K, and CDROM
*/
{
ULONG A, B, Value;
if (IsNextToken(TokenInteger, true))
{
A = CurrentInteger;
if (IsNextToken(TokenPeriod, false))
{
if (!IsNextToken(TokenInteger, false))
return CAB_STATUS_FAILURE;
B = CurrentInteger;
}
else
B = 0;
if (CurrentToken == TokenIdentifier)
{
switch (CurrentString[0])
{
case 'K':
if (B != 0)
return CAB_STATUS_FAILURE;
if (A == 720)
/* 720K disk */
Value = 730112;
else if (A == 360)
/* 360K disk */
Value = 362496;
else
return CAB_STATUS_FAILURE;
break;
case 'M':
if (A == 1)
{
if (B == 44)
/* 1.44M disk */
Value = 1457664;
else if (B == 25)
/* 1.25M disk */
Value = 1300000; // FIXME: Value?
else if (B == 2)
/* 1.2M disk */
Value = 1213952;
else
return CAB_STATUS_FAILURE;
}
else if (A == 2)
{
if (B == 88)
/* 2.88M disk */
Value = 2915328;
else
return CAB_STATUS_FAILURE;
}
else
return CAB_STATUS_FAILURE;
break;
default:
DPRINT(MID_TRACE, ("Bad suffix (%c)\n", CurrentString[0]));
return CAB_STATUS_FAILURE;
}
}
else
Value = A;
}
else
{
if ((CurrentToken != TokenString) &&
(strcasecmp(CurrentString, "CDROM") != 0))
return CAB_STATUS_FAILURE;
/* CDROM */
Value = 640*1024*1024; // FIXME: Correct size for CDROM?
}
if (NumberValid)
return (SetDiskNumber(&MaxDiskSize, Number, Value)?
CAB_STATUS_SUCCESS : CAB_STATUS_FAILURE);
MaxDiskSizeAll = Value;
MaxDiskSizeAllSet = true;
SetMaxDiskSize(Value);
return CAB_STATUS_SUCCESS;
}
void CDFParser::DoInfFileName(char* FileName)
/*
* FUNCTION: Sets filename of the generated .inf file
* ARGUMENTS:
* FileName = Pointer to .inf filename
*/
{
DPRINT(MID_TRACE, ("Setting .inf filename to '%s'\n", FileName));
strcpy(InfFileName, FileName);
InfFileNameSet = true;
}
ULONG CDFParser::SetupNewDisk()
/*
* FUNCTION: Sets up parameters for a new disk
* RETURNS:
* Status of operation
*/
{
ULONG Value;
if (!GetDiskNumber(&MaxDiskSize, GetCurrentDiskNumber(), &Value))
{
if (MaxDiskSizeAllSet)
Value = MaxDiskSizeAll;
else
Value = 0;
}
SetMaxDiskSize(Value);
return CAB_STATUS_SUCCESS;
}
ULONG CDFParser::PerformSetCommand()
/*
* FUNCTION: Performs a set variable command
* RETURNS:
* Status of operation
*/
{
SETTYPE SetType;
bool NumberValid = false;
ULONG Number = 0;
if (!IsNextToken(TokenIdentifier, true))
return CAB_STATUS_FAILURE;
if (strcasecmp(CurrentString, "DiskLabel") == 0)
SetType = stDiskLabel;
else if (strcasecmp(CurrentString, "DiskLabelTemplate") == 0)
SetType = stDiskLabelTemplate;
else if (strcasecmp(CurrentString, "CabinetName") == 0)
SetType = stCabinetName;
else if (strcasecmp(CurrentString, "CabinetNameTemplate") == 0)
SetType = stCabinetNameTemplate;
else if (strcasecmp(CurrentString, "MaxDiskSize") == 0)
SetType = stMaxDiskSize;
else if (strcasecmp(CurrentString, "InfFileName") == 0)
SetType = stInfFileName;
else
return CAB_STATUS_FAILURE;
if ((SetType == stDiskLabel) || (SetType == stCabinetName))
{
if (!IsNextToken(TokenInteger, false))
return CAB_STATUS_FAILURE;
Number = CurrentInteger;
if (!IsNextToken(TokenEqual, true))
return CAB_STATUS_FAILURE;
}
else if (SetType == stMaxDiskSize)
{
if (IsNextToken(TokenInteger, false))
{
NumberValid = true;
Number = CurrentInteger;
}
else
{
NumberValid = false;
while (CurrentToken == TokenSpace)
NextToken();
if (CurrentToken != TokenEqual)
return CAB_STATUS_FAILURE;
}
}
else if (!IsNextToken(TokenEqual, true))
return CAB_STATUS_FAILURE;
if (SetType != stMaxDiskSize)
{
if (!IsNextToken(TokenString, true))
return CAB_STATUS_FAILURE;
}
switch (SetType)
{
case stDiskLabel:
if (!DoDiskLabel(Number, CurrentString))
DPRINT(MIN_TRACE, ("Not enough available free memory.\n"));
return CAB_STATUS_SUCCESS;
case stCabinetName:
if (!DoCabinetName(Number, CurrentString))
DPRINT(MIN_TRACE, ("Not enough available free memory.\n"));
return CAB_STATUS_SUCCESS;
case stDiskLabelTemplate:
DoDiskLabelTemplate(CurrentString);
return CAB_STATUS_SUCCESS;
case stCabinetNameTemplate:
DoCabinetNameTemplate(CurrentString);
return CAB_STATUS_SUCCESS;
case stMaxDiskSize:
return DoMaxDiskSize(NumberValid, Number);
case stInfFileName:
DoInfFileName(CurrentString);
return CAB_STATUS_SUCCESS;
default:
return CAB_STATUS_FAILURE;
}
}
ULONG CDFParser::PerformNewCommand()
/*
* FUNCTION: Performs a new disk|cabinet|folder command
* RETURNS:
* Status of operation
*/
{
NEWTYPE NewType;
ULONG Status;
if (!IsNextToken(TokenIdentifier, true))
return CAB_STATUS_FAILURE;
if (strcasecmp(CurrentString, "Disk") == 0)
NewType = ntDisk;
else if (strcasecmp(CurrentString, "Cabinet") == 0)
NewType = ntCabinet;
else if (strcasecmp(CurrentString, "Folder") == 0)
NewType = ntFolder;
else
return CAB_STATUS_FAILURE;
switch (NewType)
{
case ntDisk:
if (DiskCreated)
{
Status = WriteDisk(true);
if (Status == CAB_STATUS_SUCCESS)
Status = CloseDisk();
if (Status != CAB_STATUS_SUCCESS)
{
DPRINT(MIN_TRACE, ("Cannot write disk (%u).\n", (UINT)Status));
return CAB_STATUS_SUCCESS;
}
DiskCreated = false;
}
Status = NewDisk();
if (Status != CAB_STATUS_SUCCESS)
{
DPRINT(MIN_TRACE, ("Cannot create disk (%u).\n", (UINT)Status));
return CAB_STATUS_SUCCESS;
}
DiskCreated = true;
SetupNewDisk();
return CAB_STATUS_SUCCESS;
case ntCabinet:
if (DiskCreated)
{
Status = WriteDisk(true);
if (Status == CAB_STATUS_SUCCESS)
Status = CloseDisk();
if (Status != CAB_STATUS_SUCCESS)
{
DPRINT(MIN_TRACE, ("Cannot write disk (%u).\n", (UINT)Status));
return CAB_STATUS_SUCCESS;
}
DiskCreated = false;
}
Status = NewCabinet();
if (Status != CAB_STATUS_SUCCESS)
{
DPRINT(MIN_TRACE, ("Cannot create cabinet (%u).\n", (UINT)Status));
return CAB_STATUS_SUCCESS;
}
DiskCreated = true;
SetupNewDisk();
return CAB_STATUS_SUCCESS;
case ntFolder:
Status = NewFolder();
ASSERT(Status == CAB_STATUS_SUCCESS);
return CAB_STATUS_SUCCESS;
default:
return CAB_STATUS_FAILURE;
}
}
ULONG CDFParser::PerformInfBeginCommand()
/*
* FUNCTION: Begins inf mode
* RETURNS:
* Status of operation
*/
{
InfModeEnabled = true;
return CAB_STATUS_SUCCESS;
}
ULONG CDFParser::PerformInfEndCommand()
/*
* FUNCTION: Begins inf mode
* RETURNS:
* Status of operation
*/
{
InfModeEnabled = false;
return CAB_STATUS_SUCCESS;
}
ULONG CDFParser::PerformCommand()
/*
* FUNCTION: Performs a command
* RETURNS:
* Status of operation
*/
{
if (strcasecmp(CurrentString, "Set") == 0)
return PerformSetCommand();
if (strcasecmp(CurrentString, "New") == 0)
return PerformNewCommand();
if (strcasecmp(CurrentString, "InfBegin") == 0)
return PerformInfBeginCommand();
if (strcasecmp(CurrentString, "InfEnd") == 0)
return PerformInfEndCommand();
return CAB_STATUS_FAILURE;
}
ULONG CDFParser::PerformFileCopy()
/*
* FUNCTION: Performs a file copy
* RETURNS:
* Status of operation
*/
{
ULONG Status;
ULONG i, j;
char ch;
char SrcName[PATH_MAX];
char DstName[PATH_MAX];
char InfLine[PATH_MAX];
char Options[128];
char BaseFilename[PATH_MAX];
*SrcName = '\0';
*DstName = '\0';
*Options = '\0';
// source file
i = CurrentChar;
while ((i < LineLength) &&
((ch = Line[i]) != ' ') &&
(ch != 0x09) &&
(ch != ';'))
{
CurrentString[i] = ch;
i++;
}
CurrentString[i] = '\0';
CurrentToken = TokenString;
CurrentChar = i + 1;
strcpy(BaseFilename, CurrentString);
strcat(SrcName, BaseFilename);
// destination
SkipSpaces();
if (CurrentToken != TokenEnd)
{
j = (ULONG)strlen(CurrentString); i = 0;
while ((CurrentChar + i < LineLength) &&
((ch = Line[CurrentChar + i]) != ' ') &&
(ch != 0x09) &&
(ch != ';'))
{
CurrentString[j + i] = ch;
i++;
}
CurrentString[j + i] = '\0';
CurrentToken = TokenString;
CurrentChar += i + 1;
strcpy(DstName, CurrentString);
}
// options (it may be empty)
SkipSpaces ();
if (CurrentToken != TokenEnd)
{
j = (ULONG)strlen(CurrentString); i = 0;
while ((CurrentChar + i < LineLength) &&
((ch = Line[CurrentChar + i]) != ' ') &&
(ch != 0x09) &&
(ch != ';'))
{
CurrentString[j + i] = ch;
i++;
}
CurrentString[j + i] = '\0';
CurrentToken = TokenString;
CurrentChar += i + 1;
strcpy(Options, CurrentString);
}
if (!CabinetCreated)
{
DPRINT(MID_TRACE, ("Creating cabinet.\n"));
Status = NewCabinet();
if (Status != CAB_STATUS_SUCCESS)
{
DPRINT(MIN_TRACE, ("Cannot create cabinet (%u).\n", (UINT)Status));
printf("ERROR: Cannot create cabinet.\n");
return CAB_STATUS_FAILURE;
}
CabinetCreated = true;
DPRINT(MID_TRACE, ("Creating disk.\n"));
Status = NewDisk();
if (Status != CAB_STATUS_SUCCESS)
{
DPRINT(MIN_TRACE, ("Cannot create disk (%u).\n", (UINT)Status));
printf("ERROR: Cannot create disk.\n");
return CAB_STATUS_FAILURE;
}
DiskCreated = true;
SetupNewDisk();
}
DPRINT(MID_TRACE, ("Adding file: '%s' destination: '%s'.\n", SrcName, DstName));
Status = AddFile(SrcName, std::string());
if (Status == CAB_STATUS_CANNOT_OPEN)
{
strcpy(SrcName, FileRelativePath.c_str());
strcat(SrcName, BaseFilename);
Status = AddFile(SrcName, std::string());
}
switch (Status)
{
case CAB_STATUS_SUCCESS:
sprintf(InfLine, "%s=%s", GetFileName(SrcName).c_str(), DstName);
WriteInfLine(InfLine);
break;
case CAB_STATUS_CANNOT_OPEN:
if (strstr(Options,"optional"))
{
Status = CAB_STATUS_SUCCESS;
printf("Optional file skipped (does not exist): %s.\n", SrcName);
}
else
printf("ERROR: File not found: %s.\n", SrcName);
break;
case CAB_STATUS_NOMEMORY:
printf("ERROR: Insufficient memory to add file: %s.\n", SrcName);
break;
default:
printf("ERROR: Cannot add file: %s (%u).\n", SrcName, (UINT)Status);
break;
}
return Status;
}
void CDFParser::SkipSpaces()
/*
* FUNCTION: Skips any spaces in the current line
*/
{
NextToken();
while (CurrentToken == TokenSpace)
NextToken();
}
bool CDFParser::IsNextToken(DFP_TOKEN Token, bool NoSpaces)
/*
* FUNCTION: Checks if next token equals Token
* ARGUMENTS:
* Token = Token to compare with
* SkipSp = true if spaces should be skipped
* RETURNS:
* false if next token is diffrent from Token
*/
{
if (NoSpaces)
SkipSpaces();
else
NextToken();
return (CurrentToken == Token);
}
bool CDFParser::ReadLine()
/*
* FUNCTION: Reads the next line into the line buffer
* RETURNS:
* true if there is a new line, false if not
*/
{
ULONG i, j;
char ch;
if (CurrentOffset >= FileBufferSize)
return false;
i = 0;
while (((j = CurrentOffset + i) < FileBufferSize) && (i < sizeof(Line) - 1) &&
((ch = FileBuffer[j]) != 0x0D && (ch = FileBuffer[j]) != 0x0A))
{
Line[i] = ch;
i++;
}
Line[i] = '\0';
LineLength = i;
if ((FileBuffer[CurrentOffset + i] == 0x0D) && (FileBuffer[CurrentOffset + i + 1] == 0x0A))
CurrentOffset++;
CurrentOffset += i + 1;
CurrentChar = 0;
CurrentLine++;
NextToken();
return true;
}
void CDFParser::NextToken()
/*
* FUNCTION: Reads the next token from the current line
*/
{
ULONG i;
char ch = ' ';
if (CurrentChar >= LineLength)
{
CurrentToken = TokenEnd;
return;
}
switch (Line[CurrentChar])
{
case ' ':
case 0x09:
CurrentToken = TokenSpace;
break;
case ';':
CurrentToken = TokenSemi;
break;
case '=':
CurrentToken = TokenEqual;
break;
case '.':
CurrentToken = TokenPeriod;
break;
case '\\':
CurrentToken = TokenBackslash;
break;
case '"':
i = 0;
while ((CurrentChar + i + 1 < LineLength) &&
((ch = Line[CurrentChar + i + 1]) != '"'))
{
CurrentString[i] = ch;
i++;
}
CurrentString[i] = '\0';
CurrentToken = TokenString;
CurrentChar += i + 2;
return;
default:
i = 0;
while ((CurrentChar + i < LineLength) &&
((ch = Line[CurrentChar + i]) >= '0') && (ch <= '9'))
{
CurrentString[i] = ch;
i++;
}
if (i > 0)
{
CurrentString[i] = '\0';
CurrentInteger = atoi(CurrentString);
CurrentToken = TokenInteger;
CurrentChar += i;
return;
}
i = 0;
while (((CurrentChar + i < LineLength) &&
(((ch = Line[CurrentChar + i]) >= 'a') && (ch <= 'z'))) ||
((ch >= 'A') && (ch <= 'Z')) || (ch == '_'))
{
CurrentString[i] = ch;
i++;
}
if (i > 0)
{
CurrentString[i] = '\0';
CurrentToken = TokenIdentifier;
CurrentChar += i;
return;
}
CurrentToken = TokenEnd;
}
CurrentChar++;
}
/* EOF */