Windows-Server-2003/sdktools/ctlcode/ctlcode.c

758 lines
18 KiB
C
Raw Normal View History

2024-08-04 01:28:15 +02:00
/*++
Copyright (c) 1995 Microsoft Corporation
Module Name:
CtlCode.c
Abstract:
A user mode app that breaks down a CTL_CODE (from IOCTL Irp)
Into its component parts of BASE, #, Method, and Access.
Environment:
User mode only
Revision History:
07-14-98 : Created by henrygab
--*/
#include <stdio.h>
#include <stdlib.h>
#include <stdarg.h>
#include <string.h>
#include <math.h>
#include <winerror.h>
#include <strsafe.h>
#include "CtlCode.h"
#if DBG
#define DEBUG_BUFFER_LENGTH 1000
ULONG DebugLevel = 0;
UCHAR DebugBuffer[DEBUG_BUFFER_LENGTH];
VOID
__cdecl
CtlCodeDebugPrint(
ULONG DebugPrintLevel,
PCCHAR DebugMessage,
...
)
{
if ((DebugPrintLevel <= (DebugLevel & 0x0000ffff)) ||
((1 << (DebugPrintLevel + 15)) & DebugLevel)
) {
HRESULT hr;
va_list ap;
va_start(ap, DebugMessage);
hr = StringCchVPrintf(DebugBuffer,
DEBUG_BUFFER_LENGTH,
DebugMessage,
ap);
if ((HRESULT_CODE(hr) == ERROR_INSUFFICIENT_BUFFER) || SUCCEEDED(hr)) {
fprintf(stderr, DebugBuffer);
}
va_end(ap);
}
}
#define DebugPrint(x) CtlCodeDebugPrint x
#else
#define DebugPrint(x)
#endif // DBG
VOID
DecodeIoctl(
PCTL_CODE CtlCode
);
BOOLEAN
IsHexNumber(
const char *szExpression
);
BOOLEAN
IsDecNumber(
const char *szExpression
);
//
// List of commands
// all command names are case sensitive
// arguments are passed into command routines
// list must be terminated with NULL command
// command will not be listed in help if description == NULL
//
ULONG32 ListCommand();
//
// prints an attenuation table based off cdrom standard volume
//
ULONG32 AttenuateCommand( int argc, char *argv[]);
VOID FindCommand(int argc, char *argv[]);
ULONG32 DecodeCommand(int argc, char *argv[]);
ULONG32 EncodeCommand(int argc, char *argv[]);
int __cdecl main(int argc, char *argv[])
/*++
Routine Description:
Parses input, showing help or calling function requested appropriately
Return Value:
0 - success
-1 - insufficient arguments
-2 - error opening device (DNE?)
--*/
{
int i = 0;
DebugPrint((3, "main => entering\n"));
if (argc < 2 ||
!strcmp(argv[1], "-?") ||
!strcmp(argv[1], "-h") ||
!strcmp(argv[1], "/?") ||
!strcmp(argv[1], "/h")
) {
DebugPrint((3, "main => Help requested...\n"));
ListCommand();
return -1;
}
if (argc != 2 && argc != 5) {
DebugPrint((3, "main => bad argc: %x, printing help\n", argc));
printf("Usage: ctl_code [parameters]\n");
return ListCommand();
}
if (argc == 5) {
DebugPrint((3, "main => encoding four args to one ioctl\n"));
EncodeCommand((argc - 1), &(argv[1]));
} else if (!IsHexNumber(argv[1])) {
//
// probably a string, so find matches?
//
DebugPrint((3, "main => non-hex argument, searching for matches\n"));
FindCommand((argc - 1), &(argv[1]));
} else {
//
// only one number passed in, so decode it
//
DebugPrint((3, "main => one hex argument, decoding\n"));
DecodeCommand((argc - 1), &(argv[1]));
}
return 0;
}
ULONG32 ListCommand()
/*++
Routine Description:
Prints out the command list (help)
Arguments:
argc - unused
argv - unused
Return Value:
STATUS_SUCCESS
--*/
{
printf("\n"
"CtlCode encodes/decodes ioctls into their four parts\n"
"(device type, function, method, access) and prints them out\n"
"symbolically. If encoding an ioctl, symbolic names can be\n"
"used for many inputs:\n"
"\tDevice Type (can drop the FILE_DEVICE prefix)\n"
"\tFunction (not applicable)\n"
"\tMethods (can drop the METHOD_ prefix)\n"
"\tAccess (can drop the FILE_ prefix and/or _ACCESS postfix)\n"
"\n"
"Also, any search string with only one match will give\n"
"full information. The following two commands are\n"
"equivalent if no other ioctl has the substring 'UNLOAD':\n"
"\tCtlCode.exe IOCTL_CDROM_UNLOAD_DRIVER\n"
"\tCtlCode.exe UNLOAD\n"
"\n"
"All input and output is in hexadecimal"
" string - prints all matches\n"
" # - decodes the ioctl\n"
" # # # # - encodes the ioctl base/#/method/access\n"
);
return 0;
}
VOID FindCommand(int argc, char *argv[])
{
char * currentPosition;
size_t arglen;
BOOLEAN found;
LONG i;
LONG j;
LONG numberOfMatches;
LONG lastMatch;
DebugPrint((3, "Find => entering\n"));
if (argc != 1) {
DebugPrint((0,
"Find !! Programming error !!\n"
"Find !! should only pass in one string !!\n"
"Find !! to match against. Passed in %2x !!\n",
argc + 1
));
return;
}
numberOfMatches = 0;
//
// for each name in the table
//
for (j=0;TableIoctlValue[j].Name != NULL;j++) {
currentPosition = TableIoctlValue[j].Name;
found = FALSE;
//
// see if we can match it to any argument
//
DebugPrint((3, "Find => matching against table entry %x\n", j));
arglen = strlen(argv[0]);
//
// accept partial matches to any substring
//
while (*currentPosition != 0) {
if (_strnicmp(argv[0],
currentPosition,
arglen)==0) {
found = TRUE;
break; // out of while loop
}
currentPosition++;
}
//
// if found, print it.
//
if (found) {
if (numberOfMatches == 0) {
//
// don't print the first match right away,
// as it may be the only match, which should
// then be decoded
//
DebugPrint((3, "Find => First Match (%x) found\n", j));
lastMatch = j;
} else if (numberOfMatches == 1) {
//
// if this is the second match, print the header
// and previous match info also
//
DebugPrint((3, "Find => Second Match (%x) found\n", j));
printf("Found the following matches:\n");
printf("\t%-40s - %16x\n",
TableIoctlValue[lastMatch].Name,
TableIoctlValue[lastMatch].Code);
printf("\t%-40s - %16x\n",
TableIoctlValue[j].Name,
TableIoctlValue[j].Code);
} else {
DebugPrint((3, "Find => Another Match (%x) found\n", j));
printf("\t%-40s - %16x\n",
TableIoctlValue[j].Name,
TableIoctlValue[j].Code);
}
numberOfMatches++;
} // end if (found) {}
} // end of loop through table
DebugPrint((2, "Find => Found %x matches total\n", numberOfMatches));
//
// if didn't find any matches, tell them so.
//
if (numberOfMatches == 0) {
printf("No matches found.\n");
} else if (numberOfMatches == 1) {
DebugPrint((2, "Find => Decoding ioctl at index (%x)\n", lastMatch));
DecodeIoctl((PVOID)&(TableIoctlValue[lastMatch].Code));
}
}
ULONG32 EncodeCommand(int argc, char *argv[])
/*++
Routine Description:
Change four components into a Ctl_Code
Arguments:
argc - the number of additional arguments. prompt if zero
argv - the additional arguments
Return Value:
STATUS_SUCCESS if successful
--*/
{
CTL_CODE maxValues;
CTL_CODE encoded;
ULONG temp;
encoded.Code = 0;
maxValues.Code = -1; // all 1's
DebugPrint((3, "Encode => entering\n"));
// device type
if (IsHexNumber(argv[0])) {
//
// read and verify the hex number
//
DebugPrint((3, "Encode => arg 1 is hex\n"));
temp = strtol(argv[0], (char**)NULL, 0x10);
if (temp > maxValues.DeviceType) {
printf("Max Device Type: %x\n", maxValues.DeviceType);
return STATUS_SUCCESS;
}
encoded.DeviceType = temp;
} else {
//
// read and match the device type
//
DebugPrint((3, "Encode => arg 1 is non-hex, attempting "
"string match\n"));
for (temp = 0; temp < MAX_IOCTL_DEVICE_TYPE; temp++) {
if (_stricmp(TableIoctlDeviceType[temp].Name, argv[0]) == 0) {
DebugPrint((2, "Encode => arg 1 matched index %x (full)\n",
temp));
encoded.DeviceType = TableIoctlDeviceType[temp].Value;
break;
}
//
// no need to have common prefixes
//
if ((strlen(TableIoctlDeviceType[temp].Name) > strlen("FILE_DEVICE_"))
&&
(_stricmp(TableIoctlDeviceType[temp].Name + strlen("FILE_DEVICE_"),argv[0]) == 0)
) {
DebugPrint((2, "Encode => arg 1 matched index %x "
"(dropped prefix)\n", temp));
encoded.DeviceType = TableIoctlDeviceType[temp].Value;
break;
}
}
if (temp == MAX_IOCTL_DEVICE_TYPE) {
printf("Device Type unknown. Known Device Types:\n");
for (temp = 0; temp < MAX_IOCTL_DEVICE_TYPE; temp++) {
printf("\t%s\n", TableIoctlDeviceType[temp].Name);
}
return STATUS_SUCCESS;
}
DebugPrint((3, "Encode => arg 1 matched string index %x\n", temp));
}
// function number
if (IsHexNumber(argv[1])) {
DebugPrint((3, "Encode => arg 2 is hex\n"));
//
// read and verify the hex number
//
temp = strtol(argv[1], (char**)NULL, 0x10);
if (temp > maxValues.Function) {
printf("Max Function: %x\n", maxValues.Function);
return STATUS_SUCCESS;
}
encoded.Function = temp;
} else {
printf("Function: must be a hex number\n");
return STATUS_SUCCESS;
}
// method
if (IsHexNumber(argv[2])) {
DebugPrint((3, "Encode => arg 3 is hex\n"));
//
// read and verify the hex number
//
temp = strtol(argv[2], (char**)NULL, 0x10);
if (temp > maxValues.Method) {
printf("Max Method: %x\n", maxValues.Method);
return STATUS_SUCCESS;
}
encoded.Method = temp;
} else {
DebugPrint((3, "Encode => arg 3 is non-hex, attempting string "
"match\n"));
//
// read and match the method
//
for (temp = 0; temp < MAX_IOCTL_METHOD; temp++) {
if (_stricmp(TableIoctlMethod[temp].Name, argv[2]) == 0) {
DebugPrint((2, "Encode => arg 3 matched index %x\n", temp));
encoded.Method = TableIoctlMethod[temp].Value;
break;
}
//
// no need to have common prefixes
//
if ((strlen(TableIoctlMethod[temp].Name) > strlen("METHOD_"))
&&
(_stricmp(TableIoctlMethod[temp].Name + strlen("METHOD_"),argv[2]) == 0)
) {
DebugPrint((2, "Encode => arg 3 matched index %x "
"(dropped prefix)\n", temp));
encoded.Method = TableIoctlMethod[temp].Value;
break;
}
} // end ioctl_method loop
if (temp == MAX_IOCTL_METHOD) {
printf("Method %s unknown. Known methods:\n", argv[2]);
for (temp = 0; temp < MAX_IOCTL_METHOD; temp++) {
printf("\t%s\n", TableIoctlMethod[temp].Name);
}
return STATUS_SUCCESS;
}
}
// access
if (IsHexNumber(argv[3])) {
//
// read and verify the hex number
//
DebugPrint((3, "Encode => arg 4 is hex\n"));
temp = strtol(argv[3], (char**)NULL, 0x10);
if (temp > maxValues.Access) {
printf("Max Device Type: %x\n", maxValues.Access);
return STATUS_SUCCESS;
}
encoded.Access = temp;
} else {
DebugPrint((3, "Encode => arg 4 is non-hex, attempting to "
"match strings\n", temp));
//
// read and match the access type
//
DebugPrint((4, "Encode => Trying to match %s\n", argv[3]));
for (temp = 0; temp < MAX_IOCTL_ACCESS; temp++) {
int tLen;
size_t tDrop;
char *string;
char *match;
//
// match the whole string?
//
string = argv[3];
match = TableIoctlAccess[temp].Name;
DebugPrint((4, "Encode ?? test match against %s\n", match));
if (_stricmp(match, string) == 0) {
DebugPrint((2, "Encode => arg 4 matched index %x (full)\n",
temp));
encoded.Access = TableIoctlAccess[temp].Value;
break;
}
//
// maybe match without the trailing _ACCESS?
//
tLen = strlen(match) - strlen("_ACCESS");
DebugPrint((4, "Encode ?? test match against %s (%x chars)\n",
match, tLen));
if (_strnicmp(match, string, tLen) == 0) {
DebugPrint((2, "Encode => arg 4 matched index %x "
"(dropped postfix)\n", temp));
encoded.Access = TableIoctlAccess[temp].Value;
break;
}
//
// no need to have common prefixes
//
match += strlen("FILE_");
DebugPrint((4, "Encode ?? test match against %s\n", match));
if (_stricmp(match, string) == 0) {
DebugPrint((2, "Encode => arg 4 matched index %x "
"(dropped prefix)\n", temp));
encoded.Access = TableIoctlAccess[temp].Value;
break;
}
tLen = strlen(match) - strlen("_ACCESS");
//
// maybe match without prefix or suffix?
//
DebugPrint((4, "Encode ?? test match against %s (%x chars)\n",
match, tLen));
if (_strnicmp(match, string, tLen) == 0) {
DebugPrint((2, "Encode => arg 4 matched index %x "
"(dropped prefix and postfix)\n", temp));
encoded.Access = TableIoctlAccess[temp].Value;
break;
}
} // end ioctl_access loop
if (temp == MAX_IOCTL_ACCESS) {
printf("Access %s unknown. Known Access Types:\n", argv[3]);
for (temp = 0; temp < MAX_IOCTL_ACCESS; temp++) {
printf("\t%s\n", TableIoctlAccess[temp].Name);
}
return STATUS_SUCCESS;
}
}
DecodeIoctl(&encoded);
//
// file type of 0 == unknown type
//
return STATUS_SUCCESS;
}
ULONG32 DecodeCommand(int argc, char *argv[])
/*++
Routine Description:
Change a Ctl_Code into four components
Arguments:
argc - the number of additional arguments. prompt if zero
argv - the additional arguments
Return Value:
STATUS_SUCCESS if successful
--*/
{
CTL_CODE ctlCode;
ULONG i;
DebugPrint((3, "Decode => Entering\n"));
ctlCode.Code = strtoul(argv[0], (char**)NULL, 0x10);
DecodeIoctl(&ctlCode);
return STATUS_SUCCESS;
}
VOID
DecodeIoctl(
PCTL_CODE CtlCode
)
{
ULONG i;
for (i = 0; TableIoctlValue[i].Name != NULL; i++) {
if (TableIoctlValue[i].Code == CtlCode->Code) break;
}
printf(" Ioctl: %08x %s\n",
CtlCode->Code,
(TableIoctlValue[i].Name ? TableIoctlValue[i].Name : "Unknown")
);
printf("DeviceType: %04x - ", CtlCode->DeviceType);
if (CtlCode->DeviceType > MAX_IOCTL_DEVICE_TYPE) {
printf("Unknown\n");
} else {
printf("%s\n", TableIoctlDeviceType[ CtlCode->DeviceType ].Name);
}
printf(" Function: %04x \n", CtlCode->Function);
printf(" Method: %04x - %s\n",
CtlCode->Method,
TableIoctlMethod[CtlCode->Method].Name
);
printf(" Access: %04x - %s\n",
CtlCode->Access,
TableIoctlAccess[CtlCode->Access].Name
);
return;
}
ULONG32 AttenuateCommand( int argc, char *argv[])
{
LONG32 i;
LONG32 j;
long double val[] = {
0xff, 0xf0, 0xe0, 0xc0,
0x80, 0x40, 0x20, 0x10,
0x0f, 0x0e, 0x0c, 0x08,
0x04, 0x02, 0x01, 0x00
};
long double temp;
printf( "\nATTENUATION AttenuationTable[] = {\n" );
for ( i=0; i < sizeof(val)/sizeof(val[0]); i++ ) {
temp = val[i];
temp = 20 * log10( temp / 256.0 );
temp = temp * 65536;
printf( " { 0x%08x, 0x%02x },\n", (LONG)temp, (LONG)val[i] );
}
printf( "};\n" );
return STATUS_SUCCESS;
}
BOOLEAN
IsHexNumber(
const char *szExpression
)
{
if (!szExpression[0]) {
return FALSE ;
}
for(;*szExpression; szExpression++) {
if ((*szExpression)< '0') { return FALSE ; }
else if ((*szExpression)> 'f') { return FALSE ; }
else if ((*szExpression)>='a') { continue ; }
else if ((*szExpression)> 'F') { return FALSE ; }
else if ((*szExpression)<='9') { continue ; }
else if ((*szExpression)>='A') { continue ; }
else { return FALSE ; }
}
return TRUE ;
}
BOOLEAN
IsDecNumber(
const char *szExpression
)
{
if (!szExpression[0]) {
return FALSE ;
}
while(*szExpression) {
if ((*szExpression)<'0') { return FALSE ; }
else if ((*szExpression)>'9') { return FALSE ; }
szExpression ++ ;
}
return TRUE ;
}