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.
757 lines
19 KiB
757 lines
19 KiB
/*++
|
|
|
|
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 ;
|
|
}
|
|
|