Files
EMARS/src/main/java/mars/mips/hardware/Coprocessor0.java
T
2022-11-09 23:49:19 -05:00

258 lines
7.6 KiB
Java

package mars.mips.hardware;
import mars.Globals;
import java.util.Observer;
/*
Copyright (c) 2003-2009, 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)
*/
/**
* Represents Coprocessor 0. We will use only its interrupt/exception registers.
*
* @author Pete Sanderson
* @version August 2005
**/
public class Coprocessor0
{
/**
* Coprocessor register names
*/
public static final int VADDR = 8;
public static final int STATUS = 12;
public static final int CAUSE = 13;
public static final int EPC = 14;
public static final int EXCEPTION_LEVEL = 1; // bit position in STATUS register
// bits 8-15 (mask for interrupt levels) all set, bit 4 (user mode) set,
// bit 1 (exception level) not set, bit 0 (interrupt enable) set.
public static final int DEFAULT_STATUS_VALUE = 0x0000FF11;
private static final Register[] registers =
{new Register("$8 (vaddr)", 8, 0),
new Register("$12 (status)", 12, DEFAULT_STATUS_VALUE),
new Register("$13 (cause)", 13, 0),
new Register("$14 (epc)", 14, 0)
};
/**
* Method for displaying the register values for debugging.
**/
public static void showRegisters()
{
for (int i = 0; i < registers.length; i++)
{
System.out.println("Name: " + registers[i].getName());
System.out.println("Number: " + registers[i].getNumber());
System.out.println("Value: " + registers[i].getValue());
System.out.println();
}
}
/**
* Sets the value of the register given to the value given.
*
* @param n name of register to set the value of ($n, where n is reg number).
* @param val The desired value for the register.
* @return old value in register prior to update
**/
public static int updateRegister(String n, int val)
{
int oldValue = 0;
for (int i = 0; i < registers.length; i++)
{
if (("$" + registers[i].getNumber()).equals(n) || registers[i].getName().equals(n))
{
oldValue = registers[i].getValue();
registers[i].setValue(val);
break;
}
}
return oldValue;
}
/**
* This method updates the register value who's number is num.
*
* @param num Number of register to set the value of.
* @param val The desired value for the register.
* @return old value in register prior to update
**/
public static int updateRegister(int num, int val)
{
int old = 0;
for (int i = 0; i < registers.length; i++)
{
if (registers[i].getNumber() == num)
{
old = (Globals.getSettings().getBackSteppingEnabled())
? Globals.program.getBackStepper().addCoprocessor0Restore(num, registers[i].setValue(val))
: registers[i].setValue(val);
break;
}
}
return old;
}
/**
* Returns the value of the register who's number is num.
*
* @param num The register number.
* @return The value of the given register. 0 for non-implemented registers
**/
public static int getValue(int num)
{
for (int i = 0; i < registers.length; i++)
{
if (registers[i].getNumber() == num)
{
return registers[i].getValue();
}
}
return 0;
}
/**
* For getting the number representation of the register.
*
* @param n The string formatted register name to look for.
* @return The number of the register represented by the string. -1 if no match.
**/
public static int getNumber(String n)
{
for (int i = 0; i < registers.length; i++)
{
if (("$" + registers[i].getNumber()).equals(n) || registers[i].getName().equals(n))
{
return registers[i].getNumber();
}
}
return -1;
}
/**
* For returning the set of registers.
*
* @return The set of registers.
**/
public static Register[] getRegisters()
{
return registers;
}
/**
* Coprocessor0 implements only selected registers, so the register number (8, 12, 13, 14) does not correspond to
* its position in the list of registers (0, 1, 2, 3).
*
* @param r A coprocessor0 Register
* @return the list position of given register, -1 if not found.
**/
public static int getRegisterPosition(Register r)
{
for (int i = 0; i < registers.length; i++)
{
if (registers[i] == r)
{
return i;
}
}
return -1;
}
/**
* Get register object corresponding to given name. If no match, return null.
*
* @param rname The register name, in $0 format.
* @return The register object,or null if not found.
**/
public static Register getRegister(String rname)
{
for (int i = 0; i < registers.length; i++)
{
if (("$" + registers[i].getNumber()).equals(rname) || registers[i].getName().equals(rname))
{
return registers[i];
}
}
return null;
}
/**
* Method to reinitialize the values of the registers.
**/
public static void resetRegisters()
{
for (int i = 0; i < registers.length; i++)
{
registers[i].resetValue();
}
}
/**
* Each individual register is a separate object and Observable. This handy method will add the given Observer to
* each one.
*/
public static void addRegistersObserver(Observer observer)
{
for (int i = 0; i < registers.length; i++)
{
registers[i].addObserver(observer);
}
}
/**
* Each individual register is a separate object and Observable. This handy method will delete the given Observer
* from each one.
*/
public static void deleteRegistersObserver(Observer observer)
{
for (int i = 0; i < registers.length; i++)
{
registers[i].deleteObserver(observer);
}
}
}