New and improved argument parsing with getopt.
Added checking that all arguments were provided. Switched many variables to use size_t, and int8_t where appropriate. Moved opening braces on to previous lines, except function braces. Renamed set_bits to more-accurate index_bits.
This commit is contained in:
parent
0a660b8939
commit
dc67c24899
105
csim.c
105
csim.c
@ -1,8 +1,23 @@
|
|||||||
#include <stdlib.h>
|
#include <stdlib.h>
|
||||||
#include <stdio.h>
|
#include <stdio.h>
|
||||||
|
#include <getopt.h>
|
||||||
|
#include <stdint.h>
|
||||||
#include "cachelab.h"
|
#include "cachelab.h"
|
||||||
|
|
||||||
int VERBOSE = 0;
|
#define mprintf(...) if (VERBOSE) printf(__VA_ARGS__)
|
||||||
|
|
||||||
|
/**
|
||||||
|
If VERBOSE is non-zero, mprintf will not print. Otherwise, it will.
|
||||||
|
Set in main().
|
||||||
|
*/
|
||||||
|
int8_t VERBOSE = 0;
|
||||||
|
|
||||||
|
/* As defined in conference. */
|
||||||
|
struct line_s {
|
||||||
|
int8_t validity;
|
||||||
|
int tag;
|
||||||
|
};
|
||||||
|
typedef struct line_s line;
|
||||||
|
|
||||||
void print_usage()
|
void print_usage()
|
||||||
{
|
{
|
||||||
@ -10,58 +25,86 @@ void print_usage()
|
|||||||
"Options:\n"
|
"Options:\n"
|
||||||
" -h Print this help.\n"
|
" -h Print this help.\n"
|
||||||
" -v Display trace info.\n"
|
" -v Display trace info.\n"
|
||||||
" -s <number> The number of sets is the base-2 logarithm of this argument.\n"
|
" -s <number> The number of set index bits. Must be a positive integer.\n"
|
||||||
" -E <number> Associativity (lines per set).\n"
|
" -E <number> Associativity (lines per set). Must be a positive integer.\n"
|
||||||
" -b <number> Block size in bits is the base-2 logarithm of this argument.\n"
|
" -b <number> The number of block bits. Must be a positive integer.\n"
|
||||||
" -t <file> Name of the file to read a valgrind trace from.");
|
" -t <file> Name of the file to read a valgrind trace from.");
|
||||||
exit(0);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
int parse_int_arg(char* arg)
|
/* TODO: Move the exit call from parse_int_arg into the getopt cases. */
|
||||||
|
/**
|
||||||
|
Parses a string
|
||||||
|
@param arg The string to parse.
|
||||||
|
@param opt The option being parsed.
|
||||||
|
@return The parsed integer if the argument was an integer.
|
||||||
|
@note Exits if argument is invalid
|
||||||
|
*/
|
||||||
|
size_t parse_int_arg(char* arg, char opt)
|
||||||
{
|
{
|
||||||
char* end;
|
char* end;
|
||||||
int i = strtol(arg, &end, 0);
|
long i = strtol(arg, &end, 0);
|
||||||
if (!i || *end != '\0')
|
|
||||||
{
|
if (!i || *end != '\0') {
|
||||||
printf("Invalid argument \"%s\"\n", arg);
|
printf("Invalid argument \"%s\" for option -%c\n", arg, opt);
|
||||||
print_usage();
|
printf("Usage: ./csim [-hv] -s <number> -E <number> -b <number> -t <file>\n");
|
||||||
|
exit(1);
|
||||||
|
}
|
||||||
|
else if (i < 0) {
|
||||||
|
printf("Argument %ld for option %c too small (must be greater than 0)", i, opt);
|
||||||
|
exit(1);
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
return (size_t)i;
|
||||||
}
|
}
|
||||||
return i;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
int main(int argc, char* argv[])
|
int main(int argc, char* argv[])
|
||||||
{
|
{
|
||||||
int set_bits = 0, lines = 0, block_bits = 0;
|
size_t index_bits = 1, lines = ~0, block_bits = 1; // Useful variables.
|
||||||
char* filename;
|
char *filename;
|
||||||
int ii;
|
int opt = 0, argflags = 0;
|
||||||
for (ii = 0; ii < argc; ii++)
|
while ((opt=getopt(argc, argv, "hvs:E:b:t:")) != -1) {
|
||||||
{
|
switch (opt) {
|
||||||
switch (argv[ii][1])
|
/* TODO: maybe add checking for optarg swallowing actual arguments. */
|
||||||
{
|
|
||||||
case 'h':
|
|
||||||
//print_usage();
|
|
||||||
break;
|
|
||||||
case 'v':
|
case 'v':
|
||||||
VERBOSE = 1;
|
VERBOSE = 1;
|
||||||
break;
|
break;
|
||||||
case 's':
|
case 's': // 2^s = number of sets
|
||||||
set_bits = parse_int_arg(argv[ii+1]);
|
index_bits = parse_int_arg(optarg, 's');
|
||||||
|
argflags |= 8; // 0b1000
|
||||||
break;
|
break;
|
||||||
case 'E':
|
case 'E': // associativity - lines per set
|
||||||
lines = parse_int_arg(argv[ii+1]);
|
lines = parse_int_arg(optarg, 'E');
|
||||||
|
argflags |= 4; // 0b0100
|
||||||
break;
|
break;
|
||||||
case 'b':
|
case 'b': // 2^b = block size
|
||||||
block_bits = parse_int_arg(argv[ii+1]);
|
block_bits = parse_int_arg(optarg, 'b');
|
||||||
|
argflags |= 2; // 0b0010
|
||||||
break;
|
break;
|
||||||
case 't':
|
case 't':
|
||||||
filename = argv[ii+1];
|
filename = optarg;
|
||||||
|
argflags |= 1; // 0b0001
|
||||||
break;
|
break;
|
||||||
|
case 'h':
|
||||||
|
print_usage();
|
||||||
|
return 0;
|
||||||
|
default:
|
||||||
|
printf("Usage: ./csim [-hv] -s <number> -E <number> -b <number> -t <file>\n");
|
||||||
|
return 1;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
/* If any required arguments were not provided. (argflags != 0b1111) */
|
||||||
|
if (argflags != 15) {
|
||||||
|
printf("Usage: ./csim [-hv] -s <number> -E <number> -b <number> -t <file>\n");
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
/* End of argument parsing. */
|
||||||
|
|
||||||
printf("Arguments: %d, %d, %d, %d, %s\n", set_bits, lines, block_bits, VERBOSE, filename);
|
printf("Arguments: %zd, %zd, %zd, %hd, %s\n", index_bits, lines, block_bits, VERBOSE, filename);
|
||||||
if(!set_bits|| !lines || !block_bits || !filename)
|
if(!index_bits || !lines || !block_bits || !filename) {
|
||||||
print_usage();
|
print_usage();
|
||||||
|
}
|
||||||
|
|
||||||
printSummary(0, 0, 0);
|
printSummary(0, 0, 0);
|
||||||
return 0;
|
return 0;
|
||||||
|
Reference in New Issue
Block a user