outl(0x840, 0x0100);
}
+void timer_handler(smi_event_t ev)
+{
+ static unsigned int ticks = 0;
+
+ smi_disable_event(SMI_EVENT_FAST_TIMER);
+ smi_enable_event(SMI_EVENT_FAST_TIMER);
+
+ outb(0x80, (ticks++) & 0xFF);
+
+ outlog();
+}
+
void smi_entry(void)
{
char statstr[512];
vgasave = inb(0x3D4);
counter++;
- outb(0x80, (counter & 0xFF));
-
sprintf(statstr, "15-412! %08x %08x", smi_status(), counter);
strblit(statstr, 0, 0);
}
smi_poll();
- outlog();
outl(0xCF8, pcisave);
outb(0x3D4, vgasave);
extern int _bss, _bssend;
+extern void timer_handler(smi_event_t ev);
+
void __firstrun_start() {
unsigned char *bp;
smram_state_t smram;
pci_write16(0, 31, 4, 0xC0, 0);
/* Turn on the SMIs we want */
- outb(0x830, inb(0x830) | ICH2_SMI_EN_SWSMI_TMR_EN);
+ smi_disable();
+ smi_register_handler(SMI_EVENT_FAST_TIMER, timer_handler);
+ smi_enable_event(SMI_EVENT_FAST_TIMER);
outb(0x848, ICH2_DEVTRAP_EN_KBC_TRP_EN);
smi_enable();
#include <reg-82801b.h>
#include <output.h>
+static smi_handler_t _handlers[SMI_EVENT_MAX] = {0};
+
static uint16_t _get_PMBASE()
{
static long pmbase = -1;
if (sts & ICH2_SMI_STS_SWSMI_TMR_STS) // Ack it, then request another.
{
+ if (_handlers[SMI_EVENT_FAST_TIMER] == SMI_HANDLER_NONE)
+ output("Unhandled: SWSMI_TMR_STS");
+ else if (_handlers[SMI_EVENT_FAST_TIMER] != SMI_HANDLER_IGNORE)
+ _handlers[SMI_EVENT_FAST_TIMER](SMI_EVENT_FAST_TIMER);
outl(_get_PMBASE() + ICH2_PMBASE_SMI_STS, ICH2_SMI_STS_SWSMI_TMR_STS);
- outl(_get_PMBASE() + ICH2_PMBASE_SMI_EN,
- inl(_get_PMBASE() + ICH2_PMBASE_SMI_EN) & ~ICH2_SMI_EN_SWSMI_TMR_EN);
- outl(_get_PMBASE() + ICH2_PMBASE_SMI_EN,
- inl(_get_PMBASE() + ICH2_PMBASE_SMI_EN) | ICH2_SMI_EN_SWSMI_TMR_EN);
}
if (sts & ICH2_SMI_STS_PM1_STS_REG)
ICH2_SMI_EN_EOS |
ICH2_SMI_EN_GBL_SMI_EN);
}
+
+int smi_register_handler(smi_event_t ev, smi_handler_t hnd)
+{
+ if (ev >= SMI_EVENT_MAX)
+ return -1;
+ _handlers[ev] = hnd;
+ return 0;
+}
+
+int smi_enable_event(smi_event_t ev)
+{
+ switch(ev)
+ {
+ case SMI_EVENT_FAST_TIMER:
+ outl(_get_PMBASE() + ICH2_PMBASE_SMI_EN,
+ inl(_get_PMBASE() + ICH2_PMBASE_SMI_EN) |
+ ICH2_SMI_EN_SWSMI_TMR_EN);
+ return 0;
+ default:
+ return -1;
+ }
+}
+
+int smi_disable_event(smi_event_t ev)
+{
+ switch(ev)
+ {
+ case SMI_EVENT_FAST_TIMER:
+ outl(_get_PMBASE() + ICH2_PMBASE_SMI_EN,
+ inl(_get_PMBASE() + ICH2_PMBASE_SMI_EN) &
+ ~ICH2_SMI_EN_SWSMI_TMR_EN);
+ return 0;
+ default:
+ return -1;
+ }
+}
extern void smi_poll();
extern unsigned long smi_status(); /* Architecturally defined; for debugging only. */
+typedef enum {
+ SMI_EVENT_FAST_TIMER = 0,
+ SMI_EVENT_MAX
+} smi_event_t;
+
+
+typedef void (*smi_handler_t)(smi_event_t);
+
+#define SMI_HANDLER_NONE ((smi_handler_t)0)
+#define SMI_HANDLER_IGNORE ((smi_handler_t)-1)
+
+extern int smi_register_handler(smi_event_t ev, smi_handler_t hnd);
+extern int smi_enable_event(smi_event_t ev);
+extern int smi_disable_event(smi_event_t ev);
+
#endif