Commit 1cd007fa authored by Andrew Nguyen's avatar Andrew Nguyen Committed by Alexandre Julliard

dxdiag: Improve the command-line parsing.

parent d8cac6ff
...@@ -25,71 +25,128 @@ ...@@ -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; info->whql_check = FALSE;
int want_arg = FALSE;
while (*cmdline)
outfile[0] = 0; {
while (*s) {
/* 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++;
break; cmdline += 5;
default: opt_help = TRUE; opt_given = TRUE; break;
if (!strncmpiW(cmdline, offW, 3))
{
info->whql_check = FALSE;
cmdline += 2;
} }
/* Skip any spaces before next option or filename */ else if (!strncmpiW(cmdline, onW, 2))
while (*s == ' ') {
s++; info->whql_check = TRUE;
if (want_arg) { cmdline++;
int i; }
if (*s == '"') else
s++; return FALSE;
for (i=0; i < MAX_PATH && *s && *s != '"'; i++, s++)
outfile[i] = *s;
outfile[i] = 0;
break; break;
default:
return FALSE;
} }
cmdline++;
} }
if (opt_help)
WINE_FIXME("help unimplemented\n"); return TRUE;
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;
} }
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment