introduce environment variables for all qemu-user options
(Edits by Riku Voipio to apply to current HEAD) Rework option parsing code for linux-user in a table-driven manner to allow environment variables for all commandline options. Also generate usage() output from option table. Fix complains from checkpatch.pl, also have envlist global Signed-off-by: Johannes Schauer <j.schauer@email.de> Signed-off-by: Riku Voipio <riku.voipio@linaro.org>
This commit is contained in:
parent
a790ae389b
commit
fc9c54124d
@ -39,6 +39,11 @@
|
|||||||
char *exec_path;
|
char *exec_path;
|
||||||
|
|
||||||
int singlestep;
|
int singlestep;
|
||||||
|
const char *filename;
|
||||||
|
const char *argv0;
|
||||||
|
int gdbstub_port;
|
||||||
|
envlist_t *envlist;
|
||||||
|
const char *cpu_model;
|
||||||
unsigned long mmap_min_addr;
|
unsigned long mmap_min_addr;
|
||||||
#if defined(CONFIG_USE_GUEST_BASE)
|
#if defined(CONFIG_USE_GUEST_BASE)
|
||||||
unsigned long guest_base;
|
unsigned long guest_base;
|
||||||
@ -46,6 +51,8 @@ int have_guest_base;
|
|||||||
unsigned long reserved_va;
|
unsigned long reserved_va;
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
static void usage(void);
|
||||||
|
|
||||||
static const char *interp_prefix = CONFIG_QEMU_INTERP_PREFIX;
|
static const char *interp_prefix = CONFIG_QEMU_INTERP_PREFIX;
|
||||||
const char *qemu_uname_release = CONFIG_UNAME_RELEASE;
|
const char *qemu_uname_release = CONFIG_UNAME_RELEASE;
|
||||||
|
|
||||||
@ -2879,57 +2886,6 @@ void cpu_loop(CPUS390XState *env)
|
|||||||
|
|
||||||
#endif /* TARGET_S390X */
|
#endif /* TARGET_S390X */
|
||||||
|
|
||||||
static void version(void)
|
|
||||||
{
|
|
||||||
printf("qemu-" TARGET_ARCH " version " QEMU_VERSION QEMU_PKGVERSION
|
|
||||||
", Copyright (c) 2003-2008 Fabrice Bellard\n");
|
|
||||||
}
|
|
||||||
|
|
||||||
static void usage(void)
|
|
||||||
{
|
|
||||||
version();
|
|
||||||
printf("usage: qemu-" TARGET_ARCH " [options] program [arguments...]\n"
|
|
||||||
"Linux CPU emulator (compiled for %s emulation)\n"
|
|
||||||
"\n"
|
|
||||||
"Standard options:\n"
|
|
||||||
"-h print this help\n"
|
|
||||||
"-version display version information and exit\n"
|
|
||||||
"-g port wait gdb connection to port\n"
|
|
||||||
"-L path set the elf interpreter prefix (default=%s)\n"
|
|
||||||
"-s size set the stack size in bytes (default=%ld)\n"
|
|
||||||
"-cpu model select CPU (-cpu ? for list)\n"
|
|
||||||
"-drop-ld-preload drop LD_PRELOAD for target process\n"
|
|
||||||
"-E var=value sets/modifies targets environment variable(s)\n"
|
|
||||||
"-U var unsets targets environment variable(s)\n"
|
|
||||||
"-0 argv0 forces target process argv[0] to be argv0\n"
|
|
||||||
#if defined(CONFIG_USE_GUEST_BASE)
|
|
||||||
"-B address set guest_base address to address\n"
|
|
||||||
"-R size reserve size bytes for guest virtual address space\n"
|
|
||||||
#endif
|
|
||||||
"\n"
|
|
||||||
"Debug options:\n"
|
|
||||||
"-d options activate log (logfile=%s)\n"
|
|
||||||
"-p pagesize set the host page size to 'pagesize'\n"
|
|
||||||
"-singlestep always run in singlestep mode\n"
|
|
||||||
"-strace log system calls\n"
|
|
||||||
"\n"
|
|
||||||
"Environment variables:\n"
|
|
||||||
"QEMU_STRACE Print system calls and arguments similar to the\n"
|
|
||||||
" 'strace' program. Enable by setting to any value.\n"
|
|
||||||
"You can use -E and -U options to set/unset environment variables\n"
|
|
||||||
"for target process. It is possible to provide several variables\n"
|
|
||||||
"by repeating the option. For example:\n"
|
|
||||||
" -E var1=val2 -E var2=val2 -U LD_PRELOAD -U LD_DEBUG\n"
|
|
||||||
"Note that if you provide several changes to single variable\n"
|
|
||||||
"last change will stay in effect.\n"
|
|
||||||
,
|
|
||||||
TARGET_ARCH,
|
|
||||||
interp_prefix,
|
|
||||||
guest_stack_size,
|
|
||||||
DEBUG_LOGFILE);
|
|
||||||
exit(1);
|
|
||||||
}
|
|
||||||
|
|
||||||
THREAD CPUState *thread_env;
|
THREAD CPUState *thread_env;
|
||||||
|
|
||||||
void task_settid(TaskState *ts)
|
void task_settid(TaskState *ts)
|
||||||
@ -2966,25 +2922,357 @@ void init_task_state(TaskState *ts)
|
|||||||
ts->sigqueue_table[i].next = NULL;
|
ts->sigqueue_table[i].next = NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static void handle_arg_help(const char *arg)
|
||||||
|
{
|
||||||
|
usage();
|
||||||
|
}
|
||||||
|
|
||||||
|
static void handle_arg_log(const char *arg)
|
||||||
|
{
|
||||||
|
int mask;
|
||||||
|
const CPULogItem *item;
|
||||||
|
|
||||||
|
mask = cpu_str_to_log_mask(arg);
|
||||||
|
if (!mask) {
|
||||||
|
printf("Log items (comma separated):\n");
|
||||||
|
for (item = cpu_log_items; item->mask != 0; item++) {
|
||||||
|
printf("%-10s %s\n", item->name, item->help);
|
||||||
|
}
|
||||||
|
exit(1);
|
||||||
|
}
|
||||||
|
cpu_set_log(mask);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void handle_arg_set_env(const char *arg)
|
||||||
|
{
|
||||||
|
char *r, *p, *token;
|
||||||
|
r = p = strdup(arg);
|
||||||
|
while ((token = strsep(&p, ",")) != NULL) {
|
||||||
|
if (envlist_setenv(envlist, token) != 0) {
|
||||||
|
usage();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
free(r);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void handle_arg_unset_env(const char *arg)
|
||||||
|
{
|
||||||
|
char *r, *p, *token;
|
||||||
|
r = p = strdup(arg);
|
||||||
|
while ((token = strsep(&p, ",")) != NULL) {
|
||||||
|
if (envlist_unsetenv(envlist, token) != 0) {
|
||||||
|
usage();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
free(r);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void handle_arg_argv0(const char *arg)
|
||||||
|
{
|
||||||
|
argv0 = strdup(arg);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void handle_arg_stack_size(const char *arg)
|
||||||
|
{
|
||||||
|
char *p;
|
||||||
|
guest_stack_size = strtoul(arg, &p, 0);
|
||||||
|
if (guest_stack_size == 0) {
|
||||||
|
usage();
|
||||||
|
}
|
||||||
|
|
||||||
|
if (*p == 'M') {
|
||||||
|
guest_stack_size *= 1024 * 1024;
|
||||||
|
} else if (*p == 'k' || *p == 'K') {
|
||||||
|
guest_stack_size *= 1024;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static void handle_arg_ld_prefix(const char *arg)
|
||||||
|
{
|
||||||
|
interp_prefix = strdup(arg);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void handle_arg_pagesize(const char *arg)
|
||||||
|
{
|
||||||
|
qemu_host_page_size = atoi(arg);
|
||||||
|
if (qemu_host_page_size == 0 ||
|
||||||
|
(qemu_host_page_size & (qemu_host_page_size - 1)) != 0) {
|
||||||
|
fprintf(stderr, "page size must be a power of two\n");
|
||||||
|
exit(1);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static void handle_arg_gdb(const char *arg)
|
||||||
|
{
|
||||||
|
gdbstub_port = atoi(arg);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void handle_arg_uname(const char *arg)
|
||||||
|
{
|
||||||
|
qemu_uname_release = strdup(arg);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void handle_arg_cpu(const char *arg)
|
||||||
|
{
|
||||||
|
cpu_model = strdup(arg);
|
||||||
|
if (cpu_model == NULL || strcmp(cpu_model, "?") == 0) {
|
||||||
|
/* XXX: implement xxx_cpu_list for targets that still miss it */
|
||||||
|
#if defined(cpu_list_id)
|
||||||
|
cpu_list_id(stdout, &fprintf, "");
|
||||||
|
#elif defined(cpu_list)
|
||||||
|
cpu_list(stdout, &fprintf); /* deprecated */
|
||||||
|
#endif
|
||||||
|
exit(1);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#if defined(CONFIG_USE_GUEST_BASE)
|
||||||
|
static void handle_arg_guest_base(const char *arg)
|
||||||
|
{
|
||||||
|
guest_base = strtol(arg, NULL, 0);
|
||||||
|
have_guest_base = 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
static void handle_arg_reserved_va(const char *arg)
|
||||||
|
{
|
||||||
|
char *p;
|
||||||
|
int shift = 0;
|
||||||
|
reserved_va = strtoul(arg, &p, 0);
|
||||||
|
switch (*p) {
|
||||||
|
case 'k':
|
||||||
|
case 'K':
|
||||||
|
shift = 10;
|
||||||
|
break;
|
||||||
|
case 'M':
|
||||||
|
shift = 20;
|
||||||
|
break;
|
||||||
|
case 'G':
|
||||||
|
shift = 30;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
if (shift) {
|
||||||
|
unsigned long unshifted = reserved_va;
|
||||||
|
p++;
|
||||||
|
reserved_va <<= shift;
|
||||||
|
if (((reserved_va >> shift) != unshifted)
|
||||||
|
#if HOST_LONG_BITS > TARGET_VIRT_ADDR_SPACE_BITS
|
||||||
|
|| (reserved_va > (1ul << TARGET_VIRT_ADDR_SPACE_BITS))
|
||||||
|
#endif
|
||||||
|
) {
|
||||||
|
fprintf(stderr, "Reserved virtual address too big\n");
|
||||||
|
exit(1);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (*p) {
|
||||||
|
fprintf(stderr, "Unrecognised -R size suffix '%s'\n", p);
|
||||||
|
exit(1);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
|
||||||
|
static void handle_arg_singlestep(const char *arg)
|
||||||
|
{
|
||||||
|
singlestep = 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
static void handle_arg_strace(const char *arg)
|
||||||
|
{
|
||||||
|
do_strace = 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
static void handle_arg_version(const char *arg)
|
||||||
|
{
|
||||||
|
printf("qemu-" TARGET_ARCH " version " QEMU_VERSION QEMU_PKGVERSION
|
||||||
|
", Copyright (c) 2003-2008 Fabrice Bellard\n");
|
||||||
|
}
|
||||||
|
|
||||||
|
struct qemu_argument {
|
||||||
|
const char *argv;
|
||||||
|
const char *env;
|
||||||
|
bool has_arg;
|
||||||
|
void (*handle_opt)(const char *arg);
|
||||||
|
const char *example;
|
||||||
|
const char *help;
|
||||||
|
};
|
||||||
|
|
||||||
|
struct qemu_argument arg_table[] = {
|
||||||
|
{"h", "", false, handle_arg_help,
|
||||||
|
"", "print this help"},
|
||||||
|
{"g", "QEMU_GDB", true, handle_arg_gdb,
|
||||||
|
"port", "wait gdb connection to 'port'"},
|
||||||
|
{"L", "QEMU_LD_PREFIX", true, handle_arg_ld_prefix,
|
||||||
|
"path", "set the elf interpreter prefix to 'path'"},
|
||||||
|
{"s", "QEMU_STACK_SIZE", true, handle_arg_stack_size,
|
||||||
|
"size", "set the stack size to 'size' bytes"},
|
||||||
|
{"cpu", "QEMU_CPU", true, handle_arg_cpu,
|
||||||
|
"model", "select CPU (-cpu ? for list)"},
|
||||||
|
{"E", "QEMU_SET_ENV", true, handle_arg_set_env,
|
||||||
|
"var=value", "sets targets environment variable (see below)"},
|
||||||
|
{"U", "QEMU_UNSET_ENV", true, handle_arg_unset_env,
|
||||||
|
"var", "unsets targets environment variable (see below)"},
|
||||||
|
{"0", "QEMU_ARGV0", true, handle_arg_argv0,
|
||||||
|
"argv0", "forces target process argv[0] to be 'argv0'"},
|
||||||
|
{"r", "QEMU_UNAME", true, handle_arg_uname,
|
||||||
|
"uname", "set qemu uname release string to 'uname'"},
|
||||||
|
#if defined(CONFIG_USE_GUEST_BASE)
|
||||||
|
{"B", "QEMU_GUEST_BASE", true, handle_arg_guest_base,
|
||||||
|
"address", "set guest_base address to 'address'"},
|
||||||
|
{"R", "QEMU_RESERVED_VA", true, handle_arg_reserved_va,
|
||||||
|
"size", "reserve 'size' bytes for guest virtual address space"},
|
||||||
|
#endif
|
||||||
|
{"d", "QEMU_LOG", true, handle_arg_log,
|
||||||
|
"options", "activate log"},
|
||||||
|
{"p", "QEMU_PAGESIZE", true, handle_arg_pagesize,
|
||||||
|
"pagesize", "set the host page size to 'pagesize'"},
|
||||||
|
{"singlestep", "QEMU_SINGLESTEP", false, handle_arg_singlestep,
|
||||||
|
"", "run in singlestep mode"},
|
||||||
|
{"strace", "QEMU_STRACE", false, handle_arg_strace,
|
||||||
|
"", "log system calls"},
|
||||||
|
{"version", "QEMU_VERSION", false, handle_arg_version,
|
||||||
|
"", "log system calls"},
|
||||||
|
{NULL, NULL, false, NULL, NULL, NULL}
|
||||||
|
};
|
||||||
|
|
||||||
|
static void usage(void)
|
||||||
|
{
|
||||||
|
struct qemu_argument *arginfo;
|
||||||
|
int maxarglen;
|
||||||
|
int maxenvlen;
|
||||||
|
|
||||||
|
printf("usage: qemu-" TARGET_ARCH " [options] program [arguments...]\n"
|
||||||
|
"Linux CPU emulator (compiled for " TARGET_ARCH " emulation)\n"
|
||||||
|
"\n"
|
||||||
|
"Options and associated environment variables:\n"
|
||||||
|
"\n");
|
||||||
|
|
||||||
|
maxarglen = maxenvlen = 0;
|
||||||
|
|
||||||
|
for (arginfo = arg_table; arginfo->handle_opt != NULL; arginfo++) {
|
||||||
|
if (strlen(arginfo->env) > maxenvlen) {
|
||||||
|
maxenvlen = strlen(arginfo->env);
|
||||||
|
}
|
||||||
|
if (strlen(arginfo->argv) > maxarglen) {
|
||||||
|
maxarglen = strlen(arginfo->argv);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
printf("%-*s%-*sDescription\n", maxarglen+3, "Argument",
|
||||||
|
maxenvlen+1, "Env-variable");
|
||||||
|
|
||||||
|
for (arginfo = arg_table; arginfo->handle_opt != NULL; arginfo++) {
|
||||||
|
if (arginfo->has_arg) {
|
||||||
|
printf("-%s %-*s %-*s %s\n", arginfo->argv,
|
||||||
|
(int)(maxarglen-strlen(arginfo->argv)), arginfo->example,
|
||||||
|
maxenvlen, arginfo->env, arginfo->help);
|
||||||
|
} else {
|
||||||
|
printf("-%-*s %-*s %s\n", maxarglen+1, arginfo->argv,
|
||||||
|
maxenvlen, arginfo->env,
|
||||||
|
arginfo->help);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
printf("\n"
|
||||||
|
"Defaults:\n"
|
||||||
|
"QEMU_LD_PREFIX = %s\n"
|
||||||
|
"QEMU_STACK_SIZE = %ld byte\n"
|
||||||
|
"QEMU_LOG = %s\n",
|
||||||
|
interp_prefix,
|
||||||
|
guest_stack_size,
|
||||||
|
DEBUG_LOGFILE);
|
||||||
|
|
||||||
|
printf("\n"
|
||||||
|
"You can use -E and -U options or the QEMU_SET_ENV and\n"
|
||||||
|
"QEMU_UNSET_ENV environment variables to set and unset\n"
|
||||||
|
"environment variables for the target process.\n"
|
||||||
|
"It is possible to provide several variables by separating them\n"
|
||||||
|
"by commas in getsubopt(3) style. Additionally it is possible to\n"
|
||||||
|
"provide the -E and -U options multiple times.\n"
|
||||||
|
"The following lines are equivalent:\n"
|
||||||
|
" -E var1=val2 -E var2=val2 -U LD_PRELOAD -U LD_DEBUG\n"
|
||||||
|
" -E var1=val2,var2=val2 -U LD_PRELOAD,LD_DEBUG\n"
|
||||||
|
" QEMU_SET_ENV=var1=val2,var2=val2 QEMU_UNSET_ENV=LD_PRELOAD,LD_DEBUG\n"
|
||||||
|
"Note that if you provide several changes to a single variable\n"
|
||||||
|
"the last change will stay in effect.\n");
|
||||||
|
|
||||||
|
exit(1);
|
||||||
|
}
|
||||||
|
|
||||||
|
static int parse_args(int argc, char **argv)
|
||||||
|
{
|
||||||
|
const char *r;
|
||||||
|
int optind;
|
||||||
|
struct qemu_argument *arginfo;
|
||||||
|
|
||||||
|
for (arginfo = arg_table; arginfo->handle_opt != NULL; arginfo++) {
|
||||||
|
if (arginfo->env == NULL) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
r = getenv(arginfo->env);
|
||||||
|
if (r != NULL) {
|
||||||
|
arginfo->handle_opt(r);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
optind = 1;
|
||||||
|
for (;;) {
|
||||||
|
if (optind >= argc) {
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
r = argv[optind];
|
||||||
|
if (r[0] != '-') {
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
optind++;
|
||||||
|
r++;
|
||||||
|
if (!strcmp(r, "-")) {
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
for (arginfo = arg_table; arginfo->handle_opt != NULL; arginfo++) {
|
||||||
|
if (!strcmp(r, arginfo->argv)) {
|
||||||
|
if (optind >= argc) {
|
||||||
|
usage();
|
||||||
|
}
|
||||||
|
|
||||||
|
arginfo->handle_opt(argv[optind]);
|
||||||
|
|
||||||
|
if (arginfo->has_arg) {
|
||||||
|
optind++;
|
||||||
|
}
|
||||||
|
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/* no option matched the current argv */
|
||||||
|
if (arginfo->handle_opt == NULL) {
|
||||||
|
usage();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (optind >= argc) {
|
||||||
|
usage();
|
||||||
|
}
|
||||||
|
|
||||||
|
filename = argv[optind];
|
||||||
|
exec_path = argv[optind];
|
||||||
|
|
||||||
|
return optind;
|
||||||
|
}
|
||||||
|
|
||||||
int main(int argc, char **argv, char **envp)
|
int main(int argc, char **argv, char **envp)
|
||||||
{
|
{
|
||||||
const char *filename;
|
|
||||||
const char *cpu_model;
|
|
||||||
const char *log_file = DEBUG_LOGFILE;
|
const char *log_file = DEBUG_LOGFILE;
|
||||||
const char *log_mask = NULL;
|
|
||||||
struct target_pt_regs regs1, *regs = ®s1;
|
struct target_pt_regs regs1, *regs = ®s1;
|
||||||
struct image_info info1, *info = &info1;
|
struct image_info info1, *info = &info1;
|
||||||
struct linux_binprm bprm;
|
struct linux_binprm bprm;
|
||||||
TaskState *ts;
|
TaskState *ts;
|
||||||
CPUState *env;
|
CPUState *env;
|
||||||
int optind;
|
int optind;
|
||||||
const char *r;
|
|
||||||
int gdbstub_port = 0;
|
|
||||||
char **target_environ, **wrk;
|
char **target_environ, **wrk;
|
||||||
char **target_argv;
|
char **target_argv;
|
||||||
int target_argc;
|
int target_argc;
|
||||||
envlist_t *envlist = NULL;
|
|
||||||
const char *argv0 = NULL;
|
|
||||||
int i;
|
int i;
|
||||||
int ret;
|
int ret;
|
||||||
|
|
||||||
@ -3019,156 +3307,9 @@ int main(int argc, char **argv, char **envp)
|
|||||||
cpudef_setup(); /* parse cpu definitions in target config file (TBD) */
|
cpudef_setup(); /* parse cpu definitions in target config file (TBD) */
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
optind = 1;
|
|
||||||
for(;;) {
|
|
||||||
if (optind >= argc)
|
|
||||||
break;
|
|
||||||
r = argv[optind];
|
|
||||||
if (r[0] != '-')
|
|
||||||
break;
|
|
||||||
optind++;
|
|
||||||
r++;
|
|
||||||
if (!strcmp(r, "-")) {
|
|
||||||
break;
|
|
||||||
} else if (!strcmp(r, "d")) {
|
|
||||||
if (optind >= argc) {
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
log_mask = argv[optind++];
|
|
||||||
} else if (!strcmp(r, "D")) {
|
|
||||||
if (optind >= argc) {
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
log_file = argv[optind++];
|
|
||||||
} else if (!strcmp(r, "E")) {
|
|
||||||
r = argv[optind++];
|
|
||||||
if (envlist_setenv(envlist, r) != 0)
|
|
||||||
usage();
|
|
||||||
} else if (!strcmp(r, "ignore-environment")) {
|
|
||||||
envlist_free(envlist);
|
|
||||||
if ((envlist = envlist_create()) == NULL) {
|
|
||||||
(void) fprintf(stderr, "Unable to allocate envlist\n");
|
|
||||||
exit(1);
|
|
||||||
}
|
|
||||||
} else if (!strcmp(r, "U")) {
|
|
||||||
r = argv[optind++];
|
|
||||||
if (envlist_unsetenv(envlist, r) != 0)
|
|
||||||
usage();
|
|
||||||
} else if (!strcmp(r, "0")) {
|
|
||||||
r = argv[optind++];
|
|
||||||
argv0 = r;
|
|
||||||
} else if (!strcmp(r, "s")) {
|
|
||||||
if (optind >= argc)
|
|
||||||
break;
|
|
||||||
r = argv[optind++];
|
|
||||||
guest_stack_size = strtoul(r, (char **)&r, 0);
|
|
||||||
if (guest_stack_size == 0)
|
|
||||||
usage();
|
|
||||||
if (*r == 'M')
|
|
||||||
guest_stack_size *= 1024 * 1024;
|
|
||||||
else if (*r == 'k' || *r == 'K')
|
|
||||||
guest_stack_size *= 1024;
|
|
||||||
} else if (!strcmp(r, "L")) {
|
|
||||||
interp_prefix = argv[optind++];
|
|
||||||
} else if (!strcmp(r, "p")) {
|
|
||||||
if (optind >= argc)
|
|
||||||
break;
|
|
||||||
qemu_host_page_size = atoi(argv[optind++]);
|
|
||||||
if (qemu_host_page_size == 0 ||
|
|
||||||
(qemu_host_page_size & (qemu_host_page_size - 1)) != 0) {
|
|
||||||
fprintf(stderr, "page size must be a power of two\n");
|
|
||||||
exit(1);
|
|
||||||
}
|
|
||||||
} else if (!strcmp(r, "g")) {
|
|
||||||
if (optind >= argc)
|
|
||||||
break;
|
|
||||||
gdbstub_port = atoi(argv[optind++]);
|
|
||||||
} else if (!strcmp(r, "r")) {
|
|
||||||
qemu_uname_release = argv[optind++];
|
|
||||||
} else if (!strcmp(r, "cpu")) {
|
|
||||||
cpu_model = argv[optind++];
|
|
||||||
if (cpu_model == NULL || strcmp(cpu_model, "?") == 0) {
|
|
||||||
/* XXX: implement xxx_cpu_list for targets that still miss it */
|
|
||||||
#if defined(cpu_list_id)
|
|
||||||
cpu_list_id(stdout, &fprintf, "");
|
|
||||||
#elif defined(cpu_list)
|
|
||||||
cpu_list(stdout, &fprintf); /* deprecated */
|
|
||||||
#endif
|
|
||||||
exit(1);
|
|
||||||
}
|
|
||||||
#if defined(CONFIG_USE_GUEST_BASE)
|
|
||||||
} else if (!strcmp(r, "B")) {
|
|
||||||
guest_base = strtol(argv[optind++], NULL, 0);
|
|
||||||
have_guest_base = 1;
|
|
||||||
} else if (!strcmp(r, "R")) {
|
|
||||||
char *p;
|
|
||||||
int shift = 0;
|
|
||||||
reserved_va = strtoul(argv[optind++], &p, 0);
|
|
||||||
switch (*p) {
|
|
||||||
case 'k':
|
|
||||||
case 'K':
|
|
||||||
shift = 10;
|
|
||||||
break;
|
|
||||||
case 'M':
|
|
||||||
shift = 20;
|
|
||||||
break;
|
|
||||||
case 'G':
|
|
||||||
shift = 30;
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
if (shift) {
|
|
||||||
unsigned long unshifted = reserved_va;
|
|
||||||
p++;
|
|
||||||
reserved_va <<= shift;
|
|
||||||
if (((reserved_va >> shift) != unshifted)
|
|
||||||
#if HOST_LONG_BITS > TARGET_VIRT_ADDR_SPACE_BITS
|
|
||||||
|| (reserved_va > (1ul << TARGET_VIRT_ADDR_SPACE_BITS))
|
|
||||||
#endif
|
|
||||||
) {
|
|
||||||
fprintf(stderr, "Reserved virtual address too big\n");
|
|
||||||
exit(1);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
if (*p) {
|
|
||||||
fprintf(stderr, "Unrecognised -R size suffix '%s'\n", p);
|
|
||||||
exit(1);
|
|
||||||
}
|
|
||||||
#endif
|
|
||||||
} else if (!strcmp(r, "drop-ld-preload")) {
|
|
||||||
(void) envlist_unsetenv(envlist, "LD_PRELOAD");
|
|
||||||
} else if (!strcmp(r, "singlestep")) {
|
|
||||||
singlestep = 1;
|
|
||||||
} else if (!strcmp(r, "strace")) {
|
|
||||||
do_strace = 1;
|
|
||||||
} else if (!strcmp(r, "version")) {
|
|
||||||
version();
|
|
||||||
exit(0);
|
|
||||||
} else {
|
|
||||||
usage();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
/* init debug */
|
/* init debug */
|
||||||
cpu_set_log_filename(log_file);
|
cpu_set_log_filename(log_file);
|
||||||
if (log_mask) {
|
optind = parse_args(argc, argv);
|
||||||
int mask;
|
|
||||||
const CPULogItem *item;
|
|
||||||
|
|
||||||
mask = cpu_str_to_log_mask(log_mask);
|
|
||||||
if (!mask) {
|
|
||||||
printf("Log items (comma separated):\n");
|
|
||||||
for (item = cpu_log_items; item->mask != 0; item++) {
|
|
||||||
printf("%-10s %s\n", item->name, item->help);
|
|
||||||
}
|
|
||||||
exit(1);
|
|
||||||
}
|
|
||||||
cpu_set_log(mask);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (optind >= argc) {
|
|
||||||
usage();
|
|
||||||
}
|
|
||||||
filename = argv[optind];
|
|
||||||
exec_path = argv[optind];
|
|
||||||
|
|
||||||
/* Zero out regs */
|
/* Zero out regs */
|
||||||
memset(regs, 0, sizeof(struct target_pt_regs));
|
memset(regs, 0, sizeof(struct target_pt_regs));
|
||||||
|
Loading…
Reference in New Issue
Block a user