Leaked source code of windows server 2003
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.
 
 
 
 
 
 

843 lines
24 KiB

/*--------------------------------------------------------------------------
setupu.c - setup driver utilitities.
|--------------------------------------------------------------------------*/
#include "precomp.h"
char *szSlash = {"\\"};
char *szSYSTEM = {"SYSTEM"};
char *szCurrentControlSet = {"CurrentControlSet"};
char *szEnum = {"Enum"};
char *szServices = {"Services"};
char *szParameters = {"Parameters"};
char *szSecurity = {"Security"};
char *szLinkage = {"Linkage"};
char *szEventLog = {"EventLog"};
char *szSystem = {"System"};
// just turn debug for this module off
#define D_Level 0
/*----------------------------------------------------------------------
| setup_install_info - Get the version of Windows info, and the source path
of where we are running from
|----------------------------------------------------------------------*/
int APIENTRY setup_install_info(InstallPaths *ip,
HINSTANCE hinst,
char *NewServiceName, // "RocketPort" or "VSLink"
char *NewDriverName, // "rocket.sys" or "vslink.sys"
char *NewAppName, // "Comtrol RocketPort,RocketModem Install"
char *NewAppDir) // "rocket" or "vslink"
{
DWORD ver_info;
char *str;
int i;
struct w_ver_struct {
BYTE win_major;
BYTE win_minor;
BYTE dos_major;
BYTE dos_minor;
} *w_ver;
ver_info = GetVersion();
w_ver = (struct w_ver_struct *) &ver_info;
if (ver_info < 0x80000000)
{
ip->win_type = WIN_NT;
//wsprintf (szVersion, "Microsoft Windows NT %u.%u (Build: %u)",
//(DWORD)(LOBYTE(LOWORD(dwVersion))),
//(DWORD)(HIBYTE(LOWORD(dwVersion))),
// (DWORD)(HIWORD(dwVersion)) );
}
ip->major_ver = w_ver->win_major;
ip->minor_ver = w_ver->win_minor;
if (ip->win_type == WIN_UNKNOWN)
{
ip->win_type = WIN_NT; // force it
}
//GetWindowsDirectory(ip->win_dir,144);
//GetSystemDirectory(ip->system_dir,144);
ip->hinst = hinst;
// Initialize the default source path so that it uses the same
// drive that the SETUP.EXE application was executed from.
GetModuleFileName(hinst, ip->src_dir, sizeof(ip->src_dir));
// chop off file name leaving only the directory
str = ip->src_dir;
i = strlen(str);
if (i > 0) --i;
while ((str[i] != '\\') && (i != 0))
--i;
if (i==0)
str[0] = 0; // problem, no install dir
else
{
str[i] = 0; // terminate over "\"
}
strcpy(ip->szServiceName, NewServiceName);
strcpy(ip->szDriverName, NewDriverName);
strcpy(ip->szAppName, NewAppName);
strcpy(ip->szAppDir, NewAppDir);
GetSystemDirectory(ip->dest_dir, 144);
strcat(ip->dest_dir, "\\");
strcat(ip->dest_dir, NewAppDir);
return 0;
}
/*------------------------------------------------------------------------
| unattended_add_port_entries - Add port entries so that RAS will "see"
some ports which we can install to. Normally the driver puts these
entries in the reg on startup, the reg hardware area gets re-built
every startup. This is a kludge so that unattended install can
go on to add RAS ports.
|------------------------------------------------------------------------*/
int APIENTRY unattended_add_port_entries(InstallPaths *ip,
int num_entries,
int start_port)
{
int i;
static char *szSHDSt = {"HARDWARE\\DEVICEMAP\\SERIALCOMM"};
char szName[120];
char szCom[20];
char str[20];
//DWORD dwstat;
if (start_port == 0)
start_port = 5;
reg_create_key(NULL, szSHDSt); // "HARDWARE\\DEVICEMAP\\SERIALCOMM"
for (i=0; i<num_entries; i++)
{
wsprintf(szCom, "COM%d", i+start_port);
strncpy(szName, ip->szAppDir, strlen(ip->szAppDir) + 1); // "Rocket" or "VSLink"
wsprintf(str, "%d", i);
strncat(szName, str, strlen(str) + 1);
reg_set_str(NULL, szSHDSt, szName, szCom, REG_SZ);
}
return 0;
}
/*-----------------------------------------------------------------
remove_driver_reg_entries -
|------------------------------------------------------------------*/
int APIENTRY remove_driver_reg_entries(char *ServiceName)
{
int stat;
char str[180];
//our_message("Nuking RocketPort Reg Entry",MB_OK);
make_szSCS(str, ServiceName);
stat = reg_delete_key(NULL, str, szEnum);
stat = reg_delete_key(NULL, str, szParameters);
stat = reg_delete_key(NULL, str, szSecurity);
stat = reg_delete_key(NULL, str, szLinkage);
make_szSCS(str, NULL);
stat = reg_delete_key(NULL, str, ServiceName);
if (stat) {
DbgPrintf(D_Level, ("Error 4E\n"))
}
// get rid of the Services\EventLog\..RocketPort entry
make_szSCSES(str, NULL);
stat = reg_delete_key(NULL, str, ServiceName);
if (stat) {
DbgPrintf(D_Level, ("Error 4F\n"))
}
#ifdef NT50
#if 0
remove_pnp_reg_entries();
#endif
#endif
return 0;
}
#if 0
this is experimental
/*--------------------------------------------------------------------------
| remove_pnp_reg_entries -
|--------------------------------------------------------------------------*/
int APIENTRY remove_pnp_reg_entries(void)
{
DWORD stat, keystat;
static char *enum_pci = {"SYSTEM\\CurrentControlSet\\Enum\\PCI"};
static char *enum_root = {"SYSTEM\\CurrentControlSet\\Enum\\root"};
//static char root_name[124];
char node_name[140];
char *pc;
DWORD node_i;
HKEY hKey;
DbgPrintf(D_Level, ("Looking\n"))
keystat = RegOpenKeyEx (HKEY_LOCAL_MACHINE,
enum_pci,
0,
KEY_ENUMERATE_SUB_KEYS | KEY_EXECUTE | KEY_QUERY_VALUE,
&hKey);
if (keystat != ERROR_SUCCESS)
{
DbgPrintf(D_Level, ("Err14a\n"))
return 1;
}
node_i = 0;
stat = RegEnumKey (hKey, node_i++, node_name, 138);
while (stat == ERROR_SUCCESS)
{
pc = node_name;
while (*pc)
{
*pc = toupper(*pc);
++pc;
}
if (strstr(node_name, "VEN_11FE") != 0)
{
// found a Comtrol hardware node
DbgPrintf(D_Level, ("Found Node:%s\n", node_name))
stat = RegDeleteKeyNT(hKey, node_name);
if (stat != ERROR_SUCCESS)
{
DbgPrintf(D_Level, ("No Delete\n"))
}
//stat = reg_delete_key(NULL, node_name, node_name);
}
stat = RegEnumKey (hKey, node_i++, node_name, 68);
}
RegCloseKey (hKey); // Close the key handle.
return 0;
}
#endif
/*--------------------------------------------------------------------------
RegDeleteKeyNT -
A registry key that is opened by an application can be deleted
without error by another application in both Windows 95 and
Windows NT. This is by design.
|--------------------------------------------------------------------------*/
DWORD APIENTRY RegDeleteKeyNT(HKEY hStartKey , LPTSTR pKeyName )
{
DWORD dwRtn, dwSubKeyLength;
LPTSTR pSubKey = NULL;
TCHAR szSubKey[256]; // (256) this should be dynamic.
HKEY hKey;
// do not allow NULL or empty key name
if ( pKeyName && lstrlen(pKeyName))
{
if( (dwRtn=RegOpenKeyEx(hStartKey,pKeyName,
0, KEY_ENUMERATE_SUB_KEYS | DELETE, &hKey )) == ERROR_SUCCESS)
{
while (dwRtn == ERROR_SUCCESS )
{
dwSubKeyLength = 250;
dwRtn=RegEnumKeyEx(
hKey,
0, // always index zero
szSubKey,
&dwSubKeyLength,
NULL,
NULL,
NULL,
NULL
);
if(dwRtn == ERROR_NO_MORE_ITEMS)
{
dwRtn = RegDeleteKey(hStartKey, pKeyName);
break;
}
else if(dwRtn == ERROR_SUCCESS)
dwRtn=RegDeleteKeyNT(hKey, szSubKey);
}
RegCloseKey(hKey);
// Do not save return code because error
// has already occurred
}
else
{
DbgPrintf(D_Level, ("Access Error\n"))
}
}
else
dwRtn = ERROR_BADKEY;
return dwRtn;
}
/*---------------------------------------------------------------------------
| modem_inf_change - go make the needed changes to the modem.inf file.
| Make a backup copy too.
|----------------------------------------------------------------------------*/
int APIENTRY modem_inf_change(InstallPaths *ip,
char *modemfile,
char *szModemInfEntry)
{
int i=0;
int stat = 1;
OUR_FILE *fp; // in
OUR_FILE *fp2; // out
OUR_FILE *fp_new; // entries file to add
char buf[202];
char *str;
char *szRAS={"\\RAS\\"};
int section = 0;
int chk;
DbgPrintf(D_Level, ("chg inf start\n"))
// first backup original modem.inf if not done already.
stat = backup_modem_inf(ip);
if (stat)
return 1; // err, couldn't backup modem.inf
DbgPrintf(D_Level, ("chg A\n"))
GetSystemDirectory(buf, 144);
strcat(buf,"\\");
strcat(buf,modemfile);
fp_new = our_fopen(buf, "rb"); // rocket\rocket35.inf
DbgPrintf(D_Level, ("chg B\n"))
if (fp_new == NULL)
{
wsprintf(ip->tmpstr,RcStr((MSGSTR+15)),buf);
stat = our_message(ip, ip->tmpstr, MB_OK);
return 1;
}
GetSystemDirectory(ip->dest_str, 144);
strcat(ip->dest_str, szRAS);
strcat(ip->dest_str,"modem.inf");
//----- now copy modem.inf to modem.rk0 as a base to read from and
// write new file
GetSystemDirectory(ip->src_str, 144);
strcat(ip->src_str, szRAS);
strcat(ip->src_str,"modem.rk0");
stat = our_copy_file(ip->src_str, ip->dest_str);
if (stat)
{
wsprintf(ip->tmpstr,RcStr((MSGSTR+16)), ip->dest_str, ip->src_str, stat);
stat = our_message(ip, ip->tmpstr, MB_OK);
return 2; // err
}
fp = our_fopen(ip->src_str, "rb"); // modem.rk0
if (fp == NULL)
{
wsprintf(ip->tmpstr,RcStr((MSGSTR+17)), ip->src_str);
stat = our_message(ip, ip->tmpstr, MB_OK);
return 1;
}
fp2 = our_fopen(ip->dest_str, "wb"); // modem.inf
if (fp2 == NULL)
{
wsprintf(ip->tmpstr, "Tried to open the %s file for changes, but could not open it.", ip->dest_str);
our_fclose(fp);
return 1;
}
chk = 0;
while ((our_fgets(buf, 200, fp)) && (!our_feof(fp)) && (!our_ferror(fp)) && (!our_ferror(fp2)) &&
(chk < 30000))
{
++chk;
// search and kill any 0x1a eof markers
str = buf;
while (*str != 0)
{
if (*str == 0x1a)
*str = ' '; // change eof to space
++str;
}
// pass up spaces
str = buf;
while (*str == ' ')
++str;
if (*str == '[')
{
if (str[0] == '[') // starting new section
{
section = 0; // not our section to worry about
if (my_substr_lstricmp(str, szModemInfEntry) == 0) // match
{
// make sure
section = 1;
}
} // end of new [] section header
}
// process all entries here
if (section == 1)
str[0] = 0; // first delete all "[Comtrol RocketModem]" entries
if (str[0] != 0)
{
str = buf; // don't skip spaces
our_fputs(buf,fp2);
}
} // end of while fgets();
stat = 0;
if ( (our_ferror(fp)) || (our_ferror(fp2)) || (chk >= 10000))
{
stat = 3; // error
}
if (stat)
{
our_fclose(fp);
our_fclose(fp2);
our_fclose(fp_new);
// try to restore to the read file backup(modem.rk0)
stat = our_copy_file(ip->dest_str, ip->src_str);
wsprintf(ip->tmpstr, "Errors encountered while making %s file changes.", ip->dest_str);
stat = 3; // error
}
else
{
// append the changes to it
our_fputs("\x0d\x0a",fp2);
our_fputs(szModemInfEntry, fp2);
our_fputs("\x0d\x0a;-------------------\x0d\x0a",fp2);
while ((our_fgets(buf, 200, fp_new)) && (!our_feof(fp_new)) && (!our_ferror(fp2)))
{
our_fputs(buf,fp2);
}
our_fclose(fp);
our_fclose(fp2);
our_fclose(fp_new);
our_remove(ip->src_str); // kill temporary modem.rk0 file
}
return stat;
}
/*---------------------------------------------------------------------------
| backup_modem_inf - backs up to modem.bak.
|----------------------------------------------------------------------------*/
int APIENTRY backup_modem_inf(InstallPaths *ip)
{
int stat = 1;
char *szRAS = {"\\RAS\\"};
OUR_FILE *fp;
// first copy modem.inf over to our directory as a backup
GetSystemDirectory(ip->dest_str, 144);
strcat(ip->dest_str, szRAS);
strcat(ip->dest_str,"modem.bak");
GetSystemDirectory(ip->src_str, 144);
strcat(ip->src_str, szRAS);
strcat(ip->src_str,"modem.inf");
fp = our_fopen(ip->dest_str, "rb"); // see if already backed up
if (fp == NULL)
{
stat = our_copy_file(ip->dest_str, ip->src_str);
if (stat != 0)
{
wsprintf(ip->tmpstr,RcStr((MSGSTR+18)), ip->src_str, stat);
our_message(ip, ip->tmpstr, MB_OK);
return 1; // err
}
}
else our_fclose(fp);
return 0; // ok
}
/*---------------------------------------------------------------------------
| service_man - Handle installation, removal, starting, stopping of NT
services(or drivers.).
|----------------------------------------------------------------------------*/
int APIENTRY service_man(LPSTR lpServiceName, LPSTR lpBinaryPath, int chore)
{
SC_HANDLE hSCManager = NULL;
SC_HANDLE hService = NULL;
SC_LOCK lSCLock = NULL;
// SERVICE_STATUS ServiceStatus;
BOOL Status = 1; // success
int stat = 0; // ret status(0=ok)
hSCManager = OpenSCManager( NULL, NULL, SC_MANAGER_ALL_ACCESS );
if (hSCManager == NULL)
{
return 2;
}
//---- Lock the service database
lSCLock = LockServiceDatabase( hSCManager );
if (lSCLock == NULL)
{
CloseServiceHandle( hSCManager );
return 1;
}
if ((chore != CHORE_INSTALL) && (chore != CHORE_INSTALL_SERVICE))
{
hService = OpenService( hSCManager, lpServiceName, SERVICE_ALL_ACCESS);
if (hService == NULL)
{
UnlockServiceDatabase( lSCLock );
CloseServiceHandle( hSCManager );
return 2;
}
}
switch(chore)
{
case CHORE_INSTALL:
// Create the service
hService = CreateService( hSCManager,
lpServiceName, // Service's name
lpServiceName, // Display name (new for NT)
SERVICE_ALL_ACCESS,// Access (allow all)
SERVICE_KERNEL_DRIVER, // Service type
SERVICE_AUTO_START, // Startup behavior
0x1, // Error control
lpBinaryPath, // Full pathname of binary
NULL, // Load order group
NULL, // Tag ID
NULL, // Dependencies (none)
NULL, // Account name
NULL // Password
);
if (hService == NULL)
stat = 5;
Status = 0;
break;
case CHORE_START:
// Unlock the database
if (lSCLock != NULL)
{
UnlockServiceDatabase( lSCLock );
lSCLock = NULL;
}
Status = StartService(hService, 0, NULL);
//if (Status !=
if (Status == 0) // false
{
stat = GetLastError();
Status = 1;
}
break;
case CHORE_STOP:
{
SERVICE_STATUS ss;
Status = ControlService(hService, SERVICE_CONTROL_STOP, &ss);
if (Status == 0) // false
{
stat = GetLastError();
if (stat == 0)
stat = 1234;
Status = 1;
}
}
break;
case CHORE_REMOVE:
Status = DeleteService(hService);
break;
case CHORE_INSTALL_SERVICE:
// Create the service
hService = CreateService( hSCManager,
lpServiceName, // Service's name
lpServiceName, // Display name (new for NT)
SERVICE_ALL_ACCESS,// Access (allow all)
SERVICE_WIN32_OWN_PROCESS, // Service type
SERVICE_AUTO_START, // Startup behavior
SERVICE_ERROR_NORMAL, // Error control
lpBinaryPath, // Full pathname of binary
NULL, // Load order group
NULL, // Tag ID
NULL, // Dependencies (none)
NULL, // Account name
NULL // Password
);
if (hService == NULL)
stat = 6;
//Status = 0;
break;
case CHORE_IS_INSTALLED:
// return without error to indicate it is installed.
break;
}
if (Status == 0) // false
stat = 8;
// Close our handle to the new service
if (hService != NULL)
CloseServiceHandle(hService);
// Unlock the database
if (lSCLock != NULL)
UnlockServiceDatabase( lSCLock );
// Free our handle to the service control manager
CloseServiceHandle( hSCManager );
return stat;
}
/*-----------------------------------------------------------------
make_szSCS - Services area.
form ascii string: "SYSTEM\CurrentControlSet\Services"
|------------------------------------------------------------------*/
int APIENTRY make_szSCS(char *str, const char *szName)
{
strcpy(str, szSYSTEM); strcat(str, szSlash);
strcat(str, szCurrentControlSet); strcat(str, szSlash);
strcat(str, szServices);
if (szName != NULL)
{
strcat(str, szSlash);
strcat(str, szName);
}
return 0;
}
/*-----------------------------------------------------------------
make_szSCSES - Event log reg area
form ascii string: "SYSTEM\CurrentControlSet\Services\EventLog\System"
|------------------------------------------------------------------*/
int APIENTRY make_szSCSES(char *str, const char *szName)
{
strcpy(str, szSYSTEM); strcat(str, szSlash);
strcat(str, szCurrentControlSet); strcat(str, szSlash);
strcat(str, szServices); strcat(str, szSlash);
strcat(str, szEventLog); strcat(str, szSlash);
strcat(str, szSystem);
if (szName != NULL)
{
strcat(str, szSlash);
strcat(str, szName);
}
return 0;
}
/*---------------------------------------------------------------------------
| copy_files - copy a list of files from wi->src_dir to wi->dest_dir
| Uses the wi->src_str & wi->dest_str. Assumes you want the same name
| as copying from.
|----------------------------------------------------------------------------*/
int APIENTRY copy_files(InstallPaths *ip, char **files)
{
int i=0;
int stat;
if (my_lstricmp(ip->src_dir, ip->dest_dir) == 0) // src_dir == dest_drv
return 0;
while (files[i] != NULL)
{
strcpy(ip->src_str, ip->src_dir);
strcat(ip->src_str, szSlash);
strcat(ip->src_str, files[i]);
strcpy(ip->dest_str, ip->dest_dir);
strcat(ip->dest_str, szSlash);
strcat(ip->dest_str, files[i]);
again1:
stat = our_copy_file(ip->dest_str, ip->src_str);
if (stat)
{
//if (stat == 1) // error opening read file
// return 1; // don't report errors, since some driver sets differ in
// which files they include
// (that was a stupid idea(karl to karl)!)
wsprintf(ip->tmpstr,RcStr((MSGSTR+19)), ip->src_str, ip->dest_str);
stat = our_message(ip, ip->tmpstr, MB_ABORTRETRYIGNORE);
if (stat == IDABORT)
return 1; // error
if (stat == IDRETRY) goto again1;
}
++i;
}
return 0;
}
/*---------------------------------------------------------------------------
| our_copy_file - copy a file from here to there.
|----------------------------------------------------------------------------*/
int APIENTRY our_copy_file(char *dest, char *src)
{
int stat;
// just use the stock Win32 function
stat = CopyFile(src, dest,
0); // 1=fail if exist
if (stat)
return 0; // ok, worked
return 1; // failed
#ifdef COMMENT_OUT
char *buf;
unsigned int bytes, wbytes;
int err = 0;
int chk = 0;
OUR_FILE *fp1, *fp2;
buf = (char *) malloc(0x4010);
if (buf == NULL)
{
//our_message("Error, no memory",MB_OK);
return 6; // no mem
}
fp1 = our_fopen(src,"rb");
if (fp1 == NULL)
{
//our_message("Error Opening to read",MB_OK);
free(buf);
return 1; // no src
}
fp2 = our_fopen(dest,"wb");
if (fp2 == NULL)
{
//our_message("Error Opening to write",MB_OK);
free(buf);
our_fclose(fp1);
return 2; /* err opening dest */
}
bytes = our_fread(buf, 1, 0x4000, fp1);
while ((bytes > 0) && (!err))
{
++chk;
if (chk > 10000)
err = 5;
wbytes = our_fwrite(buf, 1, bytes, fp2);
if (wbytes != bytes)
{
err = 3;
}
bytes = our_fread(buf, 1, 0x4000, fp1);
if (our_ferror(fp1))
{
//our_message("Error reading",MB_OK);
err = 4;
}
if (our_ferror(fp2))
{
//our_message("Error writing",MB_OK);
err = 6;
}
}
free(buf);
our_fclose(fp1);
our_fclose(fp2);
return err; // 0=ok, else error
#endif
}
/*-----------------------------------------------------------------------------
| our_message -
|-----------------------------------------------------------------------------*/
int APIENTRY our_message(InstallPaths *ip, char *str, WORD option)
{
if (ip->prompting_off)
{
// we are doing unattended install, and don't want user interface
// popping up. So just say YES or OK to all prompts.
if (option == MB_YESNO)
return IDYES;
return IDOK;
}
return MessageBox(GetFocus(), str, ip->szAppName, option);
}
/*-----------------------------------------------------------------------------
| load_str -
|-----------------------------------------------------------------------------*/
int APIENTRY load_str(HINSTANCE hinst, int id, char *dest, int str_size)
{
dest[0] = 0;
if (!LoadString(hinst, id, dest, str_size) )
{
wsprintf(dest,"Err,str-id %d", id);
MessageBox(GetFocus(), dest, "Error", MB_OK);
return 1;
}
return 0;
}
#if 0
/*-----------------------------------------------------------------------------
| our_id_message -
|-----------------------------------------------------------------------------*/
int APIENTRY our_id_message(InstallPaths *ip, int id, WORD prompt)
{
int stat;
if (ip->prompting_off)
{
// we are doing unattended install, and don't want user interface
// popping up. So just say YES or OK to all prompts.
if (option == MB_YESNO)
return IDYES;
return IDOK;
}
load_str(ip->hinst, id, ip->tmpstr, CharSizeOf(ip->tmpstr));
stat = our_message(ip, ip->tmpstr, prompt);
return stat;
}
#endif
/*---------------------------------------------------------------------------
| mess - message
|---------------------------------------------------------------------------*/
void APIENTRY mess(InstallPaths *ip, char *format, ...)
{
va_list next;
char buf[200];
if (ip->prompting_off)
return;
va_start(next, format);
vsprintf(buf, format, next);
MessageBox(GetFocus(), buf, ip->szAppName, MB_OK);
}
TCHAR *
RcStr( int MsgIndex )
{
static TCHAR RcStrBuf[200];
load_str(glob_hinst, MsgIndex, RcStrBuf, CharSizeOf(RcStrBuf) );
if (!LoadString(glob_hinst, MsgIndex, RcStrBuf, CharSizeOf(RcStrBuf)) ) {
wsprintf(RcStrBuf,"Err, str-id %d", MsgIndex);
}
return RcStrBuf;
}