b556be1622
Replaced struct line_s and typedef line with linked_line_s and linked_line. Added make_set() and free_set(). Modified make_cache() and free_cache() to call make_set() and free_set().
239 lines
5.9 KiB
C
239 lines
5.9 KiB
C
/* asgoldsmith-jikomissar */
|
|
/**
|
|
@file csim.c
|
|
@author Adam Goldsmith
|
|
@author Jacob Komissar
|
|
@date 2016-04-25, 2016-04-26, 2016-04-27
|
|
|
|
@TODO ADD CORRECT NAME COMMENT AT TOP OF FILE
|
|
|
|
Exit statuses:
|
|
0 - success
|
|
1 - usage error
|
|
-1 - file error
|
|
|
|
*/
|
|
/* Headers */
|
|
#include <stdlib.h>
|
|
#include <stdio.h>
|
|
#include <getopt.h>
|
|
#include "cachelab.h"
|
|
|
|
/* Typedefs and structs */
|
|
typedef unsigned char uint8_t;
|
|
|
|
/* Linked Line and related prototypes */
|
|
typedef struct linked_line_s linked_line;
|
|
struct linked_line_s {
|
|
uint8_t validity;
|
|
long tag;
|
|
linked_line* newer;
|
|
linked_line* older;
|
|
};
|
|
|
|
|
|
/* Macros */
|
|
#define mprintf(...) if (VERBOSE) printf(__VA_ARGS__)
|
|
|
|
/* General Prototypes */
|
|
void print_usage(void);
|
|
void bad_usage_error(void);
|
|
long parse_int_arg(char *arg, char opt);
|
|
long bits_to_size(int bits);
|
|
linked_line** make_cache(long set_indices, long num_lines);
|
|
linked_line* make_set(linked_line* newer, long num_lines);
|
|
void free_cache(linked_line** cache, long num_sets);
|
|
void free_set(linked_line* line);
|
|
|
|
/* Global variables */
|
|
uint8_t VERBOSE = 0; /**< If nonzero, mprintf will not print. Set in main() if -v flag is given. */
|
|
|
|
|
|
int main(int argc, char* argv[])
|
|
{
|
|
int hits, misses, evictions;
|
|
char buffer[20];
|
|
int set_index_bits = 0, block_bits = 0; // Not-useful input variables.
|
|
long set_indices = 0, num_lines = 0, block_size = 0; //Useful variables
|
|
char *filename = NULL;
|
|
int opt = 0;
|
|
while ((opt=getopt(argc, argv, "hvs:E:b:t:")) != -1) {
|
|
switch (opt) {
|
|
/* TODO: maybe add checking for optarg swallowing actual arguments. */
|
|
case 'v':
|
|
VERBOSE = 1;
|
|
break;
|
|
case 's': // 2^s = number of sets
|
|
set_index_bits = (uint8_t)parse_int_arg(optarg, 's');
|
|
set_indices = bits_to_size(set_index_bits);
|
|
break;
|
|
case 'E': // associativity - lines per set
|
|
num_lines = parse_int_arg(optarg, 'E');
|
|
break;
|
|
case 'b': // 2^b = block size
|
|
block_bits = (uint8_t)parse_int_arg(optarg, 'b');
|
|
block_size = bits_to_size(block_bits);
|
|
break;
|
|
case 't':
|
|
filename = optarg;
|
|
break;
|
|
case 'h':
|
|
print_usage();
|
|
return 0;
|
|
default:
|
|
bad_usage_error();
|
|
}
|
|
}
|
|
/* If any required arguments were not provided. (argflags != 0b1111) */
|
|
if (!set_indices || !num_lines || !block_size || !filename) {
|
|
bad_usage_error();
|
|
}
|
|
/* End of argument parsing. */
|
|
//probably broken: printf("Arguments: s=%hhu, E=%u, b=%hhu, t=%s, v=%hhu, 2^s=%d, 2^b=%u\n", set_index_bits, num_lines, block_bits, filename, VERBOSE, set_indices, block_size);
|
|
|
|
linked_line** cache = make_cache(set_indices, num_lines);
|
|
|
|
/* FILE READING */
|
|
FILE* f = fopen(filename, "r");
|
|
if(!f)
|
|
{
|
|
printf("Invalid file name: %s", filename);
|
|
exit(-1);
|
|
}
|
|
|
|
//char buffer[20];
|
|
while (fgets(buffer, 20, f)) {
|
|
if (buffer[0] == 'I')
|
|
continue;
|
|
|
|
char* end;
|
|
char op = buffer[1];
|
|
long address = strtol(buffer+3, &end, 16);
|
|
if (*end != ',' || !(op == 'S' || op == 'L' || op == 'M')) {
|
|
printf("Invalid input file, last line:\n%s\n", buffer);
|
|
return -1;
|
|
}
|
|
printf("%c %ld\n", op, address);
|
|
}
|
|
|
|
printSummary(0, 0, 0);
|
|
|
|
fclose(f);
|
|
free_cache(cache, set_index_bits);
|
|
|
|
return 0;
|
|
}
|
|
|
|
|
|
|
|
|
|
linked_line** make_cache(long set_indices, long num_lines)
|
|
{
|
|
linked_line** cache = (linked_line**)calloc(set_indices, sizeof(linked_line*));
|
|
long ii;
|
|
for (ii = 0; ii < num_lines; ii++) {
|
|
cache[ii] = make_set(NULL, num_lines);
|
|
}
|
|
return cache;
|
|
}
|
|
|
|
/**
|
|
Creates a set by creating and linking a given number of lines.
|
|
@param newer A pointer to the previously-created element of the set.
|
|
@param num_lines The number of lines remaining to be added.
|
|
@return A pointer to the created line.
|
|
*/
|
|
linked_line* make_set(linked_line* newer, long num_lines) {
|
|
linked_line* current = (linked_line*)calloc(1, sizeof(linked_line));
|
|
current->newer = newer;
|
|
if (num_lines) { // If not on the last line
|
|
current->older = make_set(current, num_lines-1);
|
|
}
|
|
else {
|
|
current->older = NULL;
|
|
}
|
|
return current;
|
|
}
|
|
|
|
void free_cache(linked_line** cache, long num_sets)
|
|
{
|
|
int ii;
|
|
for (ii = 0; ii < num_sets; ii++) {
|
|
free_set(cache[ii]);
|
|
}
|
|
free(cache);
|
|
}
|
|
|
|
/**
|
|
Frees a set by recursing through it, then freeing each element from oldest to newest.
|
|
@param line A pointer to the next line in the set.
|
|
@param num_lines The number
|
|
*/
|
|
void free_set(linked_line* line) {
|
|
if (line != NULL) {
|
|
free_set(line->older);
|
|
free(line);
|
|
}
|
|
}
|
|
|
|
void print_usage(void)
|
|
{
|
|
printf("Usage: ./csim [-hv] -s <number> -E <number> -b <number> -t <file>\n"
|
|
"Options:\n"
|
|
" -h Print this help.\n"
|
|
" -v Display trace info.\n"
|
|
" -s <number> The number of set index bits. Must be a positive integer.\n"
|
|
" -E <number> Associativity (lines per set). Must be a positive integer.\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.");
|
|
}
|
|
|
|
/**
|
|
Prints the summarized usage and exits with status 1.
|
|
*/
|
|
void bad_usage_error(void)
|
|
{
|
|
printf("Usage: ./csim [-hv] -s <number> -E <number> -b <number> -t <file>\n");
|
|
exit(1);
|
|
}
|
|
|
|
/**
|
|
Parses a string into a size_t.
|
|
@param arg The string to parse.
|
|
@param opt The option being parsed. If 'E', the output can not be 0.
|
|
@return The parsed long integer if the argument was an integer.
|
|
@note Exits if argument is invalid.
|
|
*/
|
|
long parse_int_arg(char* arg, char opt)
|
|
{
|
|
char* end;
|
|
long i = strtol(arg, &end, 0);
|
|
|
|
if (*end != '\0' || (!i && opt=='E')) { // associativity can not be zero
|
|
printf("Invalid argument \"%s\" for option -%c\n", arg, opt);
|
|
bad_usage_error();
|
|
}
|
|
else if (i < 0) {
|
|
printf("Argument %ld for option %c too small (must be greater than 0)", i, opt);
|
|
exit(1);
|
|
}
|
|
return i;
|
|
}
|
|
|
|
/**
|
|
Calculates a size using a given number of bits.
|
|
@param bits The power of 2 to use.
|
|
@note Exits if number of bits is larger than a size_t.
|
|
*/
|
|
long bits_to_size(int bits)
|
|
{
|
|
int max_shift = 8*sizeof(int);
|
|
if (bits > max_shift-1) {
|
|
printf("Argument %hhd too large (-s and -b must be less than %hhd)", bits, max_shift);
|
|
exit(1);
|
|
}
|
|
return (size_t)1 << bits;
|
|
}
|
|
|
|
|