gem5  v20.0.0.3
isa.cc
Go to the documentation of this file.
1 /*
2  * Copyright (c) 2009 The Regents of The University of Michigan
3  * All rights reserved.
4  *
5  * Redistribution and use in source and binary forms, with or without
6  * modification, are permitted provided that the following conditions are
7  * met: redistributions of source code must retain the above copyright
8  * notice, this list of conditions and the following disclaimer;
9  * redistributions in binary form must reproduce the above copyright
10  * notice, this list of conditions and the following disclaimer in the
11  * documentation and/or other materials provided with the distribution;
12  * neither the name of the copyright holders nor the names of its
13  * contributors may be used to endorse or promote products derived from
14  * this software without specific prior written permission.
15  *
16  * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
17  * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
18  * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
19  * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
20  * OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
21  * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
22  * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
23  * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
24  * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
25  * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
26  * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
27  */
28 
29 #include "arch/x86/isa.hh"
30 
31 #include "arch/x86/decoder.hh"
32 #include "arch/x86/tlb.hh"
33 #include "cpu/base.hh"
34 #include "cpu/thread_context.hh"
35 #include "params/X86ISA.hh"
36 #include "sim/serialize.hh"
37 
38 namespace X86ISA
39 {
40 
41 void
42 ISA::updateHandyM5Reg(Efer efer, CR0 cr0,
43  SegAttr csAttr, SegAttr ssAttr, RFLAGS rflags,
44  ThreadContext *tc)
45 {
46  HandyM5Reg m5reg = 0;
47  if (efer.lma) {
48  m5reg.mode = LongMode;
49  if (csAttr.longMode)
50  m5reg.submode = SixtyFourBitMode;
51  else
52  m5reg.submode = CompatabilityMode;
53  } else {
54  m5reg.mode = LegacyMode;
55  if (cr0.pe) {
56  if (rflags.vm)
57  m5reg.submode = Virtual8086Mode;
58  else
59  m5reg.submode = ProtectedMode;
60  } else {
61  m5reg.submode = RealMode;
62  }
63  }
64  m5reg.cpl = csAttr.dpl;
65  m5reg.paging = cr0.pg;
66  m5reg.prot = cr0.pe;
67 
68  // Compute the default and alternate operand size.
69  if (m5reg.submode == SixtyFourBitMode || csAttr.defaultSize) {
70  m5reg.defOp = 2;
71  m5reg.altOp = 1;
72  } else {
73  m5reg.defOp = 1;
74  m5reg.altOp = 2;
75  }
76 
77  // Compute the default and alternate address size.
78  if (m5reg.submode == SixtyFourBitMode) {
79  m5reg.defAddr = 3;
80  m5reg.altAddr = 2;
81  } else if (csAttr.defaultSize) {
82  m5reg.defAddr = 2;
83  m5reg.altAddr = 1;
84  } else {
85  m5reg.defAddr = 1;
86  m5reg.altAddr = 2;
87  }
88 
89  // Compute the stack size
90  if (m5reg.submode == SixtyFourBitMode) {
91  m5reg.stack = 3;
92  } else if (ssAttr.defaultSize) {
93  m5reg.stack = 2;
94  } else {
95  m5reg.stack = 1;
96  }
97 
98  regVal[MISCREG_M5_REG] = m5reg;
99  if (tc)
100  tc->getDecoderPtr()->setM5Reg(m5reg);
101 }
102 
103 void
105 {
106  // Blank everything. 0 might not be an appropriate value for some things,
107  // but it is for most.
108  memset(regVal, 0, NumMiscRegs * sizeof(RegVal));
109 
110  // If some state should be non-zero after a reset, set those values here.
111  regVal[MISCREG_CR0] = 0x0000000060000010ULL;
112 
113  regVal[MISCREG_MTRRCAP] = 0x0508;
114 
115  regVal[MISCREG_MCG_CAP] = 0x104;
116 
117  regVal[MISCREG_PAT] = 0x0007040600070406ULL;
118 
119  regVal[MISCREG_SYSCFG] = 0x20601;
120 
121  regVal[MISCREG_TOP_MEM] = 0x4000000;
122 
123  regVal[MISCREG_DR6] = (mask(8) << 4) | (mask(16) << 16);
124  regVal[MISCREG_DR7] = 1 << 10;
125 
126  LocalApicBase lApicBase = 0;
127  lApicBase.base = 0xFEE00000 >> 12;
128  lApicBase.enable = 1;
129  // The "bsp" bit will be set when this register is read, since then we'll
130  // have a ThreadContext to check the contextId from.
131  regVal[MISCREG_APIC_BASE] = lApicBase;
132 }
133 
135 {
136  clear();
137 }
138 
139 const X86ISAParams *
140 ISA::params() const
141 {
142  return dynamic_cast<const Params *>(_params);
143 }
144 
145 RegVal
146 ISA::readMiscRegNoEffect(int miscReg) const
147 {
148  // Make sure we're not dealing with an illegal control register.
149  // Instructions should filter out these indexes, and nothing else should
150  // attempt to read them directly.
151  assert(isValidMiscReg(miscReg));
152 
153  return regVal[miscReg];
154 }
155 
156 RegVal
157 ISA::readMiscReg(int miscReg, ThreadContext * tc)
158 {
159  if (miscReg == MISCREG_TSC) {
160  return regVal[MISCREG_TSC] + tc->getCpuPtr()->curCycle();
161  }
162 
163  if (miscReg == MISCREG_FSW) {
164  RegVal fsw = regVal[MISCREG_FSW];
166  return insertBits(fsw, 13, 11, top);
167  }
168 
169  if (miscReg == MISCREG_APIC_BASE) {
170  LocalApicBase base = regVal[MISCREG_APIC_BASE];
171  base.bsp = (tc->contextId() == 0);
172  return base;
173  }
174 
175  return readMiscRegNoEffect(miscReg);
176 }
177 
178 void
180 {
181  // Make sure we're not dealing with an illegal control register.
182  // Instructions should filter out these indexes, and nothing else should
183  // attempt to write to them directly.
184  assert(isValidMiscReg(miscReg));
185 
186  HandyM5Reg m5Reg = regVal[MISCREG_M5_REG];
187  int reg_width = 64;
188  switch (miscReg) {
189  case MISCREG_X87_TOP:
190  reg_width = 3;
191  break;
192  case MISCREG_FTW:
193  reg_width = 8;
194  break;
195  case MISCREG_FSW:
196  case MISCREG_FCW:
197  case MISCREG_FOP:
198  reg_width = 16;
199  break;
200  case MISCREG_MXCSR:
201  reg_width = 32;
202  break;
203  case MISCREG_FISEG:
204  case MISCREG_FOSEG:
205  if (m5Reg.submode != SixtyFourBitMode)
206  reg_width = 16;
207  break;
208  case MISCREG_FIOFF:
209  case MISCREG_FOOFF:
210  if (m5Reg.submode != SixtyFourBitMode)
211  reg_width = 32;
212  break;
213  default:
214  break;
215  }
216 
217  regVal[miscReg] = val & mask(reg_width);
218 }
219 
220 void
222 {
223  RegVal newVal = val;
224  switch(miscReg)
225  {
226  case MISCREG_CR0:
227  {
228  CR0 toggled = regVal[miscReg] ^ val;
229  CR0 newCR0 = val;
230  Efer efer = regVal[MISCREG_EFER];
231  if (toggled.pg && efer.lme) {
232  if (newCR0.pg) {
233  //Turning on long mode
234  efer.lma = 1;
235  regVal[MISCREG_EFER] = efer;
236  } else {
237  //Turning off long mode
238  efer.lma = 0;
239  regVal[MISCREG_EFER] = efer;
240  }
241  }
242  if (toggled.pg) {
243  dynamic_cast<TLB *>(tc->getITBPtr())->flushAll();
244  dynamic_cast<TLB *>(tc->getDTBPtr())->flushAll();
245  }
246  //This must always be 1.
247  newCR0.et = 1;
248  newVal = newCR0;
250  newCR0,
254  tc);
255  }
256  break;
257  case MISCREG_CR2:
258  break;
259  case MISCREG_CR3:
260  dynamic_cast<TLB *>(tc->getITBPtr())->flushNonGlobal();
261  dynamic_cast<TLB *>(tc->getDTBPtr())->flushNonGlobal();
262  break;
263  case MISCREG_CR4:
264  {
265  CR4 toggled = regVal[miscReg] ^ val;
266  if (toggled.pae || toggled.pse || toggled.pge) {
267  dynamic_cast<TLB *>(tc->getITBPtr())->flushAll();
268  dynamic_cast<TLB *>(tc->getDTBPtr())->flushAll();
269  }
270  }
271  break;
272  case MISCREG_CR8:
273  break;
274  case MISCREG_CS_ATTR:
275  {
276  SegAttr toggled = regVal[miscReg] ^ val;
277  SegAttr newCSAttr = val;
278  if (toggled.longMode) {
279  if (newCSAttr.longMode) {
284  } else {
289  }
290  }
293  newCSAttr,
296  tc);
297  }
298  break;
299  case MISCREG_SS_ATTR:
303  val,
305  tc);
306  break;
307  // These segments always actually use their bases, or in other words
308  // their effective bases must stay equal to their actual bases.
309  case MISCREG_FS_BASE:
310  case MISCREG_GS_BASE:
311  case MISCREG_HS_BASE:
312  case MISCREG_TSL_BASE:
313  case MISCREG_TSG_BASE:
314  case MISCREG_TR_BASE:
315  case MISCREG_IDTR_BASE:
317  break;
318  // These segments ignore their bases in 64 bit mode.
319  // their effective bases must stay equal to their actual bases.
320  case MISCREG_ES_BASE:
321  case MISCREG_CS_BASE:
322  case MISCREG_SS_BASE:
323  case MISCREG_DS_BASE:
324  {
325  Efer efer = regVal[MISCREG_EFER];
326  SegAttr csAttr = regVal[MISCREG_CS_ATTR];
327  if (!efer.lma || !csAttr.longMode) // Check for non 64 bit mode.
328  regVal[MISCREG_SEG_EFF_BASE(miscReg -
330  }
331  break;
332  case MISCREG_TSC:
333  regVal[MISCREG_TSC] = val - tc->getCpuPtr()->curCycle();
334  return;
335  case MISCREG_DR0:
336  case MISCREG_DR1:
337  case MISCREG_DR2:
338  case MISCREG_DR3:
339  /* These should eventually set up breakpoints. */
340  break;
341  case MISCREG_DR4:
342  miscReg = MISCREG_DR6;
344  case MISCREG_DR6:
345  {
346  DR6 dr6 = regVal[MISCREG_DR6];
347  DR6 newDR6 = val;
348  dr6.b0 = newDR6.b0;
349  dr6.b1 = newDR6.b1;
350  dr6.b2 = newDR6.b2;
351  dr6.b3 = newDR6.b3;
352  dr6.bd = newDR6.bd;
353  dr6.bs = newDR6.bs;
354  dr6.bt = newDR6.bt;
355  newVal = dr6;
356  }
357  break;
358  case MISCREG_DR5:
359  miscReg = MISCREG_DR7;
361  case MISCREG_DR7:
362  {
363  DR7 dr7 = regVal[MISCREG_DR7];
364  DR7 newDR7 = val;
365  dr7.l0 = newDR7.l0;
366  dr7.g0 = newDR7.g0;
367  if (dr7.l0 || dr7.g0) {
368  panic("Debug register breakpoints not implemented.\n");
369  } else {
370  /* Disable breakpoint 0. */
371  }
372  dr7.l1 = newDR7.l1;
373  dr7.g1 = newDR7.g1;
374  if (dr7.l1 || dr7.g1) {
375  panic("Debug register breakpoints not implemented.\n");
376  } else {
377  /* Disable breakpoint 1. */
378  }
379  dr7.l2 = newDR7.l2;
380  dr7.g2 = newDR7.g2;
381  if (dr7.l2 || dr7.g2) {
382  panic("Debug register breakpoints not implemented.\n");
383  } else {
384  /* Disable breakpoint 2. */
385  }
386  dr7.l3 = newDR7.l3;
387  dr7.g3 = newDR7.g3;
388  if (dr7.l3 || dr7.g3) {
389  panic("Debug register breakpoints not implemented.\n");
390  } else {
391  /* Disable breakpoint 3. */
392  }
393  dr7.gd = newDR7.gd;
394  dr7.rw0 = newDR7.rw0;
395  dr7.len0 = newDR7.len0;
396  dr7.rw1 = newDR7.rw1;
397  dr7.len1 = newDR7.len1;
398  dr7.rw2 = newDR7.rw2;
399  dr7.len2 = newDR7.len2;
400  dr7.rw3 = newDR7.rw3;
401  dr7.len3 = newDR7.len3;
402  }
403  break;
404  case MISCREG_M5_REG:
405  // Writing anything to the m5reg with side effects makes it update
406  // based on the current values of the relevant registers. The actual
407  // value written is discarded.
408  updateHandyM5Reg(regVal[MISCREG_EFER],
409  regVal[MISCREG_CR0],
410  regVal[MISCREG_CS_ATTR],
412  regVal[MISCREG_RFLAGS],
413  tc);
414  return;
415  default:
416  break;
417  }
418  setMiscRegNoEffect(miscReg, newVal);
419 }
420 
421 void
423 {
425 }
426 
427 void
429 {
436  NULL);
437 }
438 
439 void
441 {
442  tc->getDecoderPtr()->setM5Reg(regVal[MISCREG_M5_REG]);
443 }
444 
445 }
446 
447 X86ISA::ISA *
448 X86ISAParams::create()
449 {
450  return new X86ISA::ISA(this);
451 }
#define panic(...)
This implements a cprintf based panic() function.
Definition: logging.hh:163
RegVal readMiscReg(int miscReg, ThreadContext *tc)
Definition: isa.cc:157
void serialize(CheckpointOut &cp) const override
Serialize an object.
Definition: isa.cc:422
virtual TheISA::Decoder * getDecoderPtr()=0
Definition: test.h:61
virtual BaseTLB * getDTBPtr()=0
const Params * params() const
Definition: isa.cc:140
void updateHandyM5Reg(Efer efer, CR0 cr0, SegAttr csAttr, SegAttr ssAttr, RFLAGS rflags, ThreadContext *tc)
Definition: isa.cc:42
uint64_t RegVal
Definition: types.hh:166
virtual BaseCPU * getCpuPtr()=0
Definition: cprintf.cc:40
static bool isValidMiscReg(int index)
Definition: misc.hh:402
ThreadContext is the external interface to all thread state for anything outside of the CPU...
Bitfield< 63 > val
Definition: misc.hh:769
void clear()
Definition: isa.cc:104
void setMiscReg(int miscReg, RegVal val, ThreadContext *tc)
Definition: isa.cc:221
virtual void startup()
startup() is the final initialization call before simulation.
Definition: sim_object.cc:96
RegVal readMiscRegNoEffect(int miscReg) const
Definition: isa.cc:146
#define M5_FALLTHROUGH
Definition: compiler.hh:84
void unserialize(CheckpointIn &cp) override
Unserialize an object.
Definition: isa.cc:428
mask
Definition: misc.hh:796
void setMiscRegNoEffect(int miscReg, RegVal val)
Definition: isa.cc:179
Bitfield< 51, 12 > base
Definition: pagetable.hh:141
virtual BaseTLB * getITBPtr()=0
Cycles curCycle() const
Determine the current cycle, corresponding to a tick aligned to a clock edge.
T insertBits(T val, int first, int last, B bit_val)
Returns val with bits first to last set to the LSBs of bit_val.
Definition: bitfield.hh:131
#define ULL(N)
uint64_t constant
Definition: types.hh:48
#define SERIALIZE_ARRAY(member, size)
Definition: serialize.hh:805
RegVal regVal[NUM_MISCREGS]
Definition: isa.hh:53
#define UNSERIALIZE_ARRAY(member, size)
Definition: serialize.hh:813
std::ostream CheckpointOut
Definition: serialize.hh:63
This is exposed globally, independent of the ISA.
Definition: acpi.hh:55
const SimObjectParams * _params
Cached copy of the object parameters.
Definition: sim_object.hh:111
virtual ContextID contextId() const =0
static MiscRegIndex MISCREG_SEG_EFF_BASE(int index)
Definition: misc.hh:519
Definition: isa.hh:47
Bitfield< 0 > p
Definition: pagetable.hh:151
X86ISAParams Params
Definition: isa.hh:60
ISA(Params *p)
Definition: isa.cc:134
const int NumMiscRegs
Definition: registers.hh:55

Generated on Fri Jul 3 2020 15:42:39 for gem5 by doxygen 1.8.13