461 lines
16 KiB
C
461 lines
16 KiB
C
// loaddrv.c - Dynamic driver install/start/stop/remove
|
|
// based on Paula Tomlinson's LOADDRV program.
|
|
// She describes it in her May 1995 article in Windows/DOS Developer's
|
|
// Journal (now Windows Developer's Journal).
|
|
// Modified by Chris Liechti <cliechti@gmx.net>
|
|
// I removed the old/ugly dialog, it now accepts command line options and
|
|
// prints error messages with textual description from the OS.
|
|
|
|
#include <windows.h>
|
|
#include <stdio.h>
|
|
#include <stdlib.h>
|
|
#include <string.h>
|
|
#include <unistd.h>
|
|
#include "loaddrv.h"
|
|
|
|
// globals
|
|
SC_HANDLE hSCMan = NULL;
|
|
|
|
//get ext messages for windows error codes:
|
|
void DisplayErrorText(DWORD dwLastError) {
|
|
LPSTR MessageBuffer;
|
|
DWORD dwBufferLength;
|
|
|
|
DWORD dwFormatFlags = FORMAT_MESSAGE_ALLOCATE_BUFFER |
|
|
FORMAT_MESSAGE_IGNORE_INSERTS |
|
|
FORMAT_MESSAGE_FROM_SYSTEM;
|
|
|
|
dwBufferLength = FormatMessageA(
|
|
dwFormatFlags,
|
|
NULL, // module to get message from (NULL == system)
|
|
dwLastError,
|
|
MAKELANGID(LANG_NEUTRAL, SUBLANG_DEFAULT), // default language
|
|
(LPSTR) &MessageBuffer,
|
|
0,
|
|
NULL
|
|
);
|
|
if (dwBufferLength) {
|
|
// Output message
|
|
puts(MessageBuffer);
|
|
// Free the buffer allocated by the system.
|
|
LocalFree(MessageBuffer);
|
|
}
|
|
}
|
|
|
|
int exists(char *filename) {
|
|
FILE * pFile;
|
|
pFile = fopen(filename, "r");
|
|
return pFile != NULL;
|
|
}
|
|
|
|
void usage(void) {
|
|
printf("USGAE: loaddrv command drivername [args...]\n\n"
|
|
"NT/2k/XP Driver and Service modification tool.\n"
|
|
"(C)2002 Chris Liechti <cliechti@gmx.net>\n\n"
|
|
"Suported commands:\n\n"
|
|
" install [fullpathforinstall]\n"
|
|
" Install new service. Loaded from given path. If path is not present,\n"
|
|
" the local directory is searched for a .sys file. If the service\n"
|
|
" already exists, it must be removed first.\n"
|
|
" start\n"
|
|
" Start service. It must be installed in advance.\n"
|
|
" stop\n"
|
|
" Stop service.\n"
|
|
" remove\n"
|
|
" Remove service. It must be stopped in advance.\n"
|
|
" status\n"
|
|
" Show status information about service.\n"
|
|
" starttype auto|manual|system|disable\n"
|
|
" Change startup type to the given type.\n"
|
|
);
|
|
}
|
|
|
|
int main(int argc, char *argv[]) {
|
|
DWORD status = 0;
|
|
int level = 0;
|
|
if (argc < 3) {
|
|
usage();
|
|
exit(1);
|
|
}
|
|
LoadDriverInit();
|
|
if (strcmp(argv[1], "start") == 0) {
|
|
printf("starting %s... ", argv[2]);
|
|
status = DriverStart(argv[2]);
|
|
if ( status != OKAY) {
|
|
printf("start failed (status %ld):\n", status);
|
|
level = 1;
|
|
} else {
|
|
printf("ok.\n");
|
|
}
|
|
} else if (strcmp(argv[1], "stop") == 0) {
|
|
printf("stoping %s... ", argv[2]);
|
|
status = DriverStop(argv[2]);
|
|
if ( status != OKAY) {
|
|
printf("stop failed (status %ld):\n", status);
|
|
level = 1;
|
|
} else {
|
|
printf("ok.\n");
|
|
}
|
|
} else if (strcmp(argv[1], "install") == 0) {
|
|
char path[MAX_PATH*2];
|
|
if (argc<4) {
|
|
char cwd[MAX_PATH];
|
|
getcwd(cwd, sizeof cwd);
|
|
sprintf(path, "%s\\%s.sys", cwd, argv[2]);
|
|
} else {
|
|
strncpy(path, argv[3], MAX_PATH);
|
|
}
|
|
if (exists(path)) {
|
|
printf("installing %s from %s... ", argv[2], path);
|
|
status = DriverInstall(path, argv[2]);
|
|
if ( status != OKAY) {
|
|
printf("install failed (status %ld):\n", status);
|
|
level = 2;
|
|
} else {
|
|
printf("ok.\n");
|
|
}
|
|
} else {
|
|
printf("install failed, file not found: %s\n", path);
|
|
level = 1;
|
|
}
|
|
} else if (strcmp(argv[1], "remove") == 0) {
|
|
printf("removing %s... ", argv[2]);
|
|
status = DriverRemove(argv[2]);
|
|
if ( status != OKAY) {
|
|
printf("remove failed (status %ld):\n", status);
|
|
level = 1;
|
|
} else {
|
|
printf("ok.\n");
|
|
}
|
|
} else if (strcmp(argv[1], "status") == 0) {
|
|
printf("status of %s:\n", argv[2]);
|
|
status = DriverStatus(argv[2]);
|
|
if ( status != OKAY) {
|
|
printf("stat failed (status %ld):\n", status);
|
|
level = 1;
|
|
} else {
|
|
printf("ok.\n");
|
|
}
|
|
} else if (strcmp(argv[1], "starttype") == 0) {
|
|
if (argc < 4) {
|
|
printf("Error: need start type (string) as argument.\n");
|
|
level = 2;
|
|
} else {
|
|
DWORD type = 0;
|
|
printf("set start type of %s to %s... ", argv[2], argv[3]);
|
|
if (strcmp(argv[1], "boot") == 0) {
|
|
type = SERVICE_BOOT_START;
|
|
} else if (strcmp(argv[3], "system") == 0) {
|
|
type = SERVICE_SYSTEM_START;
|
|
} else if (strcmp(argv[3], "auto") == 0) {
|
|
type = SERVICE_AUTO_START;
|
|
} else if (strcmp(argv[3], "manual") == 0) {
|
|
type = SERVICE_DEMAND_START;
|
|
} else if (strcmp(argv[3], "disabled") == 0) {
|
|
type = SERVICE_DISABLED;
|
|
} else {
|
|
printf("unknown type\n");
|
|
level = 1;
|
|
}
|
|
if (level == 0) {
|
|
status = DriverStartType(argv[2], type);
|
|
if ( status != OKAY) {
|
|
printf("set start type failed (status %ld):\n", status);
|
|
level = 1;
|
|
} else {
|
|
printf("ok.\n");
|
|
}
|
|
}
|
|
}
|
|
} else {
|
|
usage();
|
|
level = 1;
|
|
}
|
|
if (status) DisplayErrorText(status);
|
|
LoadDriverCleanup();
|
|
exit(level);
|
|
return 0;
|
|
}
|
|
|
|
|
|
DWORD LoadDriverInit(void) {
|
|
// connect to local service control manager
|
|
if ((hSCMan = OpenSCManager(NULL, NULL,
|
|
SC_MANAGER_ALL_ACCESS)) == NULL) {
|
|
return -1;
|
|
}
|
|
return OKAY;
|
|
}
|
|
|
|
void LoadDriverCleanup(void) {
|
|
if (hSCMan != NULL) CloseServiceHandle(hSCMan);
|
|
}
|
|
|
|
/**-----------------------------------------------------**/
|
|
DWORD DriverInstall(LPSTR lpPath, LPSTR lpDriver) {
|
|
BOOL dwStatus = OKAY;
|
|
SC_HANDLE hService = NULL;
|
|
|
|
// add to service control manager's database
|
|
if ((hService = CreateService(hSCMan, lpDriver,
|
|
lpDriver, SERVICE_ALL_ACCESS, SERVICE_KERNEL_DRIVER,
|
|
SERVICE_DEMAND_START, SERVICE_ERROR_NORMAL, lpPath,
|
|
NULL, NULL, NULL, NULL, NULL)) == NULL)
|
|
dwStatus = GetLastError();
|
|
else CloseServiceHandle(hService);
|
|
|
|
return dwStatus;
|
|
} // DriverInstall
|
|
|
|
/**-----------------------------------------------------**/
|
|
DWORD DriverStart(LPSTR lpDriver) {
|
|
BOOL dwStatus = OKAY;
|
|
SC_HANDLE hService = NULL;
|
|
|
|
// get a handle to the service
|
|
if ((hService = OpenService(hSCMan, lpDriver,
|
|
SERVICE_ALL_ACCESS)) != NULL)
|
|
{
|
|
// start the driver
|
|
if (!StartService(hService, 0, NULL))
|
|
dwStatus = GetLastError();
|
|
} else dwStatus = GetLastError();
|
|
|
|
if (hService != NULL) CloseServiceHandle(hService);
|
|
return dwStatus;
|
|
} // DriverStart
|
|
|
|
/**-----------------------------------------------------**/
|
|
DWORD DriverStop(LPSTR lpDriver)
|
|
{
|
|
BOOL dwStatus = OKAY;
|
|
SC_HANDLE hService = NULL;
|
|
SERVICE_STATUS serviceStatus;
|
|
|
|
// get a handle to the service
|
|
if ((hService = OpenService(hSCMan, lpDriver,
|
|
SERVICE_ALL_ACCESS)) != NULL)
|
|
{
|
|
// stop the driver
|
|
if (!ControlService(hService, SERVICE_CONTROL_STOP,
|
|
&serviceStatus))
|
|
dwStatus = GetLastError();
|
|
} else dwStatus = GetLastError();
|
|
|
|
if (hService != NULL) CloseServiceHandle(hService);
|
|
return dwStatus;
|
|
} // DriverStop
|
|
|
|
/**-----------------------------------------------------**/
|
|
DWORD DriverRemove(LPSTR lpDriver)
|
|
{
|
|
BOOL dwStatus = OKAY;
|
|
SC_HANDLE hService = NULL;
|
|
|
|
// get a handle to the service
|
|
if ((hService = OpenService(hSCMan, lpDriver,
|
|
SERVICE_ALL_ACCESS)) != NULL)
|
|
{ // remove the driver
|
|
if (!DeleteService(hService))
|
|
dwStatus = GetLastError();
|
|
} else dwStatus = GetLastError();
|
|
|
|
if (hService != NULL) CloseServiceHandle(hService);
|
|
return dwStatus;
|
|
} // DriverRemove
|
|
|
|
/**-----------------------------------------------------**/
|
|
////extensions by Lch
|
|
/**-----------------------------------------------------**/
|
|
DWORD DriverStatus(LPSTR lpDriver) {
|
|
BOOL dwStatus = OKAY;
|
|
SC_HANDLE hService = NULL;
|
|
DWORD dwBytesNeeded;
|
|
|
|
// get a handle to the service
|
|
if ((hService = OpenService(hSCMan, lpDriver,
|
|
SERVICE_ALL_ACCESS)) != NULL)
|
|
{
|
|
LPQUERY_SERVICE_CONFIG lpqscBuf;
|
|
//~ LPSERVICE_DESCRIPTION lpqscBuf2;
|
|
// Allocate a buffer for the configuration information.
|
|
if ((lpqscBuf = (LPQUERY_SERVICE_CONFIG) LocalAlloc(
|
|
LPTR, 4096)) != NULL)
|
|
{
|
|
//~ if ((lpqscBuf2 = (LPSERVICE_DESCRIPTION) LocalAlloc(
|
|
//~ LPTR, 4096)) != NULL)
|
|
{
|
|
// Get the configuration information.
|
|
if (QueryServiceConfig(
|
|
hService,
|
|
lpqscBuf,
|
|
4096,
|
|
&dwBytesNeeded) //&&
|
|
//~ QueryServiceConfig2(
|
|
//~ hService,
|
|
//~ SERVICE_CONFIG_DESCRIPTION,
|
|
//~ lpqscBuf2,
|
|
//~ 4096,
|
|
//~ &dwBytesNeeded
|
|
)
|
|
{
|
|
// Print the configuration information.
|
|
printf("Type: [0x%02lx] ", lpqscBuf->dwServiceType);
|
|
switch (lpqscBuf->dwServiceType) {
|
|
case SERVICE_WIN32_OWN_PROCESS:
|
|
printf("The service runs in its own process.");
|
|
break;
|
|
case SERVICE_WIN32_SHARE_PROCESS:
|
|
printf("The service shares a process with other services.");
|
|
break;
|
|
case SERVICE_KERNEL_DRIVER:
|
|
printf("Kernel driver.");
|
|
break;
|
|
case SERVICE_FILE_SYSTEM_DRIVER:
|
|
printf("File system driver.");
|
|
break;
|
|
case SERVICE_INTERACTIVE_PROCESS:
|
|
printf("The service can interact with the desktop.");
|
|
break;
|
|
default:
|
|
printf("Unknown type.");
|
|
}
|
|
printf("\nStart Type: [0x%02lx] ", lpqscBuf->dwStartType);
|
|
switch (lpqscBuf->dwStartType) {
|
|
case SERVICE_BOOT_START:
|
|
printf("Boot");
|
|
break;
|
|
case SERVICE_SYSTEM_START:
|
|
printf("System");
|
|
break;
|
|
case SERVICE_AUTO_START:
|
|
printf("Automatic");
|
|
break;
|
|
case SERVICE_DEMAND_START:
|
|
printf("Manual");
|
|
break;
|
|
case SERVICE_DISABLED:
|
|
printf("Disabled");
|
|
break;
|
|
default:
|
|
printf("Unknown.");
|
|
}
|
|
printf("\nError Control: [0x%02lx] ", lpqscBuf->dwErrorControl);
|
|
switch (lpqscBuf->dwErrorControl) {
|
|
case SERVICE_ERROR_IGNORE:
|
|
printf("IGNORE: Ignore.");
|
|
break;
|
|
case SERVICE_ERROR_NORMAL:
|
|
printf("NORMAL: Display a message box.");
|
|
break;
|
|
case SERVICE_ERROR_SEVERE:
|
|
printf("SEVERE: Restart with last-known-good config.");
|
|
break;
|
|
case SERVICE_ERROR_CRITICAL:
|
|
printf("CRITICAL: Restart w/ last-known-good config.");
|
|
break;
|
|
default:
|
|
printf("Unknown.");
|
|
}
|
|
printf("\nBinary path: %s\n", lpqscBuf->lpBinaryPathName);
|
|
|
|
if (lpqscBuf->lpLoadOrderGroup != NULL)
|
|
printf("Load order grp: %s\n", lpqscBuf->lpLoadOrderGroup);
|
|
if (lpqscBuf->dwTagId != 0)
|
|
printf("Tag ID: %ld\n", lpqscBuf->dwTagId);
|
|
if (lpqscBuf->lpDependencies != NULL)
|
|
printf("Dependencies: %s\n", lpqscBuf->lpDependencies);
|
|
if (lpqscBuf->lpServiceStartName != NULL)
|
|
printf("Start Name: %s\n", lpqscBuf->lpServiceStartName);
|
|
//~ if (lpqscBuf2->lpDescription != NULL)
|
|
//~ printf("Description: %s\n", lpqscBuf2->lpDescription);
|
|
}
|
|
//~ LocalFree(lpqscBuf2);
|
|
}
|
|
LocalFree(lpqscBuf);
|
|
} else {
|
|
dwStatus = GetLastError();
|
|
}
|
|
} else {
|
|
dwStatus = GetLastError();
|
|
}
|
|
|
|
if (hService != NULL) CloseServiceHandle(hService);
|
|
return dwStatus;
|
|
} // DriverStatus
|
|
|
|
/**-----------------------------------------------------**/
|
|
DWORD DriverStartType(LPSTR lpDriver, DWORD dwStartType) {
|
|
BOOL dwStatus = OKAY;
|
|
SC_HANDLE hService = NULL;
|
|
|
|
SC_LOCK sclLock;
|
|
LPQUERY_SERVICE_LOCK_STATUS lpqslsBuf;
|
|
DWORD dwBytesNeeded;
|
|
|
|
// Need to acquire database lock before reconfiguring.
|
|
sclLock = LockServiceDatabase(hSCMan);
|
|
|
|
// If the database cannot be locked, report the details.
|
|
if (sclLock == NULL) {
|
|
// Exit if the database is not locked by another process.
|
|
if (GetLastError() == ERROR_SERVICE_DATABASE_LOCKED) {
|
|
|
|
// Allocate a buffer to get details about the lock.
|
|
lpqslsBuf = (LPQUERY_SERVICE_LOCK_STATUS) LocalAlloc(
|
|
LPTR, sizeof(QUERY_SERVICE_LOCK_STATUS)+256);
|
|
if (lpqslsBuf != NULL) {
|
|
// Get and print the lock status information.
|
|
if (QueryServiceLockStatus(
|
|
hSCMan,
|
|
lpqslsBuf,
|
|
sizeof(QUERY_SERVICE_LOCK_STATUS)+256,
|
|
&dwBytesNeeded) )
|
|
{
|
|
if (lpqslsBuf->fIsLocked) {
|
|
printf("Locked by: %s, duration: %ld seconds\n",
|
|
lpqslsBuf->lpLockOwner,
|
|
lpqslsBuf->dwLockDuration
|
|
);
|
|
} else {
|
|
printf("No longer locked\n");
|
|
}
|
|
}
|
|
LocalFree(lpqslsBuf);
|
|
}
|
|
}
|
|
dwStatus = GetLastError();
|
|
} else {
|
|
// The database is locked, so it is safe to make changes.
|
|
// Open a handle to the service.
|
|
hService = OpenService(
|
|
hSCMan, // SCManager database
|
|
lpDriver, // name of service
|
|
SERVICE_CHANGE_CONFIG
|
|
); // need CHANGE access
|
|
if (hService != NULL) {
|
|
// Make the changes.
|
|
if (!ChangeServiceConfig(
|
|
hService, // handle of service
|
|
SERVICE_NO_CHANGE, // service type: no change
|
|
dwStartType, // change service start type
|
|
SERVICE_NO_CHANGE, // error control: no change
|
|
NULL, // binary path: no change
|
|
NULL, // load order group: no change
|
|
NULL, // tag ID: no change
|
|
NULL, // dependencies: no change
|
|
NULL, // account name: no change
|
|
NULL, // password: no change
|
|
NULL) ) // display name: no change
|
|
{
|
|
dwStatus = GetLastError();
|
|
}
|
|
}
|
|
// Release the database lock.
|
|
UnlockServiceDatabase(sclLock);
|
|
}
|
|
|
|
if (hService != NULL) CloseServiceHandle(hService);
|
|
return dwStatus;
|
|
} // DriverStartType
|