package mars.assembler; import mars.*; import java.util.*; /* Copyright (c) 2003-2006, Pete Sanderson and Kenneth Vollmar Developed by Pete Sanderson (psanderson@otterbein.edu) and Kenneth Vollmar (kenvollmar@missouristate.edu) Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the "Software"), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the following conditions: The above copyright notice and this permission notice shall be included in all copies or substantial portions of the Software. THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. (MIT license, http://www.opensource.org/licenses/mit-license.html) */ /** * Creats a table of Symbol objects. * @author Jason Bumgarner, Jason Shrewsbury * @version June 2003 **/ public class SymbolTable { private static String startLabel = "main"; private String filename; private ArrayList table; // Note -1 is legal 32 bit address (0xFFFFFFFF) but it is the high address in // kernel address space so highly unlikely that any symbol will have this as // its associated address! public static final int NOT_FOUND = -1; /** * Create a new empty symbol table for given file * @param filename name of file this symbol table is associated with. Will be * used only for output/display so it can be any descriptive string. */ public SymbolTable(String filename) { this.filename = filename; this.table = new ArrayList(); } /** * Adds a Symbol object into the array of Symbols. * @param token The token representing the Symbol. * @param address The address of the Symbol. * @param b The type of Symbol, true for data, false for text. * @param errors List to which to add any processing errors that occur. **/ public void addSymbol(Token token, int address, boolean b, ErrorList errors) { String label = token.getValue(); if (getSymbol(label) != null) { errors.add(new ErrorMessage(token.getSourceMIPSprogram(), token.getSourceLine(),token.getStartPos(),"label \""+label+"\" already defined")); } else { Symbol s= new Symbol(label, address, b); table.add(s); if (Globals.debug) System.out.println("The symbol " + label + " with address " + address + " has been added to the "+this.filename+" symbol table."); } } /** * Removes a symbol from the Symbol table. If not found, it does nothing. * This will rarely happen (only when variable is declared .globl after already * being defined in the local symbol table). * @param token The token representing the Symbol. **/ public void removeSymbol(Token token) { String label = token.getValue(); for (int i=0; i < table.size(); i++) { if (((Symbol)(table.get(i))).getName().equals(label)){ table.remove(i); if (Globals.debug) System.out.println("The symbol " + label + " has been removed from the "+this.filename+" symbol table."); break; } } return; } /** * Method to return the address associated with the given label. * @param s The label. * @return The memory address of the label given, or NOT_FOUND if not found in symbol table. **/ public int getAddress(String s){ for(int i=0; i < table.size(); i++){ if (((Symbol)(table.get(i))).getName().equals(s)){ return((Symbol) table.get(i)).getAddress(); } } return NOT_FOUND; } /** * Method to return the address associated with the given label. Look first * in this (local) symbol table then in symbol table of labels declared * global (.globl directive). * @param s The label. * @return The memory address of the label given, or NOT_FOUND if not found in symbol table. **/ public int getAddressLocalOrGlobal(String s) { int address = this.getAddress(s); return (address==NOT_FOUND) ? Globals.symbolTable.getAddress(s) : address ; } /** * Produce Symbol object from symbol table that corresponds to given String. * @param s target String * @return Symbol object for requested target, null if not found in symbol table. **/ public Symbol getSymbol(String s){ for(int i=0; i < table.size(); i++){ if (((Symbol)(table.get(i))).getName().equals(s)){ return (Symbol) table.get(i); } } return null; } /** * Produce Symbol object from symbol table that has the given address. * @param s String representing address * @return Symbol object having requested address, null if address not found in symbol table. **/ public Symbol getSymbolGivenAddress(String s){ int address = 0; try { address = mars.util.Binary.stringToInt(s);// DPS 2-Aug-2010: was Integer.parseInt(s) but croaked on hex } catch (NumberFormatException e) { return null; } for(int i=0; i < table.size(); i++){ if (((Symbol)(table.get(i))).getAddress() == address){ return (Symbol) table.get(i); } } return null; } /** * Produce Symbol object from either local or global symbol table that has the * given address. * @param s String representing address * @return Symbol object having requested address, null if address not found in symbol table. **/ public Symbol getSymbolGivenAddressLocalOrGlobal(String s){ Symbol sym = this.getSymbolGivenAddress(s); return (sym==null) ? Globals.symbolTable.getSymbolGivenAddress(s) : sym ; } /** * For obtaining the Data Symbols. * @return An ArrayList of Symbol objects. **/ public ArrayList getDataSymbols(){ ArrayList list= new ArrayList(); for(int i=0; i