forked from Tronix286/DOSMID
-
Notifications
You must be signed in to change notification settings - Fork 0
/
timer.c
212 lines (176 loc) · 6.01 KB
/
timer.c
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
/*
* This file contains the implementation of a more precise time than that
* provided by DOS. Routines are provided to increase the clock rate to
* around 1165 interrupts per second, for a granularity of close to 858
* microseconds between clock pulses, rather than the 55 milliseconds between
* normal PC clock pulses (18.2 times/second).
*
* Note that the timer_init() routine must be called before the timer_read()
* routines will work, and that the timer_stop() routine MUST be called
* before the program terminates, or the machine will be toasted. For this
* reason, timer_init() installs the timer_stop() routine to be called at
* program exit.
*/
#include "defines.h"
#include "timer.h" /* include self for control */
#include <stdlib.h> /* atexit() */
#ifdef MSDOS
#include <dos.h> /* _chain_intr(), _disable(), _enable(), _dos_setvect() */
#include <conio.h> /* outp() */
#else
#include <unistd.h>
#include <time.h>
#endif
#ifdef MSDOS
/* selects timer's resolution */
#define TIMER_1165_HZ
/*#define TIMER_582_HZ*/
/*#define TIMER_291_HZ*/
/* 1165.215 hz */
#ifdef TIMER_1165_HZ
#define DIV_MSB 0x04 /* a divisor of 1024 gives: */
#define DIV_LSB 0x00 /* 1193180 / 1024 = 1165 Hz */
#define ORIG_INTR_MASK 63 /* call the original clock INT every 64 calls */
#define USEC_INC 858 /* one cycle is 858.21us to be exact */
#define USEC_COMP 13 /* how many us to compensate for every ORIG_INTR_MASK+1 call */
#endif
/* 582.607 hz */
#ifdef TIMER_582_HZ
#define DIV_MSB 0x08
#define DIV_LSB 0x00
#define ORIG_INTR_MASK 31
#define USEC_INC 1716 /* 1716.42 to be exact */
#define USEC_COMP 13 /* how many us to compensate for every ORIG_INTR_MASK+1 call */
#endif
/* 291.304 hz */
#ifdef TIMER_291_HZ
#define DIV_MSB 0x10
#define DIV_LSB 0x00
#define ORIG_INTR_MASK 15
#define USEC_INC 3433 /* 3432.84 to be exact */
#define USEC_COMP -2
#endif
#define CLOCK_INT 0x08
/* redefine a few functions, as needed by OpenWatcom */
#define setvect _dos_setvect
#define getvect _dos_getvect
#define disable _disable
#define enable _enable
static unsigned long int nowtime; /* current time counter */
static void (__interrupt *oldfunc)(void); /* interrupt function pointer */
/* This routine will handle the clock interrupt at its higher rate. It will
* call the DOS handler every ORIG_INTR_MASK times it is called, to maintain
* the 18.2 times per second that DOS needs to be called. Each time through,
* it adds to the nowtime value.
* When it is not calling the DOS handler, this routine must reset the 8259A
* interrupt controller before returning. */
static void __interrupt handle_clock(void) {
static int callmod = 0;
/* increment the time */
nowtime += USEC_INC;
/* increment the callmod */
callmod++;
callmod &= ORIG_INTR_MASK;
/* if this is the 64th call, then call handler */
if (callmod == 0) {
nowtime += USEC_COMP; /* compensate for integer division inaccuracy */
_chain_intr(oldfunc);
} else { /* otherwise, clear the interrupt controller */
outp(0x20, 0x20); /* end of interrupt */
}
}
#else
#ifndef CLOCK_MONOTONIC_FAST
#define CLOCK_MONOTONIC_FAST CLOCK_MONOTONIC
#endif
static struct timespec reference;
#endif
/* reset the timer value, this can be used by the application to make sure
* no timer wrap occurs during critical parts of the code flow */
void timer_reset(void) {
#ifdef MSDOS
disable();
nowtime = 0;
enable();
#else
clock_gettime(CLOCK_MONOTONIC_FAST, &reference);
#endif
}
/* This routine will stop the timer. It has void return value so that it
* can be an exit procedure. */
void timer_stop(void) {
#ifdef MSDOS
/* Disable interrupts */
disable();
/* Reinstate the old interrupt handler */
setvect(CLOCK_INT, oldfunc);
/* Reinstate the clock rate to standard 18.2 Hz */
outp(0x43, 0x36); /* Set up for count to be sent */
outp(0x40, 0x00); /* LSB = 00 \_together make 65536 (0) */
outp(0x40, 0x00); /* MSB = 00 / */
/* Enable interrupts */
enable();
#endif
}
/* This routine will start the fast clock rate by installing the handle_clock
* routine as the interrupt service routine for the clock interrupt and then
* setting the interrupt rate up to its higher speed by programming the 8253
* timer chip. */
void timer_init(void) {
#ifdef MSDOS
/* Store the old interrupt handler */
oldfunc = getvect(CLOCK_INT);
/* Set the nowtime to zero */
nowtime = 0;
/* Disable interrupts */
disable();
/* Install the new interrupt handler */
setvect(CLOCK_INT, handle_clock);
/* Increase the clock rate */
outp(0x43, 0x36); /* Set up for count to be sent */
outp(0x40, DIV_LSB); /* LSB = 00 \_together make 2^10 = 1024 */
outp(0x40, DIV_MSB); /* MSB = 04 / */
/* Enable interrupts */
enable();
/* Install the timer_stop() routine to be called at exit */
atexit(timer_stop);
#else
timer_reset();
#endif
}
/* This routine will return the present value of the time, as a number of
* microseconds. Interrupts are disabled during this time to prevent the
* clock from changing while it is being read. */
void timer_read(unsigned long int *value) {
#ifdef MSDOS
/* Disable interrupts */
disable();
/* Read the time */
*value = nowtime;
/* Enable interrupts */
enable();
#else
struct timespec now;
clock_gettime(CLOCK_MONOTONIC_FAST, &now);
*value = (now.tv_sec - reference.tv_sec) * 1000000;
*value += (now.tv_nsec - reference.tv_nsec) / 1000;
#endif
}
/* high resolution sleeping routine, waits n microseconds */
void udelay(unsigned long int usec) {
#ifndef MSDOS
if(usec > 100) {
while(usec > 1000000) {
usleep(1000000);
usec -= 1000000;
}
usleep(usec);
return;
}
#endif
unsigned long int t1, t2;
timer_read(&t1);
do {
timer_read(&t2);
} while(t2 >= t1 && t2 - t1 < usec);
}