#include <debug.h>
#include <cmdline.h>
#include <stdio.h>
#include <stdlib.h>
#include <getopt.h>
#include <string.h>
#include <ctype.h>
extern char *optarg;
extern int optind, opterr, optopt;
static struct option long_options[] =
{
{"verbose", no_argument, 0, 'V'},
{"quiet", no_argument, 0, 'Q'},
{"shady", no_argument, 0, 'S'},
{"print", no_argument, 0, 'p'},
{"help", no_argument, 0, 'h'},
{"outfile", required_argument, 0, 'o'},
{"filter", required_argument, 0, 'f'},
{"dry", no_argument, 0, 'n'},
{"strip", no_argument, 0, 's'},
{0, 0, 0, 0},
};
/* This array must parallel long_options[] */
static
const char *descriptions[sizeof(long_options)/sizeof(long_options[0])] = {
"print verbose output",
"suppress errors and warnings",
"patch ABS symbols whose values coincide with section starts and ends",
"print the symbol table (if specified, only -V is allowed)",
"this help screen",
"specify an output file (if not provided, input file is modified)",
"specify a symbol-filter file",
"dry run (perform all calculations but do not modify the ELF file)",
"strip debug sections, if they are present"
};
void print_help(void)
{
fprintf(stdout,
"invokation:\n"
"\tsoslim file1 [file2 file3 ... fileN] [-Ldir1 -Ldir2 ... -LdirN] "
"[-Vpn]\n"
"or\n"
"\tsoslim -h\n\n");
fprintf(stdout, "options:\n");
struct option *opt = long_options;
const char **desc = descriptions;
while (opt->name) {
fprintf(stdout, "\t-%c/--%-15s %s\n",
opt->val,
opt->name,
*desc);
opt++;
desc++;
}
}
int get_options(int argc, char **argv,
char **outfile,
char **symsfile,
int *print_symtab,
int *verbose,
int *quiet,
int *shady,
int *dry_run,
int *strip_debug)
{
int c;
ASSERT(outfile);
*outfile = NULL;
ASSERT(symsfile);
*symsfile = NULL;
ASSERT(print_symtab);
*print_symtab = 0;
ASSERT(verbose);
*verbose = 0;
ASSERT(quiet);
*quiet = 0;
ASSERT(shady);
*shady = 0;
ASSERT(dry_run);
*dry_run = 0;
ASSERT(strip_debug);
*strip_debug = 0;
while (1) {
/* getopt_long stores the option index here. */
int option_index = 0;
c = getopt_long (argc, argv,
"QVSphi:o:y:Y:f:ns",
long_options,
&option_index);
/* Detect the end of the options. */
if (c == -1) break;
if (isgraph(c)) {
INFO ("option -%c with value `%s'\n", c, (optarg ?: "(null)"));
}
#define SET_STRING_OPTION(name) do { \
ASSERT(optarg); \
*name = strdup(optarg); \
} while(0)
switch (c) {
case 0:
/* If this option set a flag, do nothing else now. */
if (long_options[option_index].flag != 0)
break;
INFO ("option %s", long_options[option_index].name);
if (optarg)
INFO (" with arg %s", optarg);
INFO ("\n");
break;
case 'p': *print_symtab = 1; break;
case 'h': print_help(); exit(1); break;
case 'V': *verbose = 1; break;
case 'Q': *quiet = 1; break;
case 'S': *shady = 1; break;
case 'n': *dry_run = 1; break;
case 's': *strip_debug = 1; break;
case 'o': SET_STRING_OPTION(outfile); break;
case 'f': SET_STRING_OPTION(symsfile); break;
case '?':
/* getopt_long already printed an error message. */
break;
#undef SET_STRING_OPTION
default:
FAILIF(1, "Unknown option");
}
}
return optind;
}