/* * Copyright (C) 2002-2003 The DOSBox Team * * This program is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation; either version 2 of the License, or * (at your option) any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU Library General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program; if not, write to the Free Software * Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. */ /* $Id: pic.cpp,v 1.14 2003-11-27 18:57:42 qbix79 Exp $ */ #include #include "dosbox.h" #include "inout.h" #include "cpu.h" #include "pic.h" #include "timer.h" #define PIC_QUEUESIZE 128 struct IRQ_Block { bool masked; bool active; bool inservice; Bitu vector; char * name; PIC_EOIHandler * handler; }; struct PIC_Controller { Bitu icw_words; Bitu icw_index; Bitu masked; Bitu active; Bitu inservice; bool auto_eoi; bool request_issr; Bit8u vector_base; }; Bitu PIC_Ticks=0; Bitu PIC_IRQCheck; Bitu PIC_IRQActive; static IRQ_Block irqs[16]; static PIC_Controller pics[2]; struct PICEntry { Bitu index; PIC_EventHandler event; PICEntry * next; }; static struct { PICEntry entries[PIC_QUEUESIZE]; PICEntry * free_entry; PICEntry * next_entry; } pic; static void write_command(Bit32u port,Bit8u val) { PIC_Controller * pic=&pics[port==0x20 ? 0 : 1]; Bitu irq_base=port==0x20 ? 0 : 8; Bitu i; Bit16u IRQ_priority_table[16] = { 0,1,8,9,10,11,12,13,14,15,2,3,4,5,6,7 }; switch (val) { case 0x0A: /* select read interrupt request register */ pic->request_issr=false; break; case 0x0B: /* select read interrupt in-service register */ pic->request_issr=true; break; case 0x10: /* ICW1 */ pic->icw_index=1; pic->icw_words=2; break; case 0x11: /* ICW1 + need for ICW4 */ pic->icw_index=1; pic->icw_words=3; break; case 0x20:case 0x21:case 0x22:case 0x23:case 0x24:case 0x25:case 0x26:case 0x27: if (PIC_IRQActive<(irq_base+8)) { irqs[PIC_IRQActive].inservice=false; if (irqs[PIC_IRQActive].handler!=0) irqs[PIC_IRQActive].handler(); PIC_IRQActive=PIC_NOIRQ; for (i=0; i<=15; i++){ if(irqs[IRQ_priority_table[i]].inservice) { PIC_IRQActive=IRQ_priority_table[i]; break; } } } //TODO Warnings? break; case 0x60:case 0x61:case 0x62:case 0x63:case 0x64:case 0x65:case 0x66:case 0x67: /* Spefific EOI 0-7 */ if (PIC_IRQActive==(irq_base+val-0x60U)) { irqs[PIC_IRQActive].inservice=false; if (irqs[PIC_IRQActive].handler!=0) irqs[PIC_IRQActive].handler(); PIC_IRQActive=PIC_NOIRQ; for (i=0; i<=15; i++) { if (irqs[IRQ_priority_table[i]].inservice) { PIC_IRQActive=IRQ_priority_table[i]; break; } } }//TODO Warnings? break; case 0xC0:case 0xC1:case 0xC2:case 0xC3:case 0xC4:case 0xC5:case 0xC6:case 0xC7: /* Priority order, no need for it */ break; default: E_Exit("PIC:Unhandled command %02X",val); } } static void write_data(Bit32u port,Bit8u val) { PIC_Controller * pic=&pics[port==0x21 ? 0 : 1]; Bitu irq_base=(port==0x21) ? 0 : 8; Bitu i; switch(pic->icw_index) { case 0: /* mask register */ LOG(LOG_PIC,LOG_NORMAL)("%d mask %X",port==0x21 ? 0 : 1,val); for (i=0;i<=7;i++) { irqs[i+irq_base].masked=(val&(1<0; if (irqs[i+irq_base].active && !irqs[i+irq_base].masked) PIC_IRQCheck|=(1 << (i+irq_base)); else PIC_IRQCheck&=~(1 << (i+irq_base)); }; #if 0 if (PIC_IRQCheck) { CPU_CycleLeft+=CPU_Cycles; CPU_Cycles=0; } #endif break; case 1: /* icw2 */ LOG(LOG_PIC,LOG_NORMAL)("%d:Base vector %X",port==0x21 ? 0 : 1,val); for (i=0;i<=7;i++) { irqs[i+irq_base].vector=(val&0xf8)+i; }; if(pic->icw_index++ >= pic->icw_words) pic->icw_index=0; break; case 2: /* icw 3 */ LOG(LOG_PIC,LOG_NORMAL)("%d:ICW 3 %X",port==0x21 ? 0 : 1,val); if(pic->icw_index++ >= pic->icw_words) pic->icw_index=0; break; case 3: /* icw 4 */ /* 0 1 8086/8080 0 mcs-8085 mode 1 1 Auto EOI 1 Normal EOI 2-3 0x Non buffer Mode 10 Buffer Mode Slave 11 Buffer mode Master 4 Special/Not Special nested mode */ pic->auto_eoi=(val & 0x2)>0; LOG(LOG_PIC,LOG_NORMAL)("%d:ICW 4 %X",port==0x21 ? 0 : 1,val); if(pic->icw_index++ >= pic->icw_words) pic->icw_index=0; break; default: /* icw 3, and 4*/ LOG(LOG_PIC,LOG_NORMAL)("ICW HUH? %X",val); } } static Bit8u read_command(Bit32u port) { PIC_Controller * pic=&pics[port==0x20 ? 0 : 1]; Bitu irq_base=(port==0x20) ? 0 : 8; Bitu i;Bit8u ret=0;Bit8u b=1; if (pic->request_issr) { for (i=irq_base;i15) E_Exit("PIC:Illegal IRQ"); irqs[irq].name=name; irqs[irq].handler=handler; } void PIC_FreeIRQ(Bitu irq) { if (irq>15) E_Exit("PIC:Illegal IRQ"); irqs[irq].name=0; irqs[irq].handler=0; irqs[irq].active=0; irqs[irq].inservice=0; PIC_IRQCheck&=~(1 << irq); } void PIC_ActivateIRQ(Bitu irq) { if (irq<16) { irqs[irq].active=true; if (!irqs[irq].masked) { PIC_IRQCheck|=(1 << irq); } } } void PIC_DeActivateIRQ(Bitu irq) { if (irq<16) { irqs[irq].active=false; PIC_IRQCheck&=~(1 << irq); } } void PIC_runIRQs(void) { Bitu i; if (!GETFLAG(IF)) return; if (!PIC_IRQCheck) return; Bit16u IRQ_priority_lookup[17] = { 0,1,10,11,12,13,14,15,2,3,4,5,6,7,8,9,16 }; Bit16u activeIRQ = PIC_IRQActive; if (activeIRQ==PIC_NOIRQ) activeIRQ = 16; for (i=0;i<=15;i++) { if (IRQ_priority_lookup[i]next=0; pic.next_entry=entry; } else if (find_entry->index>entry->index) { pic.next_entry=entry; entry->next=find_entry; } else while (find_entry) { if (find_entry->next) { /* See if the next index comes later than this one */ if (find_entry->next->index > entry->index) { entry->next=find_entry->next; find_entry->next=entry; break; } else { find_entry=find_entry->next; } } else { entry->next=find_entry->next; find_entry->next=entry; break; } } Bits cycles=PIC_MakeCycles(pic.next_entry->index-PIC_Index()); if (cyclesindex=index; entry->event=handler; pic.free_entry=pic.free_entry->next; AddEntry(entry); } void PIC_RemoveEvents(PIC_EventHandler handler) { PICEntry * entry=pic.next_entry; PICEntry * prev_entry; prev_entry=0; while (entry) { if (entry->event==handler) { if (prev_entry) { prev_entry->next=entry->next; entry->next=pic.free_entry; pic.free_entry=entry; entry=prev_entry->next; continue; } else { pic.next_entry=entry->next; entry->next=pic.free_entry; pic.free_entry=entry; entry=pic.next_entry; continue; } } prev_entry=entry; entry=entry->next; } } bool PIC_RunQueue(void) { /* Check to see if a new milisecond needs to be started */ CPU_CycleLeft+=CPU_Cycles; CPU_Cycles=0; if (CPU_CycleLeft<=0) { return false; } /* Check the queue for an entry */ Bitu index=PIC_Index(); while (pic.next_entry && pic.next_entry->index<=index) { PICEntry * entry=pic.next_entry; pic.next_entry=entry->next; (entry->event)(); /* Put the entry in the free list */ entry->next=pic.free_entry; pic.free_entry=entry; } /* Check when to set the new cycle end */ if (pic.next_entry) { Bits cycles=PIC_MakeCycles(pic.next_entry->index-index); if (!cycles) cycles=1; if (cycles Timers; TIMER_Block * TIMER_RegisterTickHandler(TIMER_TickHandler handler) { Timer * new_timer=new(Timer); new_timer->type=T_TICK; new_timer->tick.handler=handler; Timers.push_front(new_timer); return (TIMER_Block *)new_timer; } TIMER_Block * TIMER_RegisterMicroHandler(TIMER_MicroHandler handler,Bitu micro) { Timer * new_timer=new(Timer); new_timer->type=T_MICRO; new_timer->micro.handler=handler; Timers.push_front(new_timer); TIMER_SetNewMicro(new_timer,micro); return (TIMER_Block *)new_timer; } void TIMER_SetNewMicro(TIMER_Block * block,Bitu micro) { Timer * timer=(Timer *)block; if (timer->type!=T_MICRO) E_Exit("TIMER:Illegal handler type"); timer->micro.total=micro; Bitu index=PIC_Index(); while ((1000-index)>micro) { PIC_AddEvent(timer->micro.handler,micro); micro+=micro; index+=micro; } timer->micro.left=timer->micro.total-(1000-index); } void TIMER_AddTick(void) { /* Setup new amount of cycles for PIC */ CPU_CycleLeft=CPU_CycleMax; CPU_Cycles=0; PIC_Ticks++; /* Go through the list of scheduled irq's and lower their index with 1000 */ PICEntry * entry=pic.next_entry; while (entry) { if (entry->index>1000) entry->index-=1000; else entry->index=0; entry=entry->next; } Bits index; /* Check if there are timer handlers that need to be called */ std::list::iterator i; for(i=Timers.begin(); i != Timers.end(); ++i) { Timer * timers=(*i); switch (timers->type) { case T_TICK: timers->tick.handler(1); break; case T_MICRO: index=1000; while (index>=timers->micro.left) { PIC_AddEvent(timers->micro.handler,timers->micro.left); index-=timers->micro.left; timers->micro.left=timers->micro.total; } timers->micro.left-=index; break; default: E_Exit("TIMER:Illegal handler type"); } } } void PIC_Init(Section* sec) { /* Setup pic0 and pic1 with initial values like DOS has normally */ PIC_IRQCheck=0; PIC_IRQActive=PIC_NOIRQ; PIC_Ticks=0; Bitu i; for (i=0;i<2;i++) { pics[i].masked=0xff; pics[i].active=0; pics[i].inservice=0; pics[i].auto_eoi=false; pics[i].auto_eoi=false; pics[i].request_issr=false; pics[i].icw_index=0; pics[i].icw_words=0; } for (i=0;i<=7;i++) { irqs[i].active=false; irqs[i].masked=true; irqs[i].inservice=false; irqs[i+8].active=false; irqs[i+8].masked=true; irqs[i+8].inservice=false; irqs[i].vector=0x8+i; irqs[i+8].vector=0x70+i; } irqs[0].masked=false; /* Enable system timer */ irqs[1].masked=false; /* Enable Keyboard IRQ */ irqs[8].masked=false; /* Enable RTC IRQ */ irqs[12].masked=false; /* Enable Mouse IRQ */ IO_RegisterReadHandler(0x20,read_command,"Master PIC Command"); IO_RegisterReadHandler(0x21,read_data,"Master PIC Data"); IO_RegisterWriteHandler(0x20,write_command,"Master PIC Command"); IO_RegisterWriteHandler(0x21,write_data,"Master PIC Data"); IO_RegisterReadHandler(0xa0,read_command,"Slave PIC Command"); IO_RegisterReadHandler(0xa1,read_data,"Slave PIC Data"); IO_RegisterWriteHandler(0xa0,write_command,"Slave PIC Command"); IO_RegisterWriteHandler(0xa1,write_data,"Slave PIC Data"); /* Initialize the pic queue */ for (i=0;i