kernel/os_text.c (view raw)
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
#include "linker.h"
#include "assembly.h"
#include "os_string.h"
#define VGA_TEXTMODE_COLUMNS 80
#define VGA_TEXTMODE_LINES 25
#define VGA_TEXTMODE_BPC 2
#define VGA_TEXTMODE_LINE_WIDTH (VGA_TEXTMODE_COLUMNS * VGA_TEXTMODE_BPC)
#define VGA_TEXTMODE_CELLS (VGA_TEXTMODE_COLUMNS * VGA_TEXTMODE_LINES)
#define VGA_MEM_POS(line, column) ((line) * VGA_TEXTMODE_COLUMNS * VGA_TEXTMODE_BPC + (column) * VGA_TEXTMODE_BPC)
#define VGA_TEXTMODE_MEM ((uint8_t*)0xB8000)
#define VGA_MEM_VALUE(line, column) (VGA_TEXTMODE_MEM[VGA_MEM_POS((line), (column))])
#define VGA_MEM_COLOR(line, column) (VGA_TEXTMODE_MEM[VGA_MEM_POS((line), (column)) + 1])
#define VGA_MEM_ADDR(line, column) (&(VGA_MEM_VALUE((line), (column))))
#define VGA_INDEX_REG ((uint16_t)0x3D4)
#define VGA_DATA_REG ((uint16_t)0x3D5)
uint32_t line = 0;
uint32_t column = 0;
uint8_t color = 0x0F;
__attribute((always_inline)) inline void set_char(char value) {
VGA_MEM_VALUE(line, column) = value;
VGA_MEM_COLOR(line, column) = color;
}
__attribute((always_inline)) inline void lfcr() {
line++;
column = 0;
if (line >= VGA_TEXTMODE_LINES) {
memcpy(
VGA_MEM_ADDR(0, 0),
VGA_MEM_ADDR(1, 0),
VGA_TEXTMODE_LINE_WIDTH * (VGA_TEXTMODE_LINES - 1)
);
memset(
VGA_MEM_ADDR(VGA_TEXTMODE_LINES - 1, 0),
0,
VGA_TEXTMODE_LINE_WIDTH
);
line = VGA_TEXTMODE_LINES - 1;
}
}
__attribute((always_inline)) inline void write_char(char value) {
if (column >= VGA_TEXTMODE_COLUMNS || value == '\n') {
lfcr();
}
switch (value) {
case '\n':
break;
case '\0':
break;
default:
set_char(value);
column++;
break;
}
}
__attribute((always_inline)) inline void set_cursor(uint16_t line, uint16_t column) {
uint16_t pos = line * VGA_TEXTMODE_COLUMNS + column;
outb(0x0F, VGA_INDEX_REG);
outb((uint8_t)(pos & 0xFF), VGA_DATA_REG);
outb(0x0E, VGA_INDEX_REG);
outb((uint8_t)(pos >> 8), VGA_DATA_REG);
}
void text_backspace(void) {
if (column == 0 && line > 0) {
line--;
column = VGA_TEXTMODE_COLUMNS - 1;
} else if (column > 0) {
column--;
}
set_char(0);
set_cursor(line, column);
}
void text_write_n(const char *string, uint32_t num) {
for (uint32_t i = 0; i < num; i++) {
write_char(string[0]);
}
set_cursor(line, column);
}
void text_write(const char *string) {
while (*string) {
write_char(*(string++));
}
set_cursor(line, column);
}
void text_write_c(const char c) {
write_char(c);
set_cursor(line, column);
}
void enable_cursor(void) {
outb(0x0A, VGA_INDEX_REG);
outb((inb(VGA_DATA_REG) & 0x0C) | 0x00, VGA_DATA_REG);
outb(0x0B, VGA_INDEX_REG);
outb((inb(VGA_DATA_REG) & 0xE0) | 0x0F, VGA_DATA_REG);
}
void text_clear(void) {
for (int i = 0; i < VGA_TEXTMODE_CELLS; i++) {
VGA_TEXTMODE_MEM[2 * i] = 0;
VGA_TEXTMODE_MEM[2 * i + 1] = 0;
}
line = 0;
column = 0;
}
int text_init(void) {
text_clear();
enable_cursor();
set_cursor(line, column);
return 1;
}