ARM: Implement WFE/WFI/SEV semantics.
This commit is contained in:
parent
ba8d64520e
commit
1b505f5291
5 changed files with 29 additions and 6 deletions
|
@ -137,6 +137,19 @@ class Interrupts : public SimObject
|
||||||
(interrupts[INT_RST]));
|
(interrupts[INT_RST]));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Check the raw interrupt state.
|
||||||
|
* This function is used to check if a wfi operation should sleep. If there
|
||||||
|
* is an interrupt pending, even if it's masked, wfi doesn't sleep.
|
||||||
|
* @return any interrupts pending
|
||||||
|
*/
|
||||||
|
bool
|
||||||
|
checkRaw() const
|
||||||
|
{
|
||||||
|
return intStatus;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
Fault
|
Fault
|
||||||
getInterrupt(ThreadContext *tc)
|
getInterrupt(ThreadContext *tc)
|
||||||
{
|
{
|
||||||
|
|
|
@ -247,6 +247,7 @@ let {{
|
||||||
NextJazelle = ((CPSR)newCpsr).j;
|
NextJazelle = ((CPSR)newCpsr).j;
|
||||||
NextItState = ((((CPSR)newCpsr).it2 << 2) & 0xFC)
|
NextItState = ((((CPSR)newCpsr).it2 << 2) & 0xFC)
|
||||||
| (((CPSR)newCpsr).it1 & 0x3);
|
| (((CPSR)newCpsr).it1 & 0x3);
|
||||||
|
SevMailbox = 1;
|
||||||
'''
|
'''
|
||||||
buildImmDataInst(mnem + 's', code, flagType,
|
buildImmDataInst(mnem + 's', code, flagType,
|
||||||
suffix = "ImmPclr", buildCc = False,
|
suffix = "ImmPclr", buildCc = False,
|
||||||
|
|
|
@ -96,6 +96,7 @@ let {{
|
||||||
IWNPC = cSwap(%s, cpsr.e) | ((Spsr & 0x20) ? 1 : 0);
|
IWNPC = cSwap(%s, cpsr.e) | ((Spsr & 0x20) ? 1 : 0);
|
||||||
NextItState = ((((CPSR)Spsr).it2 << 2) & 0xFC)
|
NextItState = ((((CPSR)Spsr).it2 << 2) & 0xFC)
|
||||||
| (((CPSR)Spsr).it1 & 0x3);
|
| (((CPSR)Spsr).it1 & 0x3);
|
||||||
|
SevMailbox = 1;
|
||||||
'''
|
'''
|
||||||
|
|
||||||
microLdrRetUopIop = InstObjParams('ldr_ret_uop', 'MicroLdrRetUop',
|
microLdrRetUopIop = InstObjParams('ldr_ret_uop', 'MicroLdrRetUop',
|
||||||
|
|
|
@ -483,11 +483,10 @@ let {{
|
||||||
|
|
||||||
wfeCode = '''
|
wfeCode = '''
|
||||||
#if FULL_SYSTEM
|
#if FULL_SYSTEM
|
||||||
if (SevMailbox) {
|
if (SevMailbox == 1) {
|
||||||
SevMailbox = 0;
|
SevMailbox = 0;
|
||||||
PseudoInst::quiesceSkip(xc->tcBase());
|
PseudoInst::quiesceSkip(xc->tcBase());
|
||||||
}
|
} else {
|
||||||
else {
|
|
||||||
PseudoInst::quiesce(xc->tcBase());
|
PseudoInst::quiesce(xc->tcBase());
|
||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
|
@ -501,7 +500,12 @@ let {{
|
||||||
|
|
||||||
wfiCode = '''
|
wfiCode = '''
|
||||||
#if FULL_SYSTEM
|
#if FULL_SYSTEM
|
||||||
PseudoInst::quiesce(xc->tcBase());
|
// WFI doesn't sleep if interrupts are pending (masked or not)
|
||||||
|
if (xc->tcBase()->getCpuPtr()->getInterruptController()->checkRaw()) {
|
||||||
|
PseudoInst::quiesceSkip(xc->tcBase());
|
||||||
|
} else {
|
||||||
|
PseudoInst::quiesce(xc->tcBase());
|
||||||
|
}
|
||||||
#endif
|
#endif
|
||||||
'''
|
'''
|
||||||
wfiIop = InstObjParams("wfi", "WfiInst", "PredOp", \
|
wfiIop = InstObjParams("wfi", "WfiInst", "PredOp", \
|
||||||
|
@ -517,8 +521,12 @@ let {{
|
||||||
System *sys = xc->tcBase()->getSystemPtr();
|
System *sys = xc->tcBase()->getSystemPtr();
|
||||||
for (int x = 0; x < sys->numContexts(); x++) {
|
for (int x = 0; x < sys->numContexts(); x++) {
|
||||||
ThreadContext *oc = sys->getThreadContext(x);
|
ThreadContext *oc = sys->getThreadContext(x);
|
||||||
if (oc != xc->tcBase()) {
|
if (oc == xc->tcBase())
|
||||||
|
continue;
|
||||||
|
// Only wake if they were sleeping
|
||||||
|
if (oc->readMiscReg(MISCREG_SEV_MAILBOX) == 0) {
|
||||||
oc->setMiscReg(MISCREG_SEV_MAILBOX, 1);
|
oc->setMiscReg(MISCREG_SEV_MAILBOX, 1);
|
||||||
|
PseudoInst::wakeCPU(xc->tcBase(), x);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
'''
|
'''
|
||||||
|
|
|
@ -53,7 +53,7 @@
|
||||||
|
|
||||||
class ArmSystem : public System
|
class ArmSystem : public System
|
||||||
{
|
{
|
||||||
private:
|
protected:
|
||||||
/**
|
/**
|
||||||
* PC based event to skip the dprink() call and emulate its
|
* PC based event to skip the dprink() call and emulate its
|
||||||
* functionality
|
* functionality
|
||||||
|
|
Loading…
Reference in a new issue