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