My Project
|
Defines the interface to symbol.c functions (do not modify) More...
Go to the source code of this file.
Data Structures | |
struct | symbol |
Macros | |
#define | SYMBOL_SIZE 997 |
Typedefs | |
typedef struct sym_table | sym_table_t |
typedef struct symbol | symbol_t |
typedef void(* | iterate_fnc_t )(symbol_t *sym, void *data) |
Functions | |
sym_table_t * | symbol_init (int lookup_by_addr) |
void | symbol_term (sym_table_t *symTab) |
void | symbol_reset (sym_table_t *symTab) |
int | symbol_add (sym_table_t *symTab, const char *name, int addr) |
struct node * | symbol_search (sym_table_t *symTab, const char *name, int *hash, int *index) |
symbol_t * | symbol_find_by_name (sym_table_t *symTab, const char *name) |
char * | symbol_find_by_addr (sym_table_t *symTab, int addr) |
void | symbol_iterate (sym_table_t *symTab, iterate_fnc_t fnc, void *data) |
This file defines the interface to a C file symbol.c that you will complete. The underlying data structure(s) used will be defined by the actual assignment. The assignment will define whether symbols are case sensitive or case in-sensitive.
In this implementation, you will learn about dynamic memory management using malloc/free. You will also learn about function pointers (callback functions).
#define SYMBOL_SIZE 997 |
997 prime number, the size of hash table
typedef void(* iterate_fnc_t)(symbol_t *sym, void *data) |
Defines the signature of a callback function (also known as a function pointer). This is how languages such as Java and C++ do dynamic binding (i.e. figure out which function to call). Recall that in Java the code obj.equals(object)
will call one of possibly many different methods depending on the actual type of obj
. This is because the method .equals() may be overridden.
In the LC3, dynamic binding is based on the JSRR opcode. With this opcode, the address of the routine to call is stored in a register and can be changed at runtime. Compare this to a JSR nameOfRoutine opcode which specifies what routine to call from the label that follows it. Thus, the address is fixed at assembly time.
This is used in the symbol_iterate() function. An interesting variation would be to have the callback function return an integer which determines whether the iteration should contibue or terminate.
symTab | - pointer to the symbol table |
typedef struct sym_table sym_table_t |
This defines an opaque type. The actual contents of the structure are hidden in the implementation and only a pointer to this structure is used externally to this file. A pointer to an opaque structure is sometimes referred to as a handle.
The symbol_find methods return a pointer to this data structure. It is up to the implementor to decide how to use this stucture in the implementation.
int symbol_add | ( | sym_table_t * | symTab, |
const char * | name, | ||
int | addr | ||
) |
Add a symbol to the symbol table.
symTab | - pointer to the symbol table |
name | - the name of the symbol |
addr | - the address of the symbol |
addr_table
. The firstsymbol with a given address should be stored in the addr_table
. char* symbol_find_by_addr | ( | sym_table_t * | symTab, |
int | addr | ||
) |
Find a name by its LC3 address
symTab | - pointer to the symbol table |
addr | - an LC3 address |
symbol_t* symbol_find_by_name | ( | sym_table_t * | symTab, |
const char * | name | ||
) |
Find a symbol by its name
symTab | - pointer to the symbol table |
name | - the symbols name |
sym_table_t* symbol_init | ( | int | lookup_by_addr | ) |
Create a new symbol table and return a pointer to it.
lookup_by_addr | - if this value is non-zero, allocate and initialize an array of char * (addr_table) that is indexed by an LC3 address to find the label (if any) associated with that address. |
void symbol_iterate | ( | sym_table_t * | symTab, |
iterate_fnc_t | fnc, | ||
void * | data | ||
) |
This function calls the function for every entry in the symbol table. The assigment will define the order in which the entries should be visited.
symTab | - pointer to the symbol table |
fnc | - the function to be called on every element |
data | - any additional information to be passed on to fnc. The called function will cast this to whatever type was actually passed. |
void symbol_reset | ( | sym_table_t * | symTab | ) |
Remove all the symbols from the symbol table. After this call the opaque symbol table pointer is still valid and new symbols may be added to it. If needed, clear the addr_table
.
symTab | - pointer to the symbol table |
struct node* symbol_search | ( | sym_table_t * | symTab, |
const char * | name, | ||
int * | hash, | ||
int * | index | ||
) |
This function is only used internally and should be declared static. It is a useful support function for the add()/find()
functions. It is declared here for documentation purposes.
symTab | - pointer to the symbol table |
name | - the name of the symbol |
hash | - pointer to location where hash value will be stored |
index | - pointer to location where index will be stored |
void symbol_term | ( | sym_table_t * | symTab | ) |
Remove all symbols from the symbol table, and free all allocated memory. There must not be any memory leaks. After executing this function, the opaque pointer to the symbol table is no longer valid.