Source code of Windows XP (NT5)
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
 
 
 
 
 
 

520 lines
11 KiB

//+-------------------------------------------------------------------------
//
// Microsoft Windows
//
// Copyright (C) Microsoft Corporation, 1996 - 1999
//
// File: fparse.cpp
//
// Contents: File parsing api -- INI file types
//
// History: 01-Oct-1997 pberkman create
//
//--------------------------------------------------------------------------
#include "global.hxx"
#include "fparse.hxx"
fParse_::fParse_(WCHAR *pwszFilename, BOOL *pfFailed, DWORD dwMaxLine0, DWORD dwFileAccess, DWORD dwFileSharing)
{
hFile = INVALID_HANDLE_VALUE;
dwMaxLine = dwMaxLine0;
pwszCurrentLine = NULL;
dwCurLineFilePos = 0;
dwLastGroupFilePos = 0;
dwLastTagFilePos = 0;
fEOF = FALSE;
pwszTempFName = NULL;
pwszLastGroupTag = NULL;
__try
{
hFile = CreateFileU(
pwszFilename,
dwFileAccess,
dwFileSharing,
NULL,
OPEN_EXISTING,
FILE_ATTRIBUTE_NORMAL,
NULL);
pwszCurrentLine = new WCHAR[dwMaxLine];
if (pwszFName = new WCHAR[wcslen(pwszFilename) + 1])
{
wcscpy(pwszFName, pwszFilename);
}
if ((hFile == INVALID_HANDLE_VALUE) ||
(pwszCurrentLine == NULL) ||
(pwszFName == NULL))
{
*pfFailed = TRUE;
}
else
{
*pfFailed = FALSE;
}
} // __try
__except(EXCEPTION_EXECUTE_HANDLER)
{
if (hFile != INVALID_HANDLE_VALUE)
{
CloseHandle(hFile);
hFile = INVALID_HANDLE_VALUE;
}
if (pwszCurrentLine != NULL)
{
DELETE_OBJECT(pwszCurrentLine);
pwszCurrentLine = NULL;
}
if (pwszFName != NULL)
{
DELETE_OBJECT(pwszFName);
pwszFName = NULL;
}
*pfFailed = FALSE;
}
}
fParse_::~fParse_(void)
{
if (hFile != INVALID_HANDLE_VALUE)
{
CloseHandle(hFile);
}
if (pwszCurrentLine)
{
DELETE_OBJECT(pwszCurrentLine);
}
DELETE_OBJECT(pwszLastGroupTag);
if (pwszTempFName)
{
CopyFileU(pwszTempFName, pwszFName, FALSE);
DeleteFileU(pwszTempFName);
delete pwszTempFName;
}
if (pwszFName != NULL)
{
DELETE_OBJECT(pwszFName);
}
}
BOOL fParse_::AddTagToFile(WCHAR *pwszGroup, WCHAR *pwszTag, WCHAR *pwszValue)
{
if (!(this->pwszCurrentLine) || (this->hFile == INVALID_HANDLE_VALUE))
{
return(FALSE);
}
char szTFile[MAX_PATH * 2];
WCHAR wszGroup[MAX_PATH];
HANDLE hTFile;
DWORD ccTFile;
DWORD cbWrite;
BOOL fWritten;
if (!(pwszLastGroupTag))
{
return(FALSE);
}
if (pwszTag[0] != L'[')
{
wcscpy(&wszGroup[0], L"[");
wcscat(&wszGroup[0], pwszGroup);
wcscat(&wszGroup[0], L"]");
}
else
{
wcscpy(&wszGroup[0], pwszTag);
}
szTFile[0] = NULL;
GetTempFileName(".", "FPS", 0, &szTFile[0]);
if (!(szTFile[0]))
{
return(FALSE);
}
ccTFile = MultiByteToWideChar(0, 0, &szTFile[0], -1, NULL, 0);
if (ccTFile < 1)
{
return(FALSE);
}
if (!(pwszTempFName = new WCHAR[ccTFile + 1]))
{
return(FALSE);
}
MultiByteToWideChar(0, 0, &szTFile[0], -1, pwszTempFName, ccTFile + 1);
hTFile = CreateFileU(pwszTempFName, GENERIC_READ | GENERIC_WRITE, FILE_SHARE_READ, NULL, CREATE_ALWAYS,
FILE_ATTRIBUTE_NORMAL, NULL);
if (hTFile == INVALID_HANDLE_VALUE)
{
return(FALSE);
}
SetFilePointer(this->hFile, 0, NULL, FILE_BEGIN);
fWritten = FALSE;
while (this->GetNextLine())
{
szTFile[0] = NULL;
WideCharToMultiByte(0, 0, this->pwszCurrentLine, wcslen(this->pwszCurrentLine) + 1,
&szTFile[0], MAX_PATH * 2, NULL, NULL);
if (szTFile[0])
{
WriteFile(hTFile, &szTFile[0], strlen(&szTFile[0]), &cbWrite, NULL);
if (!(fWritten))
{
this->EOLRemove();
if (_memicmp(this->pwszCurrentLine, &wszGroup[0],
wcslen(&wszGroup[0]) * sizeof(WCHAR)) == 0)
{
//
// add our line
//
szTFile[0] = NULL;
WideCharToMultiByte(0, 0, pwszTag, wcslen(pwszTag) + 1, &szTFile[0], MAX_PATH, NULL, NULL);
WriteFile(hTFile, &szTFile[0], strlen(&szTFile[0]), &cbWrite, NULL);
WriteFile(hTFile, "=", 1, &cbWrite, NULL);
szTFile[0] = NULL;
WideCharToMultiByte(0, 0, pwszValue, wcslen(pwszValue) + 1, &szTFile[0], MAX_PATH * 2, NULL, NULL);
WriteFile(hTFile, &szTFile[0], strlen(&szTFile[0]), &cbWrite, NULL);
WriteFile(hTFile, "\r\n", 2, &cbWrite, NULL);
fWritten = TRUE;
}
}
}
}
CloseHandle(hTFile);
this->Reset();
return(TRUE);
}
void fParse_::Reset(void)
{
this->dwCurLineFilePos = 0;
this->dwLastGroupFilePos = 0;
this->dwLastTagFilePos = 0;
SetFilePointer(this->hFile, 0, NULL, FILE_BEGIN);
}
BOOL fParse_::PositionAtLastGroup(void)
{
if (SetFilePointer(this->hFile, this->dwLastGroupFilePos, NULL,
FILE_BEGIN) == 0xFFFFFFFF)
{
return(FALSE);
}
return(TRUE);
}
BOOL fParse_::PositionAtLastTag(void)
{
if (this->dwLastTagFilePos == 0)
{
return(FALSE);
}
if (SetFilePointer(this->hFile, this->dwLastTagFilePos, NULL,
FILE_BEGIN) == 0xFFFFFFFF)
{
return(FALSE);
}
return(TRUE);
}
BOOL fParse_::GetLineInCurrentGroup(void)
{
if (this->dwLastGroupFilePos == 0)
{
return(FALSE);
}
if (this->dwLastTagFilePos == 0)
{
this->PositionAtLastGroup();
}
while (this->GetNextLine() > 0)
{
if ((this->pwszCurrentLine[0] == L'#') ||
(this->pwszCurrentLine[0] == L';') ||
(this->pwszCurrentLine[0] == 0x000d))
{
continue;
}
if (this->pwszCurrentLine[0] == L'[')
{
this->pwszCurrentLine[0] = NULL;
return(FALSE);
}
this->EOLRemove();
if (wcslen(this->pwszCurrentLine) > 0)
{
this->dwLastTagFilePos = this->dwCurLineFilePos;
return(TRUE);
}
}
this->pwszCurrentLine[0] = NULL;
return(FALSE);
}
BOOL fParse_::FindTagInCurrentGroup(WCHAR *pwszTag)
{
if (this->dwLastGroupFilePos == 0)
{
return(FALSE);
}
WCHAR wszCheck[MAX_PATH];
WCHAR wszCheck2[MAX_PATH];
LPWSTR pwszEqual;
DWORD ccRet;
DWORD ccLastMember;
BOOL fFoundLast;
wcscpy(&wszCheck[0], pwszTag);
wcscat(&wszCheck[0], L"=");
wcscpy(&wszCheck2[0], pwszTag);
wcscpy(&wszCheck2[0], L" =");
this->PositionAtLastGroup();
while (this->GetNextLine() > 0)
{
if ((this->pwszCurrentLine[0] == L'#') ||
(this->pwszCurrentLine[0] == L';') ||
(this->pwszCurrentLine[0] == 0x000d))
{
continue;
}
if (this->pwszCurrentLine[0] == L'[')
{
this->pwszCurrentLine[0] = NULL;
return(FALSE);
}
if ((_memicmp(this->pwszCurrentLine, &wszCheck[0], wcslen(&wszCheck[0]) * sizeof(WCHAR)) == 0) ||
(_memicmp(this->pwszCurrentLine, &wszCheck2[0], wcslen(&wszCheck2[0]) * sizeof(WCHAR)) == 0))
{
this->dwLastTagFilePos = this->dwCurLineFilePos;
this->EOLRemove();
return(TRUE);
}
}
this->pwszCurrentLine[0] = NULL;
return(FALSE);
}
BOOL fParse_::FindTagFromCurrentPos(WCHAR *pwszTag)
{
WCHAR wszCheck[MAX_PATH];
WCHAR wszCheck2[MAX_PATH];
LPWSTR pwszEqual;
DWORD ccRet;
DWORD ccLastMember;
BOOL fFoundLast;
wcscpy(&wszCheck[0], pwszTag);
wcscat(&wszCheck[0], L"=");
wcscpy(&wszCheck2[0], pwszTag);
wcscat(&wszCheck2[0], L" =");
this->dwLastTagFilePos++;
while (this->GetNextLine() > 0)
{
if ((this->pwszCurrentLine[0] == L'#') ||
(this->pwszCurrentLine[0] == L';') ||
(this->pwszCurrentLine[0] == 0x000d))
{
continue;
}
if ((_memicmp(this->pwszCurrentLine, &wszCheck[0], wcslen(&wszCheck[0]) * sizeof(WCHAR)) == 0) ||
(_memicmp(this->pwszCurrentLine, &wszCheck2[0], wcslen(&wszCheck2[0]) * sizeof(WCHAR)) == 0))
{
this->dwLastTagFilePos = this->dwCurLineFilePos;
this->EOLRemove();
return(TRUE);
}
}
this->pwszCurrentLine[0] = NULL;
return(FALSE);
}
DWORD fParse_::GetNextLine(void)
{
if (!(this->pwszCurrentLine) ||
(this->hFile == INVALID_HANDLE_VALUE))
{
return(0);
}
DWORD dwHold;
DWORD cbRead;
DWORD cwbRead;
DWORD dw;
int iAmt;
BYTE *pb;
if ((dwHold = SetFilePointer(this->hFile, 0, NULL, FILE_CURRENT)) == 0xFFFFFFFF)
{
return(0);
}
if (!(pb = new BYTE[dwMaxLine + 2]))
{
return(0);
}
cbRead = 0;
if (ReadFile(this->hFile, pb, dwMaxLine, &cbRead, NULL))
{
if (cbRead == 0)
{
this->fEOF = TRUE;
delete pb;
return(0);
}
pb[cbRead] = 0x00;
this->fEOF = FALSE;
if (cbRead > 0)
{
iAmt = 0;
for (dw = 0; dw < (cbRead - 1); dw++)
{
if ((pb[dw] == 0x0d) ||
(pb[dw] == 0x0a))
{
iAmt++;
if (pb[dw + 1] == 0x0a)
{
dw++;
iAmt++;
}
if (SetFilePointer(this->hFile, dwHold + (dw + 1),
NULL, FILE_BEGIN) == 0xFFFFFFFF)
{
this->dwCurLineFilePos = 0;
}
else
{
this->dwCurLineFilePos = SetFilePointer(this->hFile, 0, NULL, FILE_CURRENT) - iAmt;
}
pb[dw + 1] = 0x00;
cwbRead = MultiByteToWideChar(0, 0, (const char *)pb, -1,
pwszCurrentLine, dwMaxLine);
delete pb;
return(cwbRead + 1);
}
}
}
}
else
{
delete pb;
return(0);
}
if (pb[cbRead - 1] == 0x1a) /* EOF */
{
cbRead--;
this->dwCurLineFilePos = 0;
this->fEOF = TRUE;
}
else
{
this->dwCurLineFilePos = dwHold;
}
pb[cbRead] = 0x00;
cwbRead = MultiByteToWideChar(0, 0, (const char *)pb, -1,
pwszCurrentLine, dwMaxLine);
delete pb;
return(cwbRead);
}
void fParse_::EOLRemove(void)
{
DWORD i;
DWORD ccLen;
ccLen = wcslen(this->pwszCurrentLine);
for (i = 0; i < ccLen; i++)
{
if ((this->pwszCurrentLine[i] == (WCHAR)0x0a) ||
(this->pwszCurrentLine[i] == (WCHAR)0x0d))
{
this->pwszCurrentLine[i] = NULL;
return;
}
}
this->pwszCurrentLine[ccLen] = NULL;
}