aboutsummaryrefslogtreecommitdiff
path: root/int10vga.h
blob: a2a20a05c2b9e96893307c87d8922c8b9f1bd140 (plain)
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