diff options
Diffstat (limited to 'src/arch')
-rw-r--r-- | src/arch/x86/predecoder.hh | 63 |
1 files changed, 56 insertions, 7 deletions
diff --git a/src/arch/x86/predecoder.hh b/src/arch/x86/predecoder.hh index a1d0905c5..d7734be88 100644 --- a/src/arch/x86/predecoder.hh +++ b/src/arch/x86/predecoder.hh @@ -59,6 +59,7 @@ #define __ARCH_X86_PREDECODER_HH__ #include "arch/x86/types.hh" +#include "base/bitfield.hh" #include "sim/host.hh" class ThreadContext; @@ -85,6 +86,50 @@ namespace X86ISA //The extended machine instruction being generated ExtMachInst emi; + inline uint8_t getNextByte() + { + return (fetchChunk >> (offset * 8)) & 0xff; + } + + void getImmediate(int &collected, uint64_t ¤t, int size) + { + //Figure out how many bytes we still need to get for the + //immediate. + int toGet = size - collected; + //Figure out how many bytes are left in our "buffer" + int remaining = sizeof(MachInst) - offset; + //Get as much as we need, up to the amount available. + toGet = toGet > remaining ? remaining : toGet; + + //Shift the bytes we want to be all the way to the right + uint64_t partialDisp = fetchChunk >> (offset * 8); + //Mask off what we don't want + partialDisp &= mask(toGet * 8); + //Shift it over to overlay with our displacement. + partialDisp <<= (displacementCollected * 8); + //Put it into our displacement + current |= partialDisp; + //Update how many bytes we've collected. + collected += toGet; + consumeBytes(toGet); + } + + inline void consumeByte() + { + offset++; + assert(offset <= sizeof(MachInst)); + if(offset == sizeof(MachInst)) + outOfBytes = true; + } + + inline void consumeBytes(int numBytes) + { + offset += numBytes; + assert(offset <= sizeof(MachInst)); + if(offset == sizeof(MachInst)) + outOfBytes = true; + } + //State machine state protected: //Whether or not we're out of bytes @@ -98,23 +143,27 @@ namespace X86ISA int immediateSize; int immediateCollected; - //These are local to some of the states. I need to turn the states - //into inline functions to clean things up a bit. - int toGet; - int remaining; - MachInst partialDisp; - enum State { Prefix, Opcode, ModRM, SIB, Displacement, - Immediate + Immediate, + //We should never get to this state. Getting here is an error. + ErrorState }; State state; + //Functions to handle each of the states + State doPrefixState(uint8_t); + State doOpcodeState(uint8_t); + State doModRMState(uint8_t); + State doSIBState(uint8_t); + State doDisplacementState(); + State doImmediateState(); + public: Predecoder(ThreadContext * _tc) : tc(_tc), basePC(0), offset(0), |