URL
https://opencores.org/ocsvn/openrisc/openrisc/trunk
Subversion Repositories openrisc
[/] [openrisc/] [trunk/] [orpsocv2/] [bench/] [sysc/] [src/] [MemoryLoad.cpp] - Rev 396
Go to most recent revision | Compare with Previous | Blame | View Log
/* MemoryLoad.cpp -- Program load functions Copyright (C) 1999 Damjan Lampret, lampret@opencores.org Copyright (C) 2008 Embecosm Limited Copyright (C) 2009 Julius Baxter, julius@orsoc.se Contributor Jeremy Bennett <jeremy.bennett@embecosm.com> This file is part of Or1ksim, the OpenRISC 1000 Architectural Simulator. This program is free software; you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation; either version 3 of the License, or (at your option) any later version. This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details. You should have received a copy of the GNU General Public License along with this program. If not, see <http://www.gnu.org/licenses/>. */ /* This program is commented throughout in a fashion suitable for processing with Doxygen. */ /* System includes */ #include <stdio.h> #include <stdlib.h> #include <string.h> #include "MemoryLoad.h" #include "OrpsocMain.h" //! Constructor for the ORPSoC memory loader class //! Initializes various values //! @param[in] orpsoc The SystemC Verilated ORPSoC instance //! @param[in] accessor Accessor class for this Verilated ORPSoC model MemoryLoad::MemoryLoad ( OrpsocAccess *_accessor ) { accessor = _accessor; } // MemoryAccess () /*---------------------------------------------------------------------------*/ /*!Copy a string with null termination This function is very similar to strncpy, except it null terminates the string. A global function also used by the CUC. @param[in] dst The destination string @param[in] src The source string @param[in] n Number of chars to copy EXCLUDING the null terminator (i.e. dst had better have room for n+1 chars) @return A pointer to dst */ /*---------------------------------------------------------------------------*/ char * MemoryLoad::strstrip (char *dst, const char *src, int n) { strncpy (dst, src, n); *(dst + n) = '\0'; return dst; } /* strstrip () */ /*---------------------------------------------------------------------------*/ /*!Translate logical to physical addresses for the loader Used only by the simulator loader to translate logical addresses into physical. If loadcode() is called with valid @c virtphy_transl pointer to a table of translations then translate() performs translation otherwise physical address is equal to logical. Currently NOT used @param[in] laddr Logical address @param[in] breakpoint Unused @return The physical address */ /*---------------------------------------------------------------------------*/ oraddr_t MemoryLoad::translate (oraddr_t laddr, int *breakpoint) { /* int i; // No translation (i.e. when loading kernel into simulator) if (transl_table == 0) { return laddr; } // Try to find our translation in the table. for (i = 0; i < (MEMORY_LEN / PAGE_SIZE) * 16; i += 16) { if ((laddr & ~(PAGE_SIZE - 1)) == eval_direct32 (transl_table + i, 0, 0)) { // Page modified set_direct32 (transl_table + i + 8, -2, 0, 0); PRINTF ("found paddr=%" PRIx32 "\n", eval_direct32 (transl_table + i + 4, 0, 0) | (laddr & (PAGE_SIZE - 1))); return (oraddr_t) eval_direct32 (transl_table + i + 4, 0, 0) | (laddr & (oraddr_t) (PAGE_SIZE - 1)); } } // Allocate new phy page for us. for (i = 0; i < (MEMORY_LEN / PAGE_SIZE) * 16; i += 16) { if (eval_direct32 (transl_table + i + 8, 0, 0) == 0) { // VPN set_direct32 (transl_table + i, laddr & ~(PAGE_SIZE - 1), 0, 0); // PPN set_direct32 (transl_table + i + 4, (i / 16) * PAGE_SIZE, 0, 0); // Page modified //set_direct32 (transl_table + i + 8, -2, 0, 0); PRINTF ("newly allocated ppn=%" PRIx32 "\n", eval_direct32 (transl_table + i + 4, 0, 0)); PRINTF ("newly allocated .ppn=%" PRIxADDR "\n", transl_table + i + 4); PRINTF ("newly allocated ofs=%" PRIxADDR "\n", (laddr & (PAGE_SIZE - 1))); PRINTF ("newly allocated paddr=%" PRIx32 "\n", eval_direct32 (transl_table + i + 4, 0, 0) | (laddr & (PAGE_SIZE - 1))); return (oraddr_t) eval_direct32 (transl_table + i + 4, 0, 0) | (laddr & (oraddr_t) (PAGE_SIZE - 1)); } } // If we come this far then all phy memory is used and we can't find our page nor allocate new page. transl_error = 1; PRINTF ("can't translate %" PRIxADDR "\n", laddr); exit (1); return -1; */ return 0; } /* translate() */ #if IMM_STATS int MemoryLoad::bits (uint32_t val) { int i = 1; if (!val) return 0; while (val != 0 && (int32_t) val != -1) { i++; val = (int32_t) val >> 1; } return i; } void MemoryLoad::check_insn (uint32_t insn) { int insn_index = insn_decode (insn); struct insn_op_struct *opd = op_start[insn_index]; uint32_t data = 0; int dis = 0; const char *name; if (!insn || insn_index < 0) return; name = insn_name (insn_index); if (strcmp (name, "l.nop") == 0 || strcmp (name, "l.sys") == 0) return; while (1) { uint32_t tmp = 0 unsigned int nbits = 0; while (1) { tmp |= ((insn >> (opd->type & OPTYPE_SHR)) & ((1 << opd->data) - 1)) << nbits; nbits += opd->data; if (opd->type & OPTYPE_OP) break; opd++; } /* Do we have to sign extend? */ if (opd->type & OPTYPE_SIG) { int sbit = (opd->type & OPTYPE_SBIT) >> OPTYPE_SBIT_SHR; if (tmp & (1 << sbit)) tmp |= 0xFFFFFFFF << sbit; } if (opd->type & OPTYPE_DIS) { /* We have to read register later. */ data += tmp; dis = 1; } else { if (!(opd->type & OPTYPE_REG) || dis) { if (!dis) data = tmp; if (strcmp (name, "l.movhi") == 0) { movhi = data << 16; } else { data |= movhi; //PRINTF ("%08x %s\n", data, name); if (!(or32_opcodes[insn_index].flags & OR32_IF_DELAY)) { bcnt[bits (data)][0]++; bsum[0]++; } else { if (strcmp (name, "l.bf") == 0 || strcmp (name, "l.bnf") == 0) { bcnt[bits (data)][1]++; bsum[1]++; } else { bcnt[bits (data)][2]++; bsum[2]++; } } } } data = 0; dis = 0; } if (opd->type & OPTYPE_LAST) { return; } opd++; } } #endif /*---------------------------------------------------------------------------*/ /*!Add an instruction to the program @note insn must be in big endian format @param[in] address The address to use @param[in] insn The instruction to add @param[in] breakpoint Not used (it is passed to the translate() function, which also does not use it. */ /*---------------------------------------------------------------------------*/ void MemoryLoad::addprogram (oraddr_t address, uint32_t insn, int *breakpoint) { int vaddr = (int) address; /* Use the whole-word write */ accessor->set_mem32(vaddr, insn); PRINTF("* addprogram: addr 0x%.8x insn: 0x%.8x (conf: 0x%.8x)\n", vaddr, insn, accessor->get_mem32(vaddr)); freemem += 4; } /* addprogram () */ /*---------------------------------------------------------------------------*/ /*!Load big-endian COFF file @param[in] filename File to load @param[in] sections Number of sections in file */ /*---------------------------------------------------------------------------*/ void MemoryLoad::readfile_coff (char *filename, short sections) { FILE *inputfs; char inputbuf[4]; uint32_t insn; int32_t sectsize; COFF_AOUTHDR coffaouthdr; struct COFF_scnhdr coffscnhdr; int len; int firstthree = 0; int breakpoint = 0; if (!(inputfs = fopen (filename, "r"))) { perror ("readfile_coff"); exit (1); } if (fseek (inputfs, sizeof (COFF_FILHDR), SEEK_SET) == -1) { fclose (inputfs); perror ("readfile_coff"); exit (1); } if (fread (&coffaouthdr, sizeof (coffaouthdr), 1, inputfs) != 1) { fclose (inputfs); perror ("readfile_coff"); exit (1); } while (sections--) { uint32_t scnhdr_pos = sizeof (COFF_FILHDR) + sizeof (coffaouthdr) + sizeof (struct COFF_scnhdr) * firstthree; if (fseek (inputfs, scnhdr_pos, SEEK_SET) == -1) { fclose (inputfs); perror ("readfile_coff"); exit (1); } if (fread (&coffscnhdr, sizeof (struct COFF_scnhdr), 1, inputfs) != 1) { fclose (inputfs); perror ("readfile_coff"); exit (1); } PRINTF ("Section: %s,", coffscnhdr.s_name); PRINTF (" paddr: 0x%.8lx,", COFF_LONG_H (coffscnhdr.s_paddr)); PRINTF (" vaddr: 0x%.8lx,", COFF_LONG_H (coffscnhdr.s_vaddr)); PRINTF (" size: 0x%.8lx,", COFF_LONG_H (coffscnhdr.s_size)); PRINTF (" scnptr: 0x%.8lx\n", COFF_LONG_H (coffscnhdr.s_scnptr)); sectsize = COFF_LONG_H (coffscnhdr.s_size); ++firstthree; /* loading section */ freemem = COFF_LONG_H (coffscnhdr.s_paddr); if (fseek (inputfs, COFF_LONG_H (coffscnhdr.s_scnptr), SEEK_SET) == -1) { fclose (inputfs); perror ("readfile_coff"); exit (1); } while (sectsize > 0 && (len = fread (&inputbuf, sizeof (inputbuf), 1, inputfs))) { insn = COFF_LONG_H (inputbuf); //len = insn_len (insn_decode (insn)); len = 4; if (len == 2) { fseek (inputfs, -2, SEEK_CUR); } addprogram (freemem, insn, &breakpoint); sectsize -= len; } } if (firstthree < 3) { PRINTF ("One or more missing sections. At least"); PRINTF (" three sections expected (.text, .data, .bss).\n"); exit (1); } if (firstthree > 3) { PRINTF ("Warning: one or more extra sections. These"); PRINTF (" sections were handled as .data sections.\n"); } fclose (inputfs); PRINTF ("Finished loading COFF.\n"); return; } /* readfile_coff () */ /*---------------------------------------------------------------------------*/ /*!Load symbols from big-endian COFF file @param[in] filename File to load @param[in] symptr Symbol pointer value @param[in] syms Symbols value */ /*---------------------------------------------------------------------------*/ void MemoryLoad::readsyms_coff (char *filename, uint32_t symptr, uint32_t syms) { FILE *inputfs; struct COFF_syment coffsymhdr; int count = 0; uint32_t nsyms = syms; if (!(inputfs = fopen (filename, "r"))) { perror ("readsyms_coff"); exit (1); } if (fseek (inputfs, symptr, SEEK_SET) == -1) { fclose (inputfs); perror ("readsyms_coff"); exit (1); } while (syms--) { int i, n; if (fread (&coffsymhdr, COFF_SYMESZ, 1, inputfs) != 1) { fclose (inputfs); perror ("readsyms_coff"); exit (1); } n = (unsigned char) coffsymhdr.e_numaux[0]; /* check whether this symbol belongs to a section and is external symbol; ignore all others */ if (COFF_SHORT_H (coffsymhdr.e_scnum) >= 0 && coffsymhdr.e_sclass[0] == C_EXT) { if (*((uint32_t *) coffsymhdr.e.e.e_zeroes)) { if (strlen (coffsymhdr.e.e_name) && strlen (coffsymhdr.e.e_name) < 9) add_label (COFF_LONG_H (coffsymhdr.e_value), coffsymhdr.e.e_name); } else { uint32_t fpos = ftell (inputfs); if (fseek (inputfs, symptr + nsyms * COFF_SYMESZ + COFF_LONG_H (coffsymhdr.e.e.e_offset), SEEK_SET) == 0) { char tmp[33], *s = &tmp[0]; while (s != &tmp[32]) if ((*(s++) = fgetc (inputfs)) == 0) break; tmp[32] = 0; add_label (COFF_LONG_H (coffsymhdr.e_value), &tmp[0]); } fseek (inputfs, fpos, SEEK_SET); } } for (i = 0; i < n; i++) if (fread (&coffsymhdr, COFF_SYMESZ, 1, inputfs) != 1) { fclose (inputfs); perror ("readsyms_coff3"); exit (1); } syms -= n; count += n; } fclose (inputfs); PRINTF ("Finished loading symbols.\n"); return; } /*---------------------------------------------------------------------------*/ /*!Read an ELF file @param[in] filename File to load */ /*---------------------------------------------------------------------------*/ void MemoryLoad::readfile_elf (char *filename) { FILE *inputfs; struct elf32_hdr elfhdr; struct elf32_phdr *elf_phdata = NULL; struct elf32_shdr *elf_spnt, *elf_shdata; struct elf32_sym *sym_tbl = (struct elf32_sym *) 0; uint32_t syms = 0; char *str_tbl = (char *) 0; char *s_str = (char *) 0; int breakpoint = 0; uint32_t inputbuf; uint32_t padd; uint32_t insn; int i, j, sectsize, len; if (!(inputfs = fopen (filename, "r"))) { perror ("readfile_elf"); exit (1); } if (fread (&elfhdr, sizeof (elfhdr), 1, inputfs) != 1) { perror ("readfile_elf"); exit (1); } if ((elf_shdata = (struct elf32_shdr *) malloc (ELF_SHORT_H (elfhdr.e_shentsize) * ELF_SHORT_H (elfhdr.e_shnum))) == NULL) { perror ("readfile_elf"); exit (1); } if (fseek (inputfs, ELF_LONG_H (elfhdr.e_shoff), SEEK_SET) != 0) { perror ("readfile_elf"); exit (1); } if (fread (elf_shdata, ELF_SHORT_H (elfhdr.e_shentsize) * ELF_SHORT_H (elfhdr.e_shnum), 1, inputfs) != 1) { perror ("readfile_elf"); exit (1); } if (ELF_LONG_H (elfhdr.e_phoff)) { if ((elf_phdata = (struct elf32_phdr *) malloc (ELF_SHORT_H (elfhdr.e_phnum) * ELF_SHORT_H (elfhdr.e_phentsize))) == NULL) { perror ("readfile_elf"); exit (1); } if (fseek (inputfs, ELF_LONG_H (elfhdr.e_phoff), SEEK_SET) != 0) { perror ("readfile_elf"); exit (1); } if (fread (elf_phdata, ELF_SHORT_H (elfhdr.e_phnum) * ELF_SHORT_H (elfhdr.e_phentsize), 1, inputfs) != 1) { perror ("readfile_elf"); exit (1); } } for (i = 0, elf_spnt = elf_shdata; i < ELF_SHORT_H (elfhdr.e_shnum); i++, elf_spnt++) { if (ELF_LONG_H (elf_spnt->sh_type) == SHT_STRTAB) { if (NULL != str_tbl) { free (str_tbl); } if ((str_tbl = (char *) malloc (ELF_LONG_H (elf_spnt->sh_size))) == NULL) { perror ("readfile_elf"); exit (1); } if (fseek (inputfs, ELF_LONG_H (elf_spnt->sh_offset), SEEK_SET) != 0) { perror ("readfile_elf"); exit (1); } if (fread (str_tbl, ELF_LONG_H (elf_spnt->sh_size), 1, inputfs) != 1) { perror ("readfile_elf"); exit (1); } } else if (ELF_LONG_H (elf_spnt->sh_type) == SHT_SYMTAB) { if (NULL != sym_tbl) { free (sym_tbl); } if ((sym_tbl = (struct elf32_sym *) malloc (ELF_LONG_H (elf_spnt->sh_size))) == NULL) { perror ("readfile_elf"); exit (1); } if (fseek (inputfs, ELF_LONG_H (elf_spnt->sh_offset), SEEK_SET) != 0) { perror ("readfile_elf"); exit (1); } if (fread (sym_tbl, ELF_LONG_H (elf_spnt->sh_size), 1, inputfs) != 1) { perror ("readfile_elf"); exit (1); } syms = ELF_LONG_H (elf_spnt->sh_size) / ELF_LONG_H (elf_spnt->sh_entsize); } } if (ELF_SHORT_H (elfhdr.e_shstrndx) != SHN_UNDEF) { elf_spnt = &elf_shdata[ELF_SHORT_H (elfhdr.e_shstrndx)]; if ((s_str = (char *) malloc (ELF_LONG_H (elf_spnt->sh_size))) == NULL) { perror ("readfile_elf"); exit (1); } if (fseek (inputfs, ELF_LONG_H (elf_spnt->sh_offset), SEEK_SET) != 0) { perror ("readfile_elf"); exit (1); } if (fread (s_str, ELF_LONG_H (elf_spnt->sh_size), 1, inputfs) != 1) { perror ("readfile_elf"); exit (1); } } for (i = 0, elf_spnt = elf_shdata; i < ELF_SHORT_H (elfhdr.e_shnum); i++, elf_spnt++) { if ((ELF_LONG_H (elf_spnt->sh_type) & SHT_PROGBITS) && (ELF_LONG_H (elf_spnt->sh_flags) & SHF_ALLOC)) { padd = ELF_LONG_H (elf_spnt->sh_addr); for (j = 0; j < ELF_SHORT_H (elfhdr.e_phnum); j++) { if (ELF_LONG_H (elf_phdata[j].p_offset) && ELF_LONG_H (elf_phdata[j].p_offset) <= ELF_LONG_H (elf_spnt->sh_offset) && (ELF_LONG_H (elf_phdata[j].p_offset) + ELF_LONG_H (elf_phdata[j].p_memsz)) > ELF_LONG_H (elf_spnt->sh_offset)) padd = ELF_LONG_H (elf_phdata[j].p_paddr) + ELF_LONG_H (elf_spnt->sh_offset) - ELF_LONG_H (elf_phdata[j].p_offset); } if (ELF_LONG_H (elf_spnt->sh_name) && s_str) //PRINTF ("Section: %s,", &s_str[ELF_LONG_H (elf_spnt->sh_name)]); printf("* Section: %s,", &s_str[ELF_LONG_H (elf_spnt->sh_name)]); else //PRINTF ("Section: noname,"); printf ("* Section: noname,"); printf ("* vaddr: 0x%.8lx,", ELF_LONG_H (elf_spnt->sh_addr)); printf ("* paddr: 0x%" PRIx32, padd); printf ("* offset: 0x%.8lx,", ELF_LONG_H (elf_spnt->sh_offset)); printf ("* size: 0x%.8lx\n", ELF_LONG_H (elf_spnt->sh_size)); freemem = padd; sectsize = ELF_LONG_H (elf_spnt->sh_size); if (fseek (inputfs, ELF_LONG_H (elf_spnt->sh_offset), SEEK_SET) != 0) { perror ("readfile_elf"); free (elf_phdata); exit (1); } while (sectsize > 0 && (len = fread (&inputbuf, sizeof (inputbuf), 1, inputfs))) { insn = ELF_LONG_H (inputbuf); //PRINTF("* addprogram(%.8x, %.8x, %d)\n", freemem, insn, breakpoint); addprogram (freemem, insn, &breakpoint); sectsize -= 4; } } } if (str_tbl) { i = 0; while (syms--) { if (sym_tbl[i].st_name && sym_tbl[i].st_info && ELF_SHORT_H (sym_tbl[i].st_shndx) < 0x8000) { add_label (ELF_LONG_H (sym_tbl[i].st_value), &str_tbl[ELF_LONG_H (sym_tbl[i].st_name)]); } i++; } } if (NULL != str_tbl) { free (str_tbl); } if (NULL != sym_tbl) { free (sym_tbl); } free (s_str); free (elf_phdata); free (elf_shdata); } /* Identify file type and call appropriate readfile_X routine. It only handles orX-coff-big executables at the moment. */ void MemoryLoad::identifyfile (char *filename) { FILE *inputfs; COFF_FILHDR coffhdr; struct elf32_hdr elfhdr; if (!(inputfs = fopen (filename, "r"))) { perror (filename); fflush (stdout); fflush (stderr); exit (1); } if (fread (&coffhdr, sizeof (coffhdr), 1, inputfs) == 1) { if (COFF_SHORT_H (coffhdr.f_magic) == 0x17a) { uint32_t opthdr_size; PRINTF ("COFF magic: 0x%.4x\n", COFF_SHORT_H (coffhdr.f_magic)); PRINTF ("COFF flags: 0x%.4x\n", COFF_SHORT_H (coffhdr.f_flags)); PRINTF ("COFF symptr: 0x%.8lx\n", COFF_LONG_H (coffhdr.f_symptr)); if ((COFF_SHORT_H (coffhdr.f_flags) & COFF_F_EXEC) != COFF_F_EXEC) { PRINTF ("This COFF is not an executable.\n"); exit (1); } opthdr_size = COFF_SHORT_H (coffhdr.f_opthdr); if (opthdr_size != sizeof (COFF_AOUTHDR)) { PRINTF ("COFF optional header is missing or not recognized.\n"); PRINTF ("COFF f_opthdr: 0x%" PRIx32 "\n", opthdr_size); exit (1); } fclose (inputfs); readfile_coff (filename, COFF_SHORT_H (coffhdr.f_nscns)); readsyms_coff (filename, COFF_LONG_H (coffhdr.f_symptr), COFF_LONG_H (coffhdr.f_nsyms)); return; } else { PRINTF ("Not COFF file format\n"); fseek (inputfs, 0, SEEK_SET); } } if (fread (&elfhdr, sizeof (elfhdr), 1, inputfs) == 1) { if (elfhdr.e_ident[0] == 0x7f && elfhdr.e_ident[1] == 0x45 && elfhdr.e_ident[2] == 0x4c && elfhdr.e_ident[3] == 0x46) { PRINTF ("ELF type: 0x%.4x\n", ELF_SHORT_H (elfhdr.e_type)); PRINTF ("ELF machine: 0x%.4x\n", ELF_SHORT_H (elfhdr.e_machine)); PRINTF ("ELF version: 0x%.8lx\n", ELF_LONG_H (elfhdr.e_version)); PRINTF ("ELF sec = %d\n", ELF_SHORT_H (elfhdr.e_shnum)); if (ELF_SHORT_H (elfhdr.e_type) != ET_EXEC) { PRINTF ("This ELF is not an executable.\n"); exit (1); } fclose (inputfs); readfile_elf (filename); return; } else { PRINTF ("Not ELF file format.\n"); fseek (inputfs, 0, SEEK_SET); } } perror ("identifyfile2"); fclose (inputfs); return; } /*---------------------------------------------------------------------------*/ /*!Load file to memory Loads file to memory starting at address startaddr and returns freemem. @param[in] filename File to load @param[in] startaddr Start address at which to load @param[in] virtphy_transl Virtual to physical transation table if required. Only used for microkernel simulation, and not used in Ork1sim at present (set to NULL) @return zero on success, negative on failure. */ /*---------------------------------------------------------------------------*/ uint32_t MemoryLoad::loadcode (char *filename, oraddr_t startaddr, oraddr_t virtphy_transl) { //int breakpoint = 0; init_labels (); // jb transl_error = 0; transl_table = virtphy_transl; freemem = startaddr; PRINTF ("* MemoryLoad::loadcode: filename %s startaddr=%" PRIxADDR " virtphy_transl=%" PRIxADDR "\n", filename, startaddr, virtphy_transl); identifyfile (filename); #if IMM_STATS { int i = 0, a = 0, b = 0, c = 0; PRINTF ("index:arith/branch/jump\n"); for (i = 0; i < 33; i++) PRINTF ("%2i:\t%3.0f%% / %3.0f%%/ %3.0f%%\t%5i / %5i / %5i\n", i, 100. * (a += bcnt[i][0]) / bsum[0], 100. * (b += bcnt[i][1]) / bsum[1], 100. * (c += bcnt[i][2]) / bsum[2], bcnt[i][0], bcnt[i][1], bcnt[i][2]); PRINTF ("\nsum %i %i %i\n", bsum[0], bsum[1], bsum[2]); } #endif /* if (transl_error) return -1; else return translate (freemem, &breakpoint); */ return (uint32_t) freemem; } /* From arch sim labels.c */ void MemoryLoad::init_labels () { int i; for (i = 0; i < LABELS_HASH_SIZE; i++) label_hash[i] = NULL; } void MemoryLoad::add_label (oraddr_t addr, char *name) { struct label_entry **tmp; tmp = &(label_hash[addr % LABELS_HASH_SIZE]); for (; *tmp; tmp = &((*tmp)->next)); *tmp = (label_entry *) malloc (sizeof (**tmp)); (*tmp)->name = (char *) malloc (strlen (name) + 1); (*tmp)->addr = addr; strcpy ((*tmp)->name, name); (*tmp)->next = NULL; } struct label_entry * MemoryLoad::get_label (oraddr_t addr) { struct label_entry *tmp = label_hash[addr % LABELS_HASH_SIZE]; while (tmp) { if (tmp->addr == addr) return tmp; tmp = tmp->next; } return NULL; } struct label_entry * MemoryLoad::find_label (char *name) { int i; for (i = 0; i < LABELS_HASH_SIZE; i++) { struct label_entry *tmp = label_hash[i % LABELS_HASH_SIZE]; while (tmp) { if (strcmp (tmp->name, name) == 0) return tmp; tmp = tmp->next; } } return NULL; } /* Searches mem array for a particular label and returns label's address. If label does not exist, returns 0. */ oraddr_t MemoryLoad::eval_label (char *name) { struct label_entry *le; char *plus; char *minus; int positive_offset = 0; int negative_offset = 0; if ((plus = strchr (name, '+'))) { *plus = '\0'; positive_offset = atoi (++plus); } if ((minus = strchr (name, '-'))) { *minus = '\0'; negative_offset = atoi (++minus); } le = find_label (name); if (!le) return 0; return le->addr + positive_offset - negative_offset; }
Go to most recent revision | Compare with Previous | Blame | View Log