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
|
#ifndef INT10_H
#define INT10_H
#include <stdint.h>
#define BIOS_DATA_AREA_SEGMENT 0x40
// TODO Assuming screenwidth, videopage on ss rather than using far
static uint8_t int10_get_video_mode(uint8_t *screenwidth, uint8_t *videopage);
#pragma aux int10_get_video_mode = \
"mov al, 0" \
"mov ah, 0x0F" \
"int 0x10" \
"mov ss:[si], ah" \
"mov ss:[di], bh" \
__parm [si] [di] \
__value [al] \
__modify [ax bh]
static inline uint32_t bda_get_dword(unsigned int offset) {
uint32_t __far *p = MK_FP(BIOS_DATA_AREA_SEGMENT, offset);
return *p;
}
static inline uint16_t bda_get_word(unsigned int offset) {
uint16_t __far *p = MK_FP(BIOS_DATA_AREA_SEGMENT, offset);
return *p;
}
static inline uint8_t bda_get_byte(unsigned int offset) {
uint8_t __far *p = MK_FP(BIOS_DATA_AREA_SEGMENT, offset);
return *p;
}
#define bda_get_ebda_segment() bda_get_word(0x0e)
#define bda_get_video_mode() bda_get_byte(0x49)
#define bda_get_num_columns() bda_get_word(0x4a)
#define bda_get_video_page_size() bda_get_word(0x4c)
#define bda_get_cur_video_page() bda_get_word(0x62)
#define bda_get_last_row() bda_get_byte(0x84)
#define bda_get_tick_count() bda_get_dword(0x6c)
#define bda_get_tick_count_lo() bda_get_word(0x6c)
static inline uint16_t __far * get_video_char(uint8_t page, unsigned int x, unsigned int y)
{
return MK_FP(0xB800, (page * bda_get_video_page_size())
+ ((y * bda_get_num_columns()) + x) * 2);
}
static inline uint8_t __far * get_video_scanline(uint8_t mode, uint8_t page, unsigned int y)
{
switch (mode) {
case 4:
case 5:
case 6: // CGA modes
// Scalines are 80 bytes long, however they are interleaved.
// Even numbered scanlines begin at 0, odd lines begin at 0x2000.
// So offset by y%2 * 0x2000.
return MK_FP(0xB800, (page * bda_get_video_page_size())
+ ((y/2) * 80) + (y%2) * 0x2000);
default:
return 0;
}
}
#endif
|