succesfully but there are some minor quirks to iron out. Who would've known a DELAY SLOT introduces that much complexity?! arrgh!
Anyways, a lot of this stuff had to do with my project at MIPS and me needing to know how I was going to get this working for the MIPS
ISA. So I figured I would try to touch it up and throw it in here (I hate to introduce non-completely working components... )
src/arch/alpha/isa/mem.isa:
spacing
src/arch/mips/faults.cc:
src/arch/mips/faults.hh:
Gabe really authored this
src/arch/mips/isa/decoder.isa:
add StoreConditional Flag to instruction
src/arch/mips/isa/formats/basic.isa:
Steven really did this file
src/arch/mips/isa/formats/branch.isa:
fix bug for uncond/cond control
src/arch/mips/isa/formats/mem.isa:
Adjust O3CPU memory access to use new memory model interface.
src/arch/mips/isa/formats/util.isa:
update LoadStoreBase template
src/arch/mips/isa_traits.cc:
update SERIALIZE partially
src/arch/mips/process.cc:
src/arch/mips/process.hh:
no need for this for NOW. ASID/Virtual addressing handles it
src/arch/mips/regfile/misc_regfile.hh:
add in clear() function and comments for future usage of special misc. regs
src/cpu/base_dyn_inst.hh:
add in nextNPC variable and supporting functions.
add isCondDelaySlot function
Update predTaken and mispredicted functions
src/cpu/base_dyn_inst_impl.hh:
init nextNPC
src/cpu/o3/SConscript:
add MIPS files to compile
src/cpu/o3/alpha/thread_context.hh:
no need for my name on this file
src/cpu/o3/bpred_unit_impl.hh:
Update RAS appropriately for MIPS
src/cpu/o3/comm.hh:
add some extra communication variables to aid in handling the
delay slots
src/cpu/o3/commit.hh:
minor name fix for nextNPC functions.
src/cpu/o3/commit_impl.hh:
src/cpu/o3/decode_impl.hh:
src/cpu/o3/fetch_impl.hh:
src/cpu/o3/iew_impl.hh:
src/cpu/o3/inst_queue_impl.hh:
src/cpu/o3/rename_impl.hh:
Fix necessary variables and functions for squashes with delay slots
src/cpu/o3/cpu.cc:
Update function interface ...
adjust removeInstsNotInROB function to recognize delay slots insts
src/cpu/o3/cpu.hh:
update removeInstsNotInROB
src/cpu/o3/decode.hh:
declare necessary variables for handling delay slot
src/cpu/o3/dyn_inst.hh:
Add in MipsDynInst
src/cpu/o3/fetch.hh:
src/cpu/o3/iew.hh:
src/cpu/o3/rename.hh:
declare necessary variables and adjust functions for handling delay slot
src/cpu/o3/inst_queue.hh:
src/cpu/simple/base.cc:
no need for my name here
src/cpu/o3/isa_specific.hh:
add in MIPS files
src/cpu/o3/scoreboard.hh:
dont include alpha specific isa traits!
src/cpu/o3/thread_context.hh:
no need for my name here, i just rearranged where the file goes
src/cpu/static_inst.hh:
add isCondDelaySlot function
src/cpu/o3/mips/cpu.cc:
src/cpu/o3/mips/cpu.hh:
src/cpu/o3/mips/cpu_builder.cc:
src/cpu/o3/mips/cpu_impl.hh:
src/cpu/o3/mips/dyn_inst.cc:
src/cpu/o3/mips/dyn_inst.hh:
src/cpu/o3/mips/dyn_inst_impl.hh:
src/cpu/o3/mips/impl.hh:
src/cpu/o3/mips/params.hh:
src/cpu/o3/mips/thread_context.cc:
src/cpu/o3/mips/thread_context.hh:
MIPS file for O3CPU...mirrors ALPHA definition
--HG--
extra : convert_revision : 9bb199b4085903e49ffd5a4c8ac44d11460d988c
254 lines
7.6 KiB
C++
254 lines
7.6 KiB
C++
/*
|
|
* Copyright (c) 2006 The Regents of The University of Michigan
|
|
* All rights reserved.
|
|
*
|
|
* Redistribution and use in source and binary forms, with or without
|
|
* modification, are permitted provided that the following conditions are
|
|
* met: redistributions of source code must retain the above copyright
|
|
* notice, this list of conditions and the following disclaimer;
|
|
* redistributions in binary form must reproduce the above copyright
|
|
* notice, this list of conditions and the following disclaimer in the
|
|
* documentation and/or other materials provided with the distribution;
|
|
* neither the name of the copyright holders nor the names of its
|
|
* contributors may be used to endorse or promote products derived from
|
|
* this software without specific prior written permission.
|
|
*
|
|
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
|
|
* "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
|
|
* LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
|
|
* A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
|
|
* OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
|
|
* SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
|
|
* LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
|
|
* DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
|
|
* THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
|
* (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
|
|
* OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
|
*
|
|
* Authors: Kevin Lim
|
|
* Korey Sewell
|
|
*/
|
|
|
|
#include "config/use_checker.hh"
|
|
|
|
#include "arch/mips/faults.hh"
|
|
#include "base/cprintf.hh"
|
|
#include "base/statistics.hh"
|
|
#include "base/timebuf.hh"
|
|
#include "cpu/checker/thread_context.hh"
|
|
#include "sim/sim_events.hh"
|
|
#include "sim/stats.hh"
|
|
|
|
#include "cpu/o3/mips/cpu.hh"
|
|
#include "cpu/o3/mips/params.hh"
|
|
#include "cpu/o3/mips/thread_context.hh"
|
|
#include "cpu/o3/comm.hh"
|
|
#include "cpu/o3/thread_state.hh"
|
|
|
|
using namespace TheISA;
|
|
|
|
template <class Impl>
|
|
MipsO3CPU<Impl>::MipsO3CPU(Params *params)
|
|
: FullO3CPU<Impl>(params)
|
|
{
|
|
DPRINTF(O3CPU, "Creating MipsO3CPU object.\n");
|
|
|
|
// Setup any thread state.
|
|
this->thread.resize(this->numThreads);
|
|
|
|
for (int i = 0; i < this->numThreads; ++i) {
|
|
if (i < params->workload.size()) {
|
|
DPRINTF(O3CPU, "Workload[%i] process is %#x",
|
|
i, this->thread[i]);
|
|
this->thread[i] = new Thread(this, i, params->workload[i],
|
|
i, params->mem);
|
|
|
|
this->thread[i]->setStatus(ThreadContext::Suspended);
|
|
|
|
|
|
/* Use this port to for syscall emulation writes to memory. */
|
|
Port *mem_port;
|
|
TranslatingPort *trans_port;
|
|
trans_port = new TranslatingPort(csprintf("%s-%d-funcport",
|
|
name(), i),
|
|
params->workload[i]->pTable,
|
|
false);
|
|
mem_port = params->mem->getPort("functional");
|
|
mem_port->setPeer(trans_port);
|
|
trans_port->setPeer(mem_port);
|
|
this->thread[i]->setMemPort(trans_port);
|
|
|
|
//usedTids[i] = true;
|
|
//threadMap[i] = i;
|
|
} else {
|
|
//Allocate Empty thread so M5 can use later
|
|
//when scheduling threads to CPU
|
|
Process* dummy_proc = NULL;
|
|
|
|
this->thread[i] = new Thread(this, i, dummy_proc, i, params->mem);
|
|
//usedTids[i] = false;
|
|
}
|
|
|
|
ThreadContext *tc;
|
|
|
|
// Setup the TC that will serve as the interface to the threads/CPU.
|
|
MipsTC<Impl> *mips_tc =
|
|
new MipsTC<Impl>;
|
|
|
|
tc = mips_tc;
|
|
|
|
// If we're using a checker, then the TC should be the
|
|
// CheckerThreadContext.
|
|
#if USE_CHECKER
|
|
if (params->checker) {
|
|
tc = new CheckerThreadContext<MipsTC<Impl> >(
|
|
mips_tc, this->checker);
|
|
}
|
|
#endif
|
|
|
|
mips_tc->cpu = this;
|
|
mips_tc->thread = this->thread[i];
|
|
|
|
// Give the thread the TC.
|
|
this->thread[i]->tc = tc;
|
|
|
|
// Add the TC to the CPU's list of TC's.
|
|
this->threadContexts.push_back(tc);
|
|
}
|
|
|
|
for (int i=0; i < this->numThreads; i++) {
|
|
this->thread[i]->setFuncExeInst(0);
|
|
}
|
|
|
|
// Sets CPU pointers. These must be set at this level because the CPU
|
|
// pointers are defined to be the highest level of CPU class.
|
|
this->fetch.setCPU(this);
|
|
this->decode.setCPU(this);
|
|
this->rename.setCPU(this);
|
|
this->iew.setCPU(this);
|
|
this->commit.setCPU(this);
|
|
|
|
this->rob.setCPU(this);
|
|
this->regFile.setCPU(this);
|
|
|
|
lockAddr = 0;
|
|
lockFlag = false;
|
|
}
|
|
|
|
template <class Impl>
|
|
void
|
|
MipsO3CPU<Impl>::regStats()
|
|
{
|
|
// Register stats for everything that has stats.
|
|
this->fullCPURegStats();
|
|
this->fetch.regStats();
|
|
this->decode.regStats();
|
|
this->rename.regStats();
|
|
this->iew.regStats();
|
|
this->commit.regStats();
|
|
}
|
|
|
|
|
|
template <class Impl>
|
|
MiscReg
|
|
MipsO3CPU<Impl>::readMiscReg(int misc_reg, unsigned tid)
|
|
{
|
|
return this->regFile.readMiscReg(misc_reg, tid);
|
|
}
|
|
|
|
template <class Impl>
|
|
MiscReg
|
|
MipsO3CPU<Impl>::readMiscRegWithEffect(int misc_reg, Fault &fault,
|
|
unsigned tid)
|
|
{
|
|
return this->regFile.readMiscRegWithEffect(misc_reg, fault, tid);
|
|
}
|
|
|
|
template <class Impl>
|
|
Fault
|
|
MipsO3CPU<Impl>::setMiscReg(int misc_reg, const MiscReg &val, unsigned tid)
|
|
{
|
|
return this->regFile.setMiscReg(misc_reg, val, tid);
|
|
}
|
|
|
|
template <class Impl>
|
|
Fault
|
|
MipsO3CPU<Impl>::setMiscRegWithEffect(int misc_reg, const MiscReg &val,
|
|
unsigned tid)
|
|
{
|
|
return this->regFile.setMiscRegWithEffect(misc_reg, val, tid);
|
|
}
|
|
|
|
template <class Impl>
|
|
void
|
|
MipsO3CPU<Impl>::squashFromTC(unsigned tid)
|
|
{
|
|
this->thread[tid]->inSyscall = true;
|
|
this->commit.generateTCEvent(tid);
|
|
}
|
|
|
|
template <class Impl>
|
|
void
|
|
MipsO3CPU<Impl>::trap(Fault fault, unsigned tid)
|
|
{
|
|
// Pass the thread's TC into the invoke method.
|
|
fault->invoke(this->threadContexts[tid]);
|
|
}
|
|
|
|
#if !FULL_SYSTEM
|
|
|
|
template <class Impl>
|
|
void
|
|
MipsO3CPU<Impl>::syscall(int64_t callnum, int tid)
|
|
{
|
|
DPRINTF(O3CPU, "[tid:%i] Executing syscall().\n\n", tid);
|
|
|
|
DPRINTF(Activity,"Activity: syscall() called.\n");
|
|
|
|
// Temporarily increase this by one to account for the syscall
|
|
// instruction.
|
|
++(this->thread[tid]->funcExeInst);
|
|
|
|
// Execute the actual syscall.
|
|
this->thread[tid]->syscall(callnum);
|
|
|
|
// Decrease funcExeInst by one as the normal commit will handle
|
|
// incrementing it.
|
|
--(this->thread[tid]->funcExeInst);
|
|
|
|
DPRINTF(O3CPU, "[tid:%i] Register 2 is %i ", tid, this->readIntReg(2));
|
|
}
|
|
|
|
template <class Impl>
|
|
TheISA::IntReg
|
|
MipsO3CPU<Impl>::getSyscallArg(int i, int tid)
|
|
{
|
|
return this->readArchIntReg(MipsISA::ArgumentReg0 + i, tid);
|
|
}
|
|
|
|
template <class Impl>
|
|
void
|
|
MipsO3CPU<Impl>::setSyscallArg(int i, IntReg val, int tid)
|
|
{
|
|
this->setArchIntReg(MipsISA::ArgumentReg0 + i, val, tid);
|
|
}
|
|
|
|
template <class Impl>
|
|
void
|
|
MipsO3CPU<Impl>::setSyscallReturn(SyscallReturn return_value, int tid)
|
|
{
|
|
// check for error condition. Mips syscall convention is to
|
|
// indicate success/failure in reg a3 (r19) and put the
|
|
// return value itself in the standard return value reg (v0).
|
|
if (return_value.successful()) {
|
|
// no error
|
|
this->setArchIntReg(SyscallSuccessReg, 0, tid);
|
|
this->setArchIntReg(ReturnValueReg, return_value.value(), tid);
|
|
} else {
|
|
// got an error, return details
|
|
this->setArchIntReg(SyscallSuccessReg, (IntReg) -1, tid);
|
|
this->setArchIntReg(ReturnValueReg, -return_value.value(), tid);
|
|
}
|
|
}
|
|
#endif
|