dxdiag: Improve the command-line parsing.
This commit is contained in:
parent
d8cac6ff57
commit
1cd007fa99
|
@ -25,71 +25,128 @@
|
||||||
|
|
||||||
WINE_DEFAULT_DEBUG_CHANNEL(dxdiag);
|
WINE_DEFAULT_DEBUG_CHANNEL(dxdiag);
|
||||||
|
|
||||||
|
struct command_line_info
|
||||||
|
{
|
||||||
|
WCHAR outfile[MAX_PATH];
|
||||||
|
BOOL whql_check;
|
||||||
|
};
|
||||||
|
|
||||||
|
static void usage(void)
|
||||||
|
{
|
||||||
|
WINE_FIXME("Usage message box is not implemented\n");
|
||||||
|
ExitProcess(0);
|
||||||
|
}
|
||||||
|
|
||||||
|
static BOOL process_file_name(const WCHAR *cmdline, WCHAR *filename, size_t filename_len)
|
||||||
|
{
|
||||||
|
const WCHAR *endptr;
|
||||||
|
size_t len;
|
||||||
|
|
||||||
|
/* Skip any intervening spaces. */
|
||||||
|
while (*cmdline == ' ')
|
||||||
|
cmdline++;
|
||||||
|
|
||||||
|
/* Ignore filename quoting, if any. */
|
||||||
|
if (*cmdline == '"' && (endptr = strrchrW(cmdline, '"')))
|
||||||
|
{
|
||||||
|
/* Reject a string with only one quote. */
|
||||||
|
if (cmdline == endptr)
|
||||||
|
return FALSE;
|
||||||
|
|
||||||
|
cmdline++;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
endptr = cmdline + strlenW(cmdline);
|
||||||
|
|
||||||
|
len = endptr - cmdline;
|
||||||
|
if (len == 0 || len >= filename_len)
|
||||||
|
return FALSE;
|
||||||
|
|
||||||
|
memcpy(filename, cmdline, len * sizeof(WCHAR));
|
||||||
|
filename[len] = '\0';
|
||||||
|
|
||||||
|
return TRUE;
|
||||||
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
Process options [/WHQL:ON|OFF][/X outfile][/T outfile]
|
Process options [/WHQL:ON|OFF][/X outfile|/T outfile]
|
||||||
Returns TRUE if options were present, FALSE otherwise
|
Returns TRUE if options were present, FALSE otherwise
|
||||||
FIXME: Native behavior seems to be:
|
|
||||||
Only one of /X and /T is allowed, /WHQL must come before /X and /T,
|
Only one of /X and /T is allowed, /WHQL must come before /X and /T,
|
||||||
quotes are optional around the filename, even if it contains spaces.
|
and the rest of the command line after /X or /T is interpreted as a
|
||||||
|
filename. If a non-option portion of the command line is encountered,
|
||||||
|
dxdiag assumes that the string is a filename for the /T option.
|
||||||
|
|
||||||
|
Native does not interpret quotes, but quotes are parsed here because of how
|
||||||
|
Wine handles the command line.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
static BOOL ProcessCommandLine(const WCHAR *s)
|
static BOOL process_command_line(const WCHAR *cmdline, struct command_line_info *info)
|
||||||
{
|
{
|
||||||
WCHAR outfile[MAX_PATH+1];
|
static const WCHAR whql_colonW[] = {'w','h','q','l',':',0};
|
||||||
int opt_t = FALSE;
|
static const WCHAR offW[] = {'o','f','f',0};
|
||||||
int opt_x = FALSE;
|
static const WCHAR onW[] = {'o','n',0};
|
||||||
int opt_help = FALSE;
|
|
||||||
int opt_given = FALSE;
|
|
||||||
int want_arg = FALSE;
|
|
||||||
|
|
||||||
outfile[0] = 0;
|
info->whql_check = FALSE;
|
||||||
while (*s) {
|
|
||||||
|
while (*cmdline)
|
||||||
|
{
|
||||||
/* Skip whitespace before arg */
|
/* Skip whitespace before arg */
|
||||||
while (*s == ' ')
|
while (*cmdline == ' ')
|
||||||
s++;
|
cmdline++;
|
||||||
/* Check for option */
|
|
||||||
if (*s != '-' && *s != '/')
|
/* If no option is specified, treat the command line as a filename. */
|
||||||
return FALSE;
|
if (*cmdline != '-' && *cmdline != '/')
|
||||||
s++;
|
return process_file_name(cmdline, info->outfile, sizeof(info->outfile)/sizeof(WCHAR));
|
||||||
switch (*s++) {
|
|
||||||
|
cmdline++;
|
||||||
|
|
||||||
|
switch (*cmdline)
|
||||||
|
{
|
||||||
case 'T':
|
case 'T':
|
||||||
case 't': opt_t = TRUE; want_arg = TRUE; opt_given = TRUE; break;
|
case 't':
|
||||||
|
return process_file_name(cmdline + 1, info->outfile, sizeof(info->outfile)/sizeof(WCHAR));
|
||||||
case 'X':
|
case 'X':
|
||||||
case 'x': opt_x = TRUE; want_arg = TRUE; opt_given = TRUE; break;
|
case 'x':
|
||||||
|
return process_file_name(cmdline + 1, info->outfile, sizeof(info->outfile)/sizeof(WCHAR));
|
||||||
case 'W':
|
case 'W':
|
||||||
case 'w':
|
case 'w':
|
||||||
opt_given = TRUE;
|
if (strncmpiW(cmdline, whql_colonW, 5))
|
||||||
while (isalphaW(*s) || *s == ':')
|
return FALSE;
|
||||||
s++;
|
|
||||||
|
cmdline += 5;
|
||||||
|
|
||||||
|
if (!strncmpiW(cmdline, offW, 3))
|
||||||
|
{
|
||||||
|
info->whql_check = FALSE;
|
||||||
|
cmdline += 2;
|
||||||
|
}
|
||||||
|
else if (!strncmpiW(cmdline, onW, 2))
|
||||||
|
{
|
||||||
|
info->whql_check = TRUE;
|
||||||
|
cmdline++;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
return FALSE;
|
||||||
|
|
||||||
break;
|
break;
|
||||||
default: opt_help = TRUE; opt_given = TRUE; break;
|
default:
|
||||||
|
return FALSE;
|
||||||
}
|
}
|
||||||
/* Skip any spaces before next option or filename */
|
|
||||||
while (*s == ' ')
|
cmdline++;
|
||||||
s++;
|
|
||||||
if (want_arg) {
|
|
||||||
int i;
|
|
||||||
if (*s == '"')
|
|
||||||
s++;
|
|
||||||
for (i=0; i < MAX_PATH && *s && *s != '"'; i++, s++)
|
|
||||||
outfile[i] = *s;
|
|
||||||
outfile[i] = 0;
|
|
||||||
break;
|
|
||||||
}
|
}
|
||||||
}
|
|
||||||
if (opt_help)
|
return TRUE;
|
||||||
WINE_FIXME("help unimplemented\n");
|
|
||||||
if (opt_t)
|
|
||||||
WINE_FIXME("/t unimplemented\n");
|
|
||||||
if (opt_x)
|
|
||||||
WINE_FIXME("/x unimplemented\n");
|
|
||||||
return opt_given;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
int WINAPI wWinMain(HINSTANCE hInst, HINSTANCE hPrevInst, LPWSTR cmdline, int cmdshow)
|
int WINAPI wWinMain(HINSTANCE hInst, HINSTANCE hPrevInst, LPWSTR cmdline, int cmdshow)
|
||||||
{
|
{
|
||||||
if (ProcessCommandLine(cmdline))
|
struct command_line_info info;
|
||||||
return 0;
|
|
||||||
|
if (!process_command_line(cmdline, &info))
|
||||||
|
usage();
|
||||||
|
|
||||||
|
WINE_TRACE("WHQL check: %s\n", info.whql_check ? "TRUE" : "FALSE");
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in New Issue