mirror of
https://github.com/rn10950/RetroZilla.git
synced 2024-11-16 12:30:13 +01:00
618 lines
16 KiB
C++
618 lines
16 KiB
C++
/* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 2 -*- */
|
|
/* ***** BEGIN LICENSE BLOCK *****
|
|
* Version: MPL 1.1/GPL 2.0/LGPL 2.1
|
|
*
|
|
* The contents of this file are subject to the Mozilla Public License Version
|
|
* 1.1 (the "License"); you may not use this file except in compliance with
|
|
* the License. You may obtain a copy of the License at
|
|
* http://www.mozilla.org/MPL/
|
|
*
|
|
* Software distributed under the License is distributed on an "AS IS" basis,
|
|
* WITHOUT WARRANTY OF ANY KIND, either express or implied. See the License
|
|
* for the specific language governing rights and limitations under the
|
|
* License.
|
|
*
|
|
* The Original Code is Mozilla Communicator client code, released
|
|
* March 31, 1998.
|
|
*
|
|
* The Initial Developer of the Original Code is
|
|
* Netscape Communications Corporation.
|
|
* Portions created by the Initial Developer are Copyright (C) 1998
|
|
* the Initial Developer. All Rights Reserved.
|
|
*
|
|
* Contributor(s):
|
|
* Samir Gehani <sgehani@netscape.com>
|
|
*
|
|
* Alternatively, the contents of this file may be used under the terms of
|
|
* either of the GNU General Public License Version 2 or later (the "GPL"),
|
|
* or the GNU Lesser General Public License Version 2.1 or later (the "LGPL"),
|
|
* in which case the provisions of the GPL or the LGPL are applicable instead
|
|
* of those above. If you wish to allow use of your version of this file only
|
|
* under the terms of either the GPL or the LGPL, and not to allow others to
|
|
* use your version of this file under the terms of the MPL, indicate your
|
|
* decision by deleting the provisions above and replace them with the notice
|
|
* and other provisions required by the GPL or the LGPL. If you do not delete
|
|
* the provisions above, a recipient may use your version of this file under
|
|
* the terms of any one of the MPL, the GPL or the LGPL.
|
|
*
|
|
* ***** END LICENSE BLOCK ***** */
|
|
|
|
|
|
#include "nsINIParser.h"
|
|
|
|
#ifdef __cplusplus
|
|
extern "C" {
|
|
#endif
|
|
unsigned long GetPrivateProfileString(const char* szAppName,
|
|
const char* szKeyName,
|
|
const char* szDefault,
|
|
char* szReturnedString,
|
|
int nSize,
|
|
const char* szFileName)
|
|
{
|
|
nsINIParser parser((char*)szFileName);
|
|
if (parser.GetString((char*)szAppName, (char*)szKeyName, szReturnedString, &nSize) != nsINIParser::OK) {
|
|
if (szDefault) {
|
|
strcpy(szReturnedString, szDefault);
|
|
}
|
|
}
|
|
return strlen(szReturnedString);
|
|
}
|
|
|
|
unsigned long WritePrivateProfileString(const char* szAppName,
|
|
const char* szKeyName,
|
|
const char* szValue,
|
|
const char* szFileName)
|
|
{
|
|
nsINIParser parser((char*)szFileName);
|
|
if (parser.WriteString((char*)szAppName, (char*)szKeyName, (char*)szValue) != nsINIParser::OK) {
|
|
return 0;
|
|
}
|
|
return 1;
|
|
}
|
|
#ifdef __cplusplus
|
|
}
|
|
#endif
|
|
|
|
nsINIParser::nsINIParser(char *aFilename)
|
|
{
|
|
FILE *fd = NULL;
|
|
long eofpos = 0;
|
|
int rd = 0;
|
|
|
|
mfWrite = 0;
|
|
mFileBuf = NULL;
|
|
mFilename = (char*)malloc(strlen(aFilename)+1);
|
|
strcpy(mFilename, aFilename);
|
|
mFileBufSize = 0;
|
|
mError = OK;
|
|
|
|
/* param check */
|
|
if (!aFilename)
|
|
{
|
|
mError = E_PARAM;
|
|
return;
|
|
}
|
|
|
|
/* open the file */
|
|
fd = fopen(aFilename, "r");
|
|
if (!fd)
|
|
goto bail;
|
|
|
|
/* get file size */
|
|
if (fseek(fd, 0, SEEK_END) != 0)
|
|
goto bail;
|
|
eofpos = ftell(fd);
|
|
if (eofpos == 0)
|
|
goto bail;
|
|
|
|
/* malloc an internal buf the size of the file */
|
|
mFileBuf = (char *) calloc(1, eofpos * sizeof(char));
|
|
if (!mFileBuf)
|
|
{
|
|
mError = E_MEM;
|
|
return;
|
|
}
|
|
mFileBufSize = eofpos;
|
|
|
|
/* read the file in one swoop */
|
|
if (fseek(fd, 0, SEEK_SET) != 0)
|
|
goto bail;
|
|
rd = fread((void *)mFileBuf, 1, eofpos, fd);
|
|
if (!rd)
|
|
goto bail;
|
|
|
|
/* actual number of bytes read is less than the size of the file */
|
|
/* due to /r/n -> /n conversion */
|
|
if (rd < eofpos) {
|
|
mFileBufSize = rd;
|
|
} else {
|
|
mFileBufSize = eofpos;
|
|
}
|
|
|
|
/* close file */
|
|
fclose(fd);
|
|
|
|
return;
|
|
|
|
bail:
|
|
mError = E_READ;
|
|
return;
|
|
}
|
|
|
|
nsINIParser::~nsINIParser()
|
|
{
|
|
if (mfWrite) {
|
|
FILE* hFile = fopen(mFilename, "w+");
|
|
fwrite(mFileBuf, mFileBufSize, 1, (FILE *)hFile);
|
|
fclose(hFile);
|
|
}
|
|
free(mFileBuf);
|
|
free(mFilename);
|
|
}
|
|
|
|
int
|
|
nsINIParser::GetString( char *aSection, char *aKey,
|
|
char *aValBuf, int *aIOValBufSize )
|
|
{
|
|
char *secPtr = NULL;
|
|
mError = OK;
|
|
DUMP("GetString");
|
|
|
|
/* param check */
|
|
if ( !aSection || !aValBuf ||
|
|
!aIOValBufSize || (*aIOValBufSize <= 0) )
|
|
return E_PARAM;
|
|
|
|
/* find the section if it exists */
|
|
mError = FindSection(aSection, &secPtr);
|
|
if (mError != OK)
|
|
return mError;
|
|
|
|
if (aKey) {
|
|
/* find the key if it exists in the valid section we found */
|
|
mError = GetValue(secPtr, aKey, aValBuf, aIOValBufSize);
|
|
} else {
|
|
mError = GetAllKeys(secPtr, aValBuf, aIOValBufSize);
|
|
}
|
|
|
|
|
|
return mError;
|
|
}
|
|
|
|
int
|
|
nsINIParser::GetStringAlloc( char *aSection, char *aKey,
|
|
char **aOutBuf, int *aOutBufSize )
|
|
{
|
|
char buf[MAX_VAL_SIZE];
|
|
int bufsize = MAX_VAL_SIZE;
|
|
mError = OK;
|
|
DUMP("GetStringAlloc");
|
|
|
|
mError = GetString(aSection, aKey, buf, &bufsize);
|
|
if (mError != OK)
|
|
return mError;
|
|
|
|
*aOutBuf = (char *) malloc(bufsize + 1);
|
|
strncpy(*aOutBuf, buf, bufsize);
|
|
*(*aOutBuf + bufsize) = 0;
|
|
*aOutBufSize = bufsize + 1;
|
|
|
|
return mError;
|
|
}
|
|
|
|
int
|
|
nsINIParser::GetError()
|
|
{
|
|
DUMP("GetError");
|
|
return mError;
|
|
}
|
|
|
|
char *
|
|
nsINIParser::ResolveName(char *aINIRoot)
|
|
{
|
|
char *resolved = NULL;
|
|
char *locale = NULL;
|
|
struct stat st_exists;
|
|
|
|
/* param check */
|
|
if (!aINIRoot)
|
|
return NULL;
|
|
|
|
locale = setlocale(LC_CTYPE, NULL);
|
|
if (!locale)
|
|
return NULL;
|
|
|
|
/* resolved string: "<root>.ini.<locale>\0" */
|
|
resolved = (char *) malloc(strlen(aINIRoot) + 5 + strlen(locale) + 1);
|
|
if (!resolved)
|
|
return NULL;
|
|
|
|
/* locale specific ini file name */
|
|
sprintf(resolved, "%s.ini.%s", aINIRoot, locale);
|
|
if (0 == stat(resolved, &st_exists))
|
|
return resolved;
|
|
|
|
/* fallback to general ini file name */
|
|
sprintf(resolved, "%s.ini", aINIRoot);
|
|
if (0 == stat(resolved, &st_exists))
|
|
return resolved;
|
|
|
|
/* neither existed so error returned */
|
|
return NULL;
|
|
}
|
|
|
|
int
|
|
nsINIParser::WriteString(char *aSection, char *aKey, char *aValBuf)
|
|
{
|
|
mfWrite = 1;
|
|
char *secPtr = NULL;
|
|
char *keyPtr = NULL;
|
|
mError = OK;
|
|
DUMP("GetString");
|
|
|
|
/* param check */
|
|
if ( !aSection || !aKey || !aValBuf )
|
|
return E_PARAM;
|
|
|
|
/* find the section if it exists */
|
|
mError = FindSection(aSection, &secPtr);
|
|
if (mError != OK) {
|
|
/* if the seciont doesn't exist, add it to the END of the buffer */
|
|
char szNewSection[MAX_VAL_SIZE];
|
|
strcpy(szNewSection, "[");
|
|
strcat(szNewSection, aSection);
|
|
strcat(szNewSection, "]");
|
|
strcat(szNewSection, NLSTRING);
|
|
mFileBuf = (char*)realloc(mFileBuf, mFileBufSize+strlen(szNewSection));
|
|
memcpy(&mFileBuf[mFileBufSize], szNewSection, strlen(szNewSection));
|
|
secPtr = &mFileBuf[mFileBufSize];
|
|
mFileBufSize+= strlen(szNewSection);
|
|
mError = OK;
|
|
}
|
|
|
|
/* find the key if it exists in the valid section we found */
|
|
mError = FindKey(secPtr, aKey, &keyPtr);
|
|
if (mError != OK) {
|
|
char szNewKeyValue[MAX_VAL_SIZE];
|
|
strcpy(szNewKeyValue, aKey);
|
|
strcat(szNewKeyValue, "=");
|
|
strcat(szNewKeyValue, aValBuf);
|
|
strcat(szNewKeyValue, NLSTRING);
|
|
char *mNewFileBuf = (char*)calloc(1, mFileBufSize+strlen(szNewKeyValue));
|
|
memcpy(mNewFileBuf, mFileBuf, mFileBufSize);
|
|
/* Set the section pointer to the location of the rest of the buffer */
|
|
while (*secPtr != NL) {
|
|
secPtr++;
|
|
}
|
|
secPtr++;
|
|
/* Use sectionptr to compte where in the new buf we are going to insert */
|
|
memcpy(mNewFileBuf+(secPtr-mFileBuf), szNewKeyValue, strlen(szNewKeyValue));
|
|
memcpy(mNewFileBuf+(secPtr-mFileBuf)+strlen(szNewKeyValue), secPtr, mFileBufSize-(secPtr-mFileBuf));
|
|
mFileBufSize += strlen(szNewKeyValue);
|
|
free(mFileBuf);
|
|
mFileBuf = mNewFileBuf;
|
|
mError = OK;
|
|
} else {
|
|
while (*keyPtr != '=') {
|
|
keyPtr++;
|
|
}
|
|
keyPtr++;
|
|
int length = 0;
|
|
char* lenPtr = keyPtr;
|
|
while (*lenPtr != NL) {
|
|
lenPtr++;
|
|
length++;
|
|
}
|
|
int difference;
|
|
difference = strlen(aValBuf)- length;
|
|
char *mNewFileBuf = (char*)calloc(1, mFileBufSize+difference);
|
|
memcpy(mNewFileBuf, mFileBuf, mFileBufSize);
|
|
/* Use keyptr to compte where in the new buf we are going to insert */
|
|
memcpy(mNewFileBuf+(keyPtr-mFileBuf), aValBuf, strlen(aValBuf));
|
|
|
|
|
|
memcpy(mNewFileBuf+(keyPtr-mFileBuf)+strlen(aValBuf), lenPtr, mFileBufSize-(lenPtr-mFileBuf));
|
|
mFileBufSize += difference;
|
|
free(mFileBuf);
|
|
mFileBuf = mNewFileBuf;
|
|
mError = OK;
|
|
|
|
|
|
}
|
|
|
|
return mError;
|
|
}
|
|
|
|
int
|
|
nsINIParser::FindSection(char *aSection, char **aOutSecPtr)
|
|
{
|
|
char *currChar = mFileBuf;
|
|
char *nextSec = NULL;
|
|
char *secClose = NULL;
|
|
char *nextNL = NULL;
|
|
mError = E_NO_SEC;
|
|
DUMP("FindSection");
|
|
|
|
// param check
|
|
if (!aSection || !aOutSecPtr)
|
|
{
|
|
mError = E_PARAM;
|
|
return mError;
|
|
}
|
|
|
|
while (currChar < (mFileBuf + mFileBufSize))
|
|
{
|
|
// look for first '['
|
|
nextSec = NULL;
|
|
nextSec = strchr(currChar, '[');
|
|
if (!nextSec)
|
|
break;
|
|
|
|
currChar = nextSec + 1;
|
|
|
|
// extract section name till first ']'
|
|
secClose = NULL; nextNL = NULL;
|
|
secClose = strchr(currChar, ']');
|
|
nextNL = strchr(currChar, NL);
|
|
if ((!nextNL) || (nextNL < secClose))
|
|
{
|
|
currChar = nextNL;
|
|
continue;
|
|
}
|
|
|
|
// if section name matches we succeeded
|
|
if (strnicmp(aSection, currChar, strlen(aSection)) == 0)
|
|
{
|
|
*aOutSecPtr = secClose + 1;
|
|
mError = OK;
|
|
break;
|
|
}
|
|
}
|
|
|
|
return mError;
|
|
}
|
|
|
|
int
|
|
nsINIParser::GetValue(char *aSecPtr, char *aKey, char *aVal, int *aIOValSize)
|
|
{
|
|
char *nextNL = NULL;
|
|
char *secEnd = NULL;
|
|
char *currLine = aSecPtr;
|
|
char *nextEq = NULL;
|
|
mError = E_NO_KEY;
|
|
DUMP("FindKey");
|
|
|
|
// param check
|
|
if (!aSecPtr || !aKey || !aVal || !aIOValSize || (*aIOValSize <= 0))
|
|
{
|
|
mError = E_PARAM;
|
|
return mError;
|
|
}
|
|
|
|
// determine the section end
|
|
secEnd = aSecPtr;
|
|
find_end:
|
|
if (secEnd)
|
|
secEnd = strchr(secEnd, '['); // search for next sec start
|
|
if (!secEnd)
|
|
{
|
|
secEnd = strchr(aSecPtr, '\0'); // else search for file end
|
|
if (!secEnd)
|
|
{
|
|
mError = E_SEC_CORRUPT; // else this data is corrupt
|
|
return mError;
|
|
}
|
|
}
|
|
|
|
// handle start section token ('[') in values for i18n
|
|
if (*secEnd == '[' && !(secEnd == aSecPtr || *(secEnd-1) == NL))
|
|
{
|
|
secEnd++;
|
|
goto find_end;
|
|
}
|
|
|
|
while (currLine < secEnd)
|
|
{
|
|
nextNL = NULL;
|
|
nextNL = strchr(currLine, NL);
|
|
if (!nextNL)
|
|
nextNL = mFileBuf + mFileBufSize;
|
|
|
|
// ignore commented lines (starting with ;)
|
|
if (currLine == strchr(currLine, ';'))
|
|
{
|
|
currLine = nextNL + 1;
|
|
continue;
|
|
}
|
|
|
|
// extract key before '='
|
|
nextEq = NULL;
|
|
nextEq = strchr(currLine, '=');
|
|
if (!nextEq || nextEq > nextNL)
|
|
{
|
|
currLine = nextNL + 1;
|
|
continue;
|
|
}
|
|
|
|
// if key matches we succeeded
|
|
if (strnicmp(currLine, aKey, strlen(aKey)) == 0)
|
|
{
|
|
// extract the value and return
|
|
if (*aIOValSize < nextNL - nextEq)
|
|
{
|
|
mError = E_SMALL_BUF;
|
|
*aVal = '\0';
|
|
*aIOValSize = 0;
|
|
return mError;
|
|
}
|
|
|
|
*aIOValSize = nextNL - (nextEq + 1);
|
|
strncpy(aVal, (nextEq + 1), *aIOValSize);
|
|
*(aVal + *aIOValSize) = 0; // null terminate
|
|
mError = OK;
|
|
return mError;
|
|
}
|
|
else
|
|
{
|
|
currLine = nextNL + 1;
|
|
}
|
|
}
|
|
|
|
return mError;
|
|
}
|
|
|
|
int
|
|
nsINIParser::GetAllKeys(char *aSecPtr, char *aVal, int *aIOValSize)
|
|
{
|
|
char *nextNL = NULL;
|
|
char *secEnd = NULL;
|
|
char *currLine = aSecPtr;
|
|
char *nextEq = NULL;
|
|
char *outPtr = NULL;
|
|
mError = E_NO_KEY;
|
|
DUMP("FindKey");
|
|
|
|
// param check
|
|
if (!aSecPtr || !aVal || !aIOValSize || (*aIOValSize <= 0))
|
|
{
|
|
mError = E_PARAM;
|
|
return mError;
|
|
}
|
|
|
|
// determine the section end
|
|
secEnd = aSecPtr;
|
|
find_end:
|
|
if (secEnd)
|
|
secEnd = strchr(secEnd, '['); // search for next sec start
|
|
if (!secEnd)
|
|
{
|
|
secEnd = strchr(aSecPtr, '\0'); // else search for file end
|
|
if (!secEnd)
|
|
{
|
|
mError = E_SEC_CORRUPT; // else this data is corrupt
|
|
return mError;
|
|
}
|
|
}
|
|
|
|
// handle start section token ('[') in values for i18n
|
|
if (*secEnd == '[' && !(secEnd == aSecPtr || *(secEnd-1) == NL))
|
|
{
|
|
secEnd++;
|
|
goto find_end;
|
|
}
|
|
|
|
outPtr = aVal;
|
|
|
|
while (currLine < secEnd)
|
|
{
|
|
nextNL = NULL;
|
|
nextNL = strchr(currLine, NL);
|
|
if (!nextNL)
|
|
nextNL = mFileBuf + mFileBufSize;
|
|
|
|
// ignore commented lines (starting with ;)
|
|
if (currLine == strchr(currLine, ';'))
|
|
{
|
|
currLine = nextNL + 1;
|
|
continue;
|
|
}
|
|
|
|
// extract key before '='
|
|
nextEq = NULL;
|
|
nextEq = strchr(currLine, '=');
|
|
if (!nextEq || nextEq > nextNL)
|
|
{
|
|
currLine = nextNL + 1;
|
|
continue;
|
|
}
|
|
|
|
strncpy(aVal, currLine, nextEq-currLine);
|
|
aVal+= nextEq-currLine;
|
|
*aVal = '\0';
|
|
aVal++;
|
|
|
|
currLine = nextNL + 1;
|
|
}
|
|
|
|
return OK;
|
|
}
|
|
|
|
int
|
|
nsINIParser::FindKey(char *aSecPtr, char *aKey, char **aOutKeyPtr)
|
|
{
|
|
char *nextNL = NULL;
|
|
char *secEnd = NULL;
|
|
char *currLine = aSecPtr;
|
|
char *nextEq = NULL;
|
|
mError = E_NO_KEY;
|
|
DUMP("FindKey");
|
|
|
|
// param check
|
|
if (!aSecPtr || !aKey || !aOutKeyPtr)
|
|
{
|
|
mError = E_PARAM;
|
|
return mError;
|
|
}
|
|
|
|
// determine the section end
|
|
secEnd = aSecPtr;
|
|
find_end:
|
|
if (secEnd)
|
|
secEnd = strchr(secEnd, '['); // search for next sec start
|
|
if (!secEnd)
|
|
{
|
|
secEnd = strchr(aSecPtr, '\0'); // else search for file end
|
|
if (!secEnd)
|
|
{
|
|
mError = E_SEC_CORRUPT; // else this data is corrupt
|
|
return mError;
|
|
}
|
|
}
|
|
|
|
// handle start section token ('[') in values for i18n
|
|
if (*secEnd == '[' && !(secEnd == aSecPtr || *(secEnd-1) == NL))
|
|
{
|
|
secEnd++;
|
|
goto find_end;
|
|
}
|
|
|
|
while (currLine < secEnd)
|
|
{
|
|
nextNL = NULL;
|
|
nextNL = strchr(currLine, NL);
|
|
if (!nextNL)
|
|
nextNL = mFileBuf + mFileBufSize;
|
|
|
|
// ignore commented lines (starting with ;)
|
|
if (currLine == strchr(currLine, ';'))
|
|
{
|
|
currLine = nextNL + 1;
|
|
continue;
|
|
}
|
|
|
|
// extract key before '='
|
|
nextEq = NULL;
|
|
nextEq = strchr(currLine, '=');
|
|
if (!nextEq || nextEq > nextNL)
|
|
{
|
|
currLine = nextNL + 1;
|
|
continue;
|
|
}
|
|
|
|
// if key matches we succeeded
|
|
if (strnicmp(currLine, aKey, strlen(aKey)) == 0)
|
|
{
|
|
*aOutKeyPtr = currLine;
|
|
mError = OK;
|
|
return mError;
|
|
}
|
|
else
|
|
{
|
|
currLine = nextNL + 1;
|
|
}
|
|
}
|
|
|
|
return mError;
|
|
}
|