2022-07-18 09:59:04 +03:00
|
|
|
#include <stdbool.h>
|
|
|
|
#include <stddef.h>
|
|
|
|
#include <stdint.h>
|
|
|
|
#include <string.h>
|
|
|
|
|
2022-07-20 02:23:23 +03:00
|
|
|
#include <rockos/tty.h>
|
2022-07-18 09:59:04 +03:00
|
|
|
|
|
|
|
#include "vga.h"
|
|
|
|
|
|
|
|
static const size_t VGA_WIDTH = 80;
|
|
|
|
static const size_t VGA_HEIGHT = 25;
|
|
|
|
static uint16_t* const VGA_MEMORY = (uint16_t*) 0xB8000;
|
2022-07-24 07:16:00 +03:00
|
|
|
static uint16_t* const VGA_OLDMEMORY = (uint16_t*) 0xB8000;
|
2022-07-18 09:59:04 +03:00
|
|
|
|
|
|
|
static size_t terminal_row;
|
|
|
|
static size_t terminal_column;
|
|
|
|
static uint8_t terminal_color;
|
|
|
|
static uint16_t* terminal_buffer;
|
2022-07-24 07:16:00 +03:00
|
|
|
static uint16_t* terminal_oldbuffer;
|
2022-07-18 09:59:04 +03:00
|
|
|
|
|
|
|
void terminal_initialize(void) {
|
|
|
|
terminal_row = 0;
|
|
|
|
terminal_column = 0;
|
|
|
|
terminal_color = vga_entry_color(VGA_COLOR_LIGHT_GREY, VGA_COLOR_BLACK);
|
|
|
|
terminal_buffer = VGA_MEMORY;
|
2022-07-24 07:16:00 +03:00
|
|
|
terminal_oldbuffer = VGA_OLDMEMORY;
|
2022-07-18 09:59:04 +03:00
|
|
|
for (size_t y = 0; y < VGA_HEIGHT; y++) {
|
|
|
|
for (size_t x = 0; x < VGA_WIDTH; x++) {
|
|
|
|
const size_t index = y * VGA_WIDTH + x;
|
|
|
|
terminal_buffer[index] = vga_entry(' ', terminal_color);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void terminal_setcolor(uint8_t color) {
|
|
|
|
terminal_color = color;
|
|
|
|
}
|
|
|
|
|
|
|
|
void terminal_putentryat(unsigned char c, uint8_t color, size_t x, size_t y) {
|
|
|
|
const size_t index = y * VGA_WIDTH + x;
|
|
|
|
terminal_buffer[index] = vga_entry(c, color);
|
|
|
|
}
|
|
|
|
|
2022-07-24 07:16:00 +03:00
|
|
|
void scroll_screen_fwd() {
|
|
|
|
memcpy(terminal_oldbuffer, &terminal_buffer[VGA_WIDTH], (VGA_HEIGHT*2-1) * VGA_WIDTH);
|
|
|
|
for(size_t x = 0; x < VGA_WIDTH; ++x) {
|
|
|
|
terminal_oldbuffer[(VGA_HEIGHT-1)*VGA_WIDTH + x] = vga_entry(' ', terminal_color);
|
|
|
|
}
|
|
|
|
memcpy(terminal_buffer, terminal_oldbuffer, VGA_HEIGHT*VGA_WIDTH);
|
|
|
|
--terminal_row;
|
|
|
|
}
|
|
|
|
|
2022-07-18 09:59:04 +03:00
|
|
|
void terminal_putchar(char c) {
|
|
|
|
unsigned char uc = c;
|
|
|
|
|
2022-07-24 07:16:00 +03:00
|
|
|
switch(uc) {
|
|
|
|
case '\n':
|
2022-07-18 22:37:43 +03:00
|
|
|
terminal_column = 0;
|
2022-07-24 07:16:00 +03:00
|
|
|
if (++terminal_row == VGA_HEIGHT)
|
|
|
|
scroll_screen_fwd();
|
|
|
|
break;
|
|
|
|
case 0x08:
|
|
|
|
if(!terminal_column & !terminal_row)
|
|
|
|
return;
|
|
|
|
if(--terminal_column == 0) {
|
|
|
|
terminal_column = VGA_WIDTH;
|
|
|
|
if(--terminal_row == 0) {
|
|
|
|
terminal_row =VGA_HEIGHT;
|
|
|
|
}
|
2022-07-18 22:37:43 +03:00
|
|
|
}
|
2022-07-24 07:16:00 +03:00
|
|
|
terminal_putentryat(' ', terminal_color, terminal_column, terminal_row);
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
terminal_putentryat(uc, terminal_color, terminal_column, terminal_row);
|
|
|
|
if (++terminal_column == VGA_WIDTH) {
|
|
|
|
terminal_column = 0;
|
|
|
|
if (++terminal_row == VGA_HEIGHT) {
|
|
|
|
scroll_screen_fwd();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
break;
|
2022-07-18 09:59:04 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void terminal_write(const char* data, size_t size) {
|
|
|
|
for (size_t i = 0; i < size; i++) {
|
|
|
|
terminal_putchar(data[i]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void terminal_writestring(const char* data) {
|
|
|
|
terminal_write(data, strlen(data));
|
|
|
|
}
|