2 * Copyright (c) 2007 The Hewlett-Packard Development Company
5 * Redistribution and use of this software in source and binary forms,
6 * with or without modification, are permitted provided that the
7 * following conditions are met:
9 * The software must be used only for Non-Commercial Use which means any
10 * use which is NOT directed to receiving any direct monetary
11 * compensation for, or commercial advantage from such use. Illustrative
12 * examples of non-commercial use are academic research, personal study,
13 * teaching, education and corporate research & development.
14 * Illustrative examples of commercial use are distributing products for
15 * commercial advantage and providing services using the software for
16 * commercial advantage.
18 * If you wish to use this software or functionality therein that may be
19 * covered by patents for commercial use, please contact:
20 * Director of Intellectual Property Licensing
21 * Office of Strategy and Technology
22 * Hewlett-Packard Company
24 * Palo Alto, California 94304
26 * Redistributions of source code must retain the above copyright notice,
27 * this list of conditions and the following disclaimer. Redistributions
28 * in binary form must reproduce the above copyright notice, this list of
29 * conditions and the following disclaimer in the documentation and/or
30 * other materials provided with the distribution. Neither the name of
31 * the COPYRIGHT HOLDER(s), HEWLETT-PACKARD COMPANY, nor the names of its
32 * contributors may be used to endorse or promote products derived from
33 * this software without specific prior written permission. No right of
34 * sublicense is granted herewith. Derivatives of the software and
35 * output created using the software may be prepared, but only for
36 * Non-Commercial Uses. Derivatives of the software may be shared with
37 * others provided: (i) the others agree to abide by the list of
38 * conditions herein which includes the Non-Commercial Use restrictions;
39 * and (ii) such Derivatives of the software include the above copyright
40 * notice to acknowledge the contribution from this software where
41 * applicable, this list of conditions and the disclaimer below.
43 * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
44 * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
45 * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
46 * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
47 * OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
48 * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
49 * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
50 * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
51 * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
52 * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
53 * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
58 #ifndef __ARCH_X86_PREDECODER_HH__
59 #define __ARCH_X86_PREDECODER_HH__
63 #include "arch/x86/types.hh"
64 #include "arch/x86/miscregs.hh"
65 #include "base/bitfield.hh"
66 #include "base/misc.hh"
67 #include "base/trace.hh"
68 #include "base/types.hh"
77 //These are defined and documented in predecoder_tables.cc
78 static const uint8_t Prefixes[256];
79 static const uint8_t UsesModRM[2][256];
80 static const uint8_t ImmediateType[2][256];
81 static const uint8_t SizeTypeToSize[3][10];
85 //The bytes to be predecoded
87 //The pc of the start of fetchChunk
89 //The pc the current instruction started at
91 //The offset into fetchChunk of current processing
93 //The extended machine instruction being generated
97 inline uint8_t getNextByte()
99 return ((uint8_t *)&fetchChunk)[offset];
102 void getImmediate(int &collected, uint64_t ¤t, int size)
104 //Figure out how many bytes we still need to get for the
106 int toGet = size - collected;
107 //Figure out how many bytes are left in our "buffer"
108 int remaining = sizeof(MachInst) - offset;
109 //Get as much as we need, up to the amount available.
110 toGet = toGet > remaining ? remaining : toGet;
112 //Shift the bytes we want to be all the way to the right
113 uint64_t partialImm = fetchChunk >> (offset * 8);
114 //Mask off what we don't want
115 partialImm &= mask(toGet * 8);
116 //Shift it over to overlay with our displacement.
117 partialImm <<= (immediateCollected * 8);
118 //Put it into our displacement
119 current |= partialImm;
120 //Update how many bytes we've collected.
125 inline void consumeByte()
128 assert(offset <= sizeof(MachInst));
129 if(offset == sizeof(MachInst))
133 inline void consumeBytes(int numBytes)
136 assert(offset <= sizeof(MachInst));
137 if(offset == sizeof(MachInst))
143 //State machine state
145 //Whether or not we're out of bytes
147 //Whether we've completed generating an ExtMachInst
149 //The size of the displacement value
150 int displacementSize;
151 //The size of the immediate value
153 //This is how much of any immediate value we've gotten. This is used
154 //for both the actual immediate and the displacement.
155 int immediateCollected;
165 //We should never get to this state. Getting here is an error.
171 //Functions to handle each of the states
172 State doPrefixState(uint8_t);
173 State doOpcodeState(uint8_t);
174 State doModRMState(uint8_t);
175 State doSIBState(uint8_t);
176 State doDisplacementState();
177 State doImmediateState();
180 Predecoder(ThreadContext * _tc) :
181 tc(_tc), basePC(0), origPC(0), offset(0),
182 outOfBytes(true), emiIsReady(false),
185 emi.mode.mode = LongMode;
186 emi.mode.submode = SixtyFourBitMode;
195 ThreadContext * getTC()
200 void setTC(ThreadContext * _tc)
207 //Use this to give data to the predecoder. This should be used
208 //when there is control flow.
209 void moreBytes(Addr pc, Addr fetchPC, MachInst data)
211 DPRINTF(Predecoder, "Getting more bytes.\n");
213 offset = (fetchPC >= pc) ? 0 : pc - fetchPC;
224 bool extMachInstReady()
229 //This returns a constant reference to the ExtMachInst to avoid a copy
230 const ExtMachInst & getExtMachInst()
240 "Calculating the instruction size: "
241 "basePC: %#x offset: %#x origPC: %#x\n",
242 basePC, offset, origPC);
243 return basePC + offset - origPC;
248 #endif // __ARCH_X86_PREDECODER_HH__