ref: c21c0f458fdc51335ae6b2773e60e5e6e3423ed1
dir: /src/link/symbol.c/
#include <stdio.h> #include <stdlib.h> #include <string.h> #include "link/main.h" #include "link/patch.h" #include "link/types.h" #define HASHSIZE 73 struct ISymbol { char *pzName; SLONG nValue; SLONG nBank; //-1 = const struct ISymbol *pNext; }; struct ISymbol *tHash[HASHSIZE]; SLONG calchash(char *s) { SLONG r = 0; while (*s) r += *s++; return (r % HASHSIZE); } void sym_Init(void) { SLONG i; for (i = 0; i < HASHSIZE; i += 1) tHash[i] = NULL; } SLONG sym_GetValue(char *tzName) { if (strcmp(tzName, "@") == 0) { return (nPC); } else { struct ISymbol **ppSym; ppSym = &(tHash[calchash(tzName)]); while (*ppSym) { if (strcmp(tzName, (*ppSym)->pzName)) { ppSym = &((*ppSym)->pNext); } else { return ((*ppSym)->nValue); } } fprintf(stderr, "Unknown symbol '%s'\n", tzName); exit(1); } } SLONG sym_GetBank(char *tzName) { struct ISymbol **ppSym; ppSym = &(tHash[calchash(tzName)]); while (*ppSym) { if (strcmp(tzName, (*ppSym)->pzName)) { ppSym = &((*ppSym)->pNext); } else { return ((*ppSym)->nBank); } } fprintf(stderr, "Unknown symbol '%s'\n", tzName); exit(1); } void sym_CreateSymbol(char *tzName, SLONG nValue, SBYTE nBank) { if (strcmp(tzName, "@") == 0) return; struct ISymbol **ppSym; ppSym = &(tHash[calchash(tzName)]); while (*ppSym) { if (strcmp(tzName, (*ppSym)->pzName)) { ppSym = &((*ppSym)->pNext); } else { if (nBank == -1) return; fprintf(stderr, "Symbol '%s' defined more than once\n", tzName); exit(1); } } if ((*ppSym = malloc(sizeof **ppSym))) { if (((*ppSym)->pzName = malloc(strlen(tzName) + 1))) { strcpy((*ppSym)->pzName, tzName); (*ppSym)->nValue = nValue; (*ppSym)->nBank = nBank; (*ppSym)->pNext = NULL; } } }