windows-nt/Source/XPSP1/NT/printscan/fax/provider/t30/main/registry.c
2020-09-26 16:20:57 +08:00

809 lines
18 KiB
C

/***************************************************************************
Name : REGISTRY.C
Comment : INIfile handling
Revision Log
Date Name Description
-------- ----- ---------------------------------------------------------
***************************************************************************/
#include "prep.h"
#include "efaxcb.h"
#include "t30.h"
#include "hdlc.h"
#include "debug.h"
#include "glbproto.h"
#ifdef USE_REGISTRY
// The default database to us
# define DATABASE_KEY HKEY_LOCAL_MACHINE
// These are NOT localizable items.
# define szKEYPREFIX "SOFTWARE\\Microsoft\\Fax\\Devices"
# define szKEYCLASS "DATA"
DWORD my_atoul(LPSTR lpsz);
#else // !USE_REGISTRY
BOOL expand_key(DWORD dwKey, LPSTR FAR *lplpszKey,
LPSTR FAR *lplpszProfileName);
#endif // !USE_REGISTRY
ULONG_PTR ProfileOpen(DWORD dwProfileID, LPSTR lpszSection, DWORD dwFlags)
{
ULONG_PTR dwRet = 0;
#ifdef USE_REGISTRY
char rgchKey[128];
HKEY hKey=0;
HKEY hBaseKey = DATABASE_KEY;
LONG l;
LPSTR lpszPrefix;
DWORD sam=0;
BG_CHK(sizeof(hKey)<=sizeof(ULONG_PTR));
if (dwProfileID==OEM_BASEKEY)
{
hBaseKey = HKEY_LOCAL_MACHINE;
lpszPrefix= ""; // we don't prepend szKEYPREFIX
if (!lpszSection) goto failure;
}
else if (lpszSection)
{
lpszPrefix= szKEYPREFIX "\\";
}
else
{
lpszPrefix= szKEYPREFIX;
lpszSection="";
}
if ((lstrlen(lpszPrefix)+lstrlen(lpszSection))>=sizeof(rgchKey))
goto failure;
lstrcpy(rgchKey, lpszPrefix);
lstrcat(rgchKey, lpszSection);
sam = 0;
if (dwFlags &fREG_READ) sam |= KEY_READ;
if (dwFlags &fREG_WRITE) sam |= KEY_WRITE;
if (dwFlags & fREG_CREATE)
{
DWORD dwDisposition=0;
DWORD dwOptions = (dwFlags & fREG_VOLATILE)
?REG_OPTION_VOLATILE
:REG_OPTION_NON_VOLATILE;
sam = KEY_READ | KEY_WRITE; // we force sam to this when creating.
l = RegCreateKeyEx(
hBaseKey, // handle of open key
rgchKey, // address of name of subkey to open
0, // reserved
szKEYCLASS, // address of class string
dwOptions, // special options flag
sam, // desired security access
NULL, // address of key security structure
&hKey, // address of buffer for opened handle
&dwDisposition // address of dispostion value buffer
);
}
else
{
l = RegOpenKeyEx(
hBaseKey, // handle of open key
rgchKey, // address of name of subkey to open
0, // reserved
sam , // desired security access
&hKey // address of buffer for opened handle
);
}
if (l!=ERROR_SUCCESS)
{
//LOG((_ERR, "RegCreateKeyEx returns error %ld\n", (long) l));
goto failure;
}
dwRet = (ULONG_PTR) hKey;
#else // !USE_REGISTRY
LPSTR lpszFile= szINIFILE;
ATOM aSection, aProfile;
// NULL lpszSection not supported.
if (!lpszSection) goto failure;
if (!(aSection = GlobalAddAtom(lpszSection)))
goto failure;
if (!(aProfile = GlobalAddAtom(lpszFile)))
{GlobalDeleteAtom(aSection); goto failure;}
dwRet = MAKELONG(aSection, aProfile);
#endif // !USE_REGISTRY
BG_CHK(dwRet);
return dwRet;
failure:
return 0;
}
UINT
ProfileListGetInt(
ULONG_PTR KeyList[10],
LPSTR lpszValueName,
UINT uDefault
)
{
int i;
int Num=0;
UINT uRet = uDefault;
BOOL fExist = 0;
for (i=0; i<10; i++) {
if (KeyList[i] == 0) {
Num = i-1;
break;
}
}
for (i=Num; i>=0; i--) {
uRet = ProfileGetInt (KeyList[i], lpszValueName, uDefault, &fExist);
if (fExist) {
return uRet;
}
}
return uRet;
}
UINT ProfileGetInt(ULONG_PTR dwKey, LPSTR lpszValueName, UINT uDefault, BOOL *fExist)
{
UINT uRet = uDefault;
char rgchBuf[128];
DWORD dwType;
DWORD dwcbSize=sizeof(rgchBuf);
LONG l = RegQueryValueEx(
(HKEY) dwKey,
lpszValueName,
0,
&dwType,
rgchBuf,
&dwcbSize
);
if (fExist) {
*fExist = 0;
}
if (l!=ERROR_SUCCESS)
{
//LOG((_ERR, "RegQueryValueEx returned error %ld\n", (long) l));
goto end;
}
if (fExist) {
*fExist = 1;
}
if (dwType != REG_SZ)
{
//LOG((_ERR, "RegQueryValueEx value type not string:0x%lx\n",
// (unsigned long) dwType));
goto end;
}
uRet = (UINT) my_atoul(rgchBuf);
end:
return uRet;
}
DWORD ProfileGetString(ULONG_PTR dwKey, LPSTR lpszValueName,
LPSTR lpszDefault, LPSTR lpszBuf , DWORD dwcbMax)
{
DWORD dwRet = 0;
#ifdef USE_REGISTRY
DWORD dwType;
LONG l = RegQueryValueEx(
(HKEY) dwKey,
lpszValueName,
0,
&dwType,
lpszBuf,
&dwcbMax
);
if (l!=ERROR_SUCCESS)
{
//LOG((_ERR, "RegQueryValueEx returned error %ld\n", (long) l));
goto copy_default;
}
if (dwType != REG_SZ)
{
//LOG((_ERR, "RegQueryValueEx value type not string:0x%lx\n",
// (unsigned long) dwType));
goto copy_default;
}
// Make sure we null-terminate the string and return the true string
// length..
if (dwcbMax) {lpszBuf[dwcbMax-1]=0; dwcbMax = (DWORD) lstrlen(lpszBuf);}
dwRet = dwcbMax;
goto end;
#else // USE_REGISTRY
LPSTR lpszKey;
LPSTR lpszProfileName;
if (!expand_key(dwKey, &lpszKey, &lpszProfileName)) goto copy_default;
dwRet = GetPrivateProfileString(lpszKey, lpszValueName,
lpszDefault, lpszBuf , (INT) dwcbMax,
lpszProfileName);
goto end;
#endif // USE_REGISTRY
BG_CHK(FALSE);
copy_default:
dwRet = 0;
if (!lpszDefault || !*lpszDefault)
{
if (dwcbMax) *lpszBuf=0;
}
else
{
UINT cb = _fstrlen(lpszDefault)+1;
if (cb>(UINT)dwcbMax) cb=dwcbMax;
if (cb)
{
_fmemcpy(lpszBuf, lpszDefault, cb);
lpszBuf[cb-1]=0;
dwRet = cb-1;
}
}
// fall through...
end:
return dwRet;
}
BOOL
ProfileWriteString(
ULONG_PTR dwKey,
LPSTR lpszValueName,
LPSTR lpszBuf,
BOOL fRemoveCR
)
{
// NOTE: If lpszValueName is null, delete the key. (can't do this in,
// the registry, unfortunately).
// If lpszBuf is null pointer -- "delete" this value.
BOOL fRet=FALSE;
#ifdef USE_REGISTRY
LONG l;
if (!lpszValueName) goto end;
if (!lpszBuf)
{
// delete value...
l = RegDeleteValue((HKEY) dwKey, lpszValueName);
if (l!=ERROR_SUCCESS) goto end;
}
else
{
if (fRemoveCR) {
RemoveCR (lpszBuf);
}
l = RegSetValueEx((HKEY) dwKey, lpszValueName, 0, REG_SZ,
lpszBuf, lstrlen(lpszBuf)+1);
if (l!=ERROR_SUCCESS)
{
//LOG((_ERR,
// "RegSetValueEx(\"%s\", \"%s\") returned error %ld\n",
// (LPSTR) lpszValueName,
// (LPSTR) lpszBuf,
// (long) l));
goto end;
}
}
fRet = TRUE;
goto end;
#else // !USE_REGISTRY
LPSTR lpszKey;
LPSTR lpszProfileName;
if (!expand_key(dwKey, &lpszKey, &lpszProfileName)) goto end;
fRet = WritePrivateProfileString(lpszKey, lpszValueName,
lpszBuf, lpszProfileName);
goto end;
#endif // !USE_REGISTRY
BG_CHK(FALSE);
end:
return fRet;
}
void ProfileClose(ULONG_PTR dwKey)
{
#ifdef USE_REGISTRY
if (RegCloseKey((HKEY)dwKey)!=ERROR_SUCCESS)
{
//LOG((_WRN, "Couldn't close registry key:%lu\n\r",
// (unsigned long) dwKey));
}
#else
BG_CHK(LOWORD(dwKey)); GlobalDeleteAtom(LOWORD(dwKey));
BG_CHK(HIWORD(dwKey)); GlobalDeleteAtom(HIWORD(dwKey));
#endif
}
BOOL ProfileDeleteSection(DWORD dwProfileID, LPSTR lpszSection)
{
#ifdef USE_REGISTRY
char rgchKey[128];
LPSTR lpszPrefix= szKEYPREFIX "\\";
if (dwProfileID==OEM_BASEKEY) goto failure; // Can't delete this
if ((lstrlen(lpszPrefix)+lstrlen(lpszSection))>=sizeof(rgchKey))
goto failure;
lstrcpy(rgchKey, lpszPrefix);
lstrcat(rgchKey, lpszSection);
return (RegDeleteKey(DATABASE_KEY, rgchKey)==ERROR_SUCCESS);
failure:
return FALSE;
#else // !USE_REGISTRY
return WritePrivateProfileString(lpszSection, NULL, NULL, szINIFILE);
#endif // !USE_REGISTRY
}
BOOL
ProfileCopyTree(
DWORD dwProfileIDTo,
LPSTR lpszSectionTo,
DWORD dwProfileIDFr,
LPSTR lpszSectionFr)
{
BOOL fRet=TRUE;
char SecTo[200];
char SecFr[200];
//
// Since there is no CopyKeyWithAllSubkeys API, it is difficult to write generic tree-walking algorithm.
// We will hard-code the keys here.
//
// copy Fax key always
ProfileCopySection(dwProfileIDTo,
lpszSectionTo,
dwProfileIDFr,
lpszSectionFr,
TRUE);
// copy Fax/Class1 key if exists
sprintf(SecTo, "%s\\Class1", lpszSectionTo);
sprintf(SecFr, "%s\\Class1", lpszSectionFr);
ProfileCopySection(dwProfileIDTo,
SecTo,
dwProfileIDFr,
SecFr,
FALSE);
// copy Fax/Class1/AdaptiveAnswer key if exists
sprintf(SecTo, "%s\\Class1\\AdaptiveAnswer", lpszSectionTo);
sprintf(SecFr, "%s\\Class1\\AdaptiveAnswer", lpszSectionFr);
ProfileCopySection(dwProfileIDTo,
SecTo,
dwProfileIDFr,
SecFr,
FALSE);
// copy Fax/Class1/AdaptiveAnswer/Answer key if exists
sprintf(SecTo, "%s\\Class1\\AdaptiveAnswer\\AnswerCommand", lpszSectionTo);
sprintf(SecFr, "%s\\Class1\\AdaptiveAnswer\\AnswerCommand", lpszSectionFr);
ProfileCopySection(dwProfileIDTo,
SecTo,
dwProfileIDFr,
SecFr,
FALSE);
// copy Fax/Class2 key if exists
sprintf(SecTo, "%s\\Class2", lpszSectionTo);
sprintf(SecFr, "%s\\Class2", lpszSectionFr);
ProfileCopySection(dwProfileIDTo,
SecTo,
dwProfileIDFr,
SecFr,
FALSE);
// copy Fax/Class2/AdaptiveAnswer key if exists
sprintf(SecTo, "%s\\Class2\\AdaptiveAnswer", lpszSectionTo);
sprintf(SecFr, "%s\\Class2\\AdaptiveAnswer", lpszSectionFr);
ProfileCopySection(dwProfileIDTo,
SecTo,
dwProfileIDFr,
SecFr,
FALSE);
// copy Fax/Class2/AdaptiveAnswer/Answer key if exists
sprintf(SecTo, "%s\\Class2\\AdaptiveAnswer\\AnswerCommand", lpszSectionTo);
sprintf(SecFr, "%s\\Class2\\AdaptiveAnswer\\AnswerCommand", lpszSectionFr);
ProfileCopySection(dwProfileIDTo,
SecTo,
dwProfileIDFr,
SecFr,
FALSE);
// copy Fax/Class2_0 key if exists
sprintf(SecTo, "%s\\Class2_0", lpszSectionTo);
sprintf(SecFr, "%s\\Class2_0", lpszSectionFr);
ProfileCopySection(dwProfileIDTo,
SecTo,
dwProfileIDFr,
SecFr,
FALSE);
// copy Fax/Class2_0/AdaptiveAnswer key if exists
sprintf(SecTo, "%s\\Class2_0\\AdaptiveAnswer", lpszSectionTo);
sprintf(SecFr, "%s\\Class2_0\\AdaptiveAnswer", lpszSectionFr);
ProfileCopySection(dwProfileIDTo,
SecTo,
dwProfileIDFr,
SecFr,
FALSE);
// copy Fax/Class2/AdaptiveAnswer/Answer key if exists
sprintf(SecTo, "%s\\Class2_0\\AdaptiveAnswer\\AnswerCommand", lpszSectionTo);
sprintf(SecFr, "%s\\Class2_0\\AdaptiveAnswer\\AnswerCommand", lpszSectionFr);
ProfileCopySection(dwProfileIDTo,
SecTo,
dwProfileIDFr,
SecFr,
FALSE);
return fRet;
}
BOOL
ProfileCopySection(
DWORD dwProfileIDTo,
LPSTR lpszSectionTo,
DWORD dwProfileIDFr,
LPSTR lpszSectionFr,
BOOL fCreateAlways
)
{
BOOL fRet=FALSE;
DWORD iValue=0;
DWORD cbValue, cbData, dwType;
char rgchValue[60], rgchData[256];
HKEY hkFr;
HKEY hkTo;
hkFr = (HKEY) ProfileOpen(dwProfileIDFr, lpszSectionFr, fREG_READ);
if ( (!hkFr) && (!fCreateAlways) ) {
return fRet;
}
hkTo = (HKEY) ProfileOpen(dwProfileIDTo, lpszSectionTo,
fREG_CREATE |fREG_READ|fREG_WRITE);
if (!hkTo || !hkFr) goto end;
iValue=0;dwType=0;cbValue=sizeof(rgchValue);cbData=sizeof(rgchData);
while(RegEnumValue(hkFr, iValue, rgchValue, &cbValue,
NULL, &dwType, rgchData, &cbData)==ERROR_SUCCESS)
{
if (RegQueryValueEx(hkFr, rgchValue, NULL, &dwType, rgchData, &cbData)
==ERROR_SUCCESS)
{
if (RegSetValueEx(hkTo, rgchValue, 0, dwType, rgchData, cbData)
== ERROR_SUCCESS)
{
fRet=TRUE;
}
}
iValue++;dwType=0;cbValue=sizeof(rgchValue);cbData=sizeof(rgchData);
}
end:
if (hkTo) RegCloseKey(hkTo);
if (hkFr) RegCloseKey(hkFr);
return fRet;
}
DWORD ProfileGetData(ULONG_PTR dwKey, LPSTR lpszValueName,
LPBYTE lpbBuf , DWORD dwcbMax)
{
#ifndef USE_REGISTRY
return 0;
#else // USE_REGISTRY
DWORD dwType;
LONG l = RegQueryValueEx(
(HKEY) dwKey,
lpszValueName,
0,
&dwType,
lpbBuf,
&dwcbMax
);
if (l!=ERROR_SUCCESS)
{
//LOG((_ERR, "RegQueryValueEx returned error %ld\n", (long) l));
goto copy_default;
}
if (dwType != REG_BINARY)
{
goto copy_default;
}
return dwcbMax;
copy_default:
return 0;
#endif // USE_REGISTRY
}
BOOL ProfileWriteData(ULONG_PTR dwKey, LPSTR lpszValueName,
LPBYTE lpbBuf, DWORD dwcb)
{
#ifndef USE_REGISTRY
return 0;
#else // USE_REGISTRY
LONG l = ~(ERROR_SUCCESS);
if (!lpszValueName) goto end;
if (!lpbBuf)
{
// delete value...
l = RegDeleteValue((HKEY) dwKey, lpszValueName);
}
else
{
l = RegSetValueEx((HKEY) dwKey, lpszValueName, 0, REG_BINARY,
lpbBuf, dwcb);
}
end:
return (l==ERROR_SUCCESS);
#endif // USE_REGISTRY
}
#ifdef USE_REGISTRY
DWORD my_atoul(LPSTR lpsz)
{
unsigned i=8, c;
unsigned long ul=0;
while(i-- && (c=*lpsz++)) {
ul*=10;
switch(c) {
case '0': break;
case '1':ul+=1; break;
case '2':ul+=2; break;
case '3':ul+=3; break;
case '4':ul+=4; break;
case '5':ul+=5; break;
case '6':ul+=6; break;
case '7':ul+=7; break;
case '8':ul+=8; break;
case '9':ul+=9; break;
default: goto end;
}
}
end:
return ul;
}
#endif // USE_REGISTRY