nuclear@7: /* nuclear@7: 256-color 3D graphics hack for real-mode DOS. nuclear@7: Copyright (C) 2011 John Tsiombikas nuclear@7: nuclear@7: This program is free software: you can redistribute it and/or modify nuclear@7: it under the terms of the GNU General Public License as published by nuclear@7: the Free Software Foundation, either version 3 of the License, or nuclear@7: (at your option) any later version. nuclear@7: nuclear@7: This program is distributed in the hope that it will be useful, nuclear@7: but WITHOUT ANY WARRANTY; without even the implied warranty of nuclear@7: MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the nuclear@7: GNU General Public License for more details. nuclear@7: nuclear@7: You should have received a copy of the GNU General Public License nuclear@7: along with this program. If not, see . nuclear@7: */ nuclear@7: #include nuclear@7: #include nuclear@7: #include nuclear@7: #include nuclear@7: #include nuclear@7: #include "pit8254.h" nuclear@7: nuclear@7: #define PIT_TIMER_INTR 8 nuclear@7: #define DOS_TIMER_INTR 0x1c nuclear@7: nuclear@7: /* macro to divide and round to the nearest integer */ nuclear@7: #define DIV_ROUND(a, b) \ nuclear@7: ((a) / (b) + ((a) % (b)) / ((b) / 2)) nuclear@7: nuclear@7: static void set_timer_reload(int reload_val); nuclear@7: static void cleanup(void); nuclear@7: static void __interrupt __far timer_irq(); nuclear@7: static void __interrupt __far dos_timer_intr(); nuclear@7: nuclear@7: static void (__interrupt __far *prev_timer_intr)(); nuclear@7: nuclear@7: static unsigned long ticks; nuclear@7: static unsigned long tick_interval, ticks_per_dos_intr; nuclear@7: static int inum; nuclear@7: nuclear@7: void init_timer(int res_hz) nuclear@7: { nuclear@7: _disable(); nuclear@7: nuclear@7: if(res_hz > 0) { nuclear@7: int reload_val = DIV_ROUND(OSC_FREQ_HZ, res_hz); nuclear@7: set_timer_reload(reload_val); nuclear@7: nuclear@7: tick_interval = DIV_ROUND(1000, res_hz); nuclear@7: ticks_per_dos_intr = DIV_ROUND(65535L, reload_val); nuclear@7: nuclear@7: inum = PIT_TIMER_INTR; nuclear@7: prev_timer_intr = _dos_getvect(inum); nuclear@7: _dos_setvect(inum, timer_irq); nuclear@7: } else { nuclear@7: tick_interval = 55; nuclear@7: nuclear@7: inum = DOS_TIMER_INTR; nuclear@7: prev_timer_intr = _dos_getvect(inum); nuclear@7: _dos_setvect(inum, dos_timer_intr); nuclear@7: } nuclear@7: _enable(); nuclear@7: nuclear@7: atexit(cleanup); nuclear@7: } nuclear@7: nuclear@7: static void cleanup(void) nuclear@7: { nuclear@7: if(!prev_timer_intr) { nuclear@7: return; /* init hasn't ran, there's nothing to cleanup */ nuclear@7: } nuclear@7: nuclear@7: _disable(); nuclear@7: if(inum == PIT_TIMER_INTR) { nuclear@7: /* restore the original timer frequency */ nuclear@7: set_timer_reload(65535); nuclear@7: } nuclear@7: nuclear@7: /* restore the original interrupt handler */ nuclear@7: _dos_setvect(inum, prev_timer_intr); nuclear@7: _enable(); nuclear@7: } nuclear@7: nuclear@7: void reset_timer(void) nuclear@7: { nuclear@7: ticks = 0; nuclear@7: } nuclear@7: nuclear@7: unsigned long get_msec(void) nuclear@7: { nuclear@7: return ticks * tick_interval; nuclear@7: } nuclear@7: nuclear@7: static void set_timer_reload(int reload_val) nuclear@7: { nuclear@7: outp(PORT_CMD, CMD_CHAN0 | CMD_ACCESS_BOTH | CMD_OP_SQWAVE); nuclear@7: outp(PORT_DATA0, reload_val & 0xff); nuclear@7: outp(PORT_DATA0, (reload_val >> 8) & 0xff); nuclear@7: } nuclear@7: nuclear@7: static void __interrupt __far dos_timer_intr() nuclear@7: { nuclear@7: ticks++; nuclear@7: _chain_intr(prev_timer_intr); /* DOES NOT RETURN */ nuclear@7: } nuclear@7: nuclear@7: /* first PIC command port */ nuclear@7: #define PIC1_CMD 0x20 nuclear@7: /* end of interrupt control word */ nuclear@7: #define OCW2_EOI (1 << 5) nuclear@7: nuclear@7: static void __interrupt __far timer_irq() nuclear@7: { nuclear@7: static unsigned long dos_ticks; nuclear@7: nuclear@7: ticks++; nuclear@7: nuclear@7: if(++dos_ticks >= ticks_per_dos_intr) { nuclear@7: /* I suppose the dos irq handler does the EOI so I shouldn't nuclear@7: * do it if I am to call the previous function nuclear@7: */ nuclear@7: dos_ticks = 0; nuclear@7: _chain_intr(prev_timer_intr); /* XXX DOES NOT RETURN */ nuclear@7: return; /* just for clarity */ nuclear@7: } nuclear@7: nuclear@7: /* send EOI to the PIC */ nuclear@7: outp(PIC1_CMD, OCW2_EOI); nuclear@7: }