138 lines
3 KiB
C
138 lines
3 KiB
C
/*
|
|
** thinlib (c) 2001 Matthew Conte (matt@conte.com)
|
|
**
|
|
**
|
|
** tl_timer.c
|
|
**
|
|
** DOS timer routines
|
|
**
|
|
** $Id: $
|
|
*/
|
|
|
|
#include <go32.h>
|
|
#include <pc.h>
|
|
#include <dpmi.h>
|
|
|
|
#include "tl_types.h"
|
|
#include "tl_djgpp.h"
|
|
#include "tl_timer.h"
|
|
|
|
#define TIMER_INT 0x08
|
|
#define TIMER_TICKS 1193182L
|
|
|
|
#define TIMER_CONTROL 0x43
|
|
#define TIMER_ACCESS 0x40
|
|
#define TIMER_DEFAULT_MODE 0x36
|
|
|
|
static _go32_dpmi_seginfo old_handler, new_handler;
|
|
|
|
static struct
|
|
{
|
|
timerhandler_t handler;
|
|
void *param;
|
|
int interval, frac, current;
|
|
uint8 current_mode;
|
|
bool initialized;
|
|
} timer;
|
|
|
|
/* port 0x43 - write control word
|
|
** port 0x40 - read/write timer count
|
|
** control word composition:
|
|
** D7-D6 - 0=counter0, 1=counter1, 2=counter2, 3=read-back
|
|
** D5-D4 - 0=latch, 1=r/w lsb, 2=r/w msb, 3=r/2 lsb then msb
|
|
** D3-D1 - 0=m0, 1=m1, 2/6=m2, 3/7=m3, 4=m4, 5=m5
|
|
** D0 - 0=16-bit binary counter, 1=BCD counter
|
|
**
|
|
** mode 0 - interrupt on terminal count
|
|
** mode 1 - hardware retriggerable one-shot
|
|
** mode 2 - rate generator
|
|
** mode 3 - square wave mode
|
|
** mode 4 - software triggered mode
|
|
** mode 5 - hardware triggered strobe (retriggerable)
|
|
*/
|
|
|
|
|
|
/* Reprogram the PIT timer to fire at a specified value */
|
|
void thin_timer_setrate(int hertz)
|
|
{
|
|
int time;
|
|
|
|
if (0 == hertz)
|
|
time = 0;
|
|
else
|
|
time = TIMER_TICKS / (long) hertz;
|
|
|
|
timer.interval = time;
|
|
timer.frac = time & 0xFFFF;
|
|
|
|
if (0 == time)
|
|
timer.current_mode = TIMER_DEFAULT_MODE; /* reset to standard */
|
|
|
|
outportb(TIMER_CONTROL, timer.current_mode);
|
|
outportb(TIMER_ACCESS, timer.frac & 0xFF);
|
|
outportb(TIMER_ACCESS, timer.frac >> 8);
|
|
}
|
|
|
|
static void _timer_int_handler(void)
|
|
{
|
|
timer.current += timer.frac;
|
|
if (timer.current >= timer.interval)
|
|
{
|
|
timer.current -= timer.interval;
|
|
timer.handler(timer.param);
|
|
}
|
|
|
|
outportb(0x20, 0x20);
|
|
}
|
|
THIN_LOCKED_STATIC_FUNC(_timer_int_handler);
|
|
|
|
/* Lock code, data, and chain an interrupt handler */
|
|
int thin_timer_init(int hertz, timerhandler_t func_ptr, void *param)
|
|
{
|
|
if (false == timer.initialized)
|
|
{
|
|
THIN_LOCK_FUNC(_timer_int_handler);
|
|
THIN_LOCK_VAR(timer);
|
|
|
|
timer.handler = func_ptr;
|
|
timer.param = param;
|
|
|
|
/* chain onto old timer interrupt */
|
|
_go32_dpmi_get_protected_mode_interrupt_vector(TIMER_INT, &old_handler);
|
|
new_handler.pm_offset = (int) _timer_int_handler;
|
|
new_handler.pm_selector = _go32_my_cs();
|
|
_go32_dpmi_chain_protected_mode_interrupt_vector(TIMER_INT, &new_handler);
|
|
|
|
timer.current = 0;
|
|
|
|
/* Set PIC to fire at desired refresh rate */
|
|
|
|
/* counter 0, lsb+msb, mode 3, binary counter */
|
|
timer.current_mode = 0x36;
|
|
|
|
timer.initialized = true;
|
|
}
|
|
|
|
thin_timer_setrate(hertz);
|
|
|
|
return 0;
|
|
}
|
|
|
|
/* Remove the timer handler */
|
|
void thin_timer_shutdown(void)
|
|
{
|
|
if (false == timer.initialized)
|
|
return;
|
|
|
|
/* Restore previous timer setting */
|
|
thin_timer_setrate(0);
|
|
|
|
/* Remove the interrupt handler */
|
|
_go32_dpmi_set_protected_mode_interrupt_vector(TIMER_INT, &old_handler);
|
|
|
|
timer.initialized = false;
|
|
}
|
|
|
|
/*
|
|
** $Log: $
|
|
*/
|