108 lines
2.6 KiB
C
108 lines
2.6 KiB
C
/*
|
|
Copyright 2019 Benji Dial
|
|
|
|
Permission to use, copy, modify, and/or distribute this
|
|
software for any purpose with or without fee is hereby
|
|
granted, provided that the above copyright notice and this
|
|
permission notice appear in all copies.
|
|
|
|
THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS
|
|
ALL WARRANTIES WITH REGARD TO THIS SOFTWARE INCLUDING ALL
|
|
IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS. IN NO
|
|
EVENT SHALL THE AUTHOR BE LIABLE FOR ANY SPECIAL, DIRECT,
|
|
INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES WHATSOEVER
|
|
RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
|
|
ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION,
|
|
ARISING OUT OF OR IN CONNECTION WITH THE USE OR PERFORMANCE
|
|
OF THIS SOFTWARE.
|
|
*/
|
|
|
|
#include "vga.h"
|
|
#include <stdbool.h>
|
|
|
|
uint16_t cursor_pos = 0;
|
|
uint8_t color = 0xf0;
|
|
#define cols 80
|
|
#define rows 25
|
|
|
|
void scroll(void) {
|
|
cursor_pos -= cols * 2;
|
|
uint32_t *limit = (uint32_t *)(VGA_BUFFER + cols * (rows - 1));
|
|
for (uint32_t *i = (uint32_t *)VGA_BUFFER; i < limit; ++i)
|
|
*i = *(i + cols / 2);
|
|
limit += cols / 2;
|
|
uint32_t fill = (color * 0x0100 + (uint8_t)' ') * 0x00010001;
|
|
for (uint32_t *i = limit - cols / 2; i < limit; ++i)
|
|
*i = fill;
|
|
}
|
|
|
|
void put_char(uint8_t ch) {
|
|
switch (ch) {
|
|
case '\b':
|
|
if (cursor_pos)
|
|
VGA_BUFFER[cursor_pos -= 2] = (uint8_t)' ';
|
|
break;
|
|
case '\t':
|
|
while ((cursor_pos % (cols * 2)) % 20)
|
|
put_char(' ');
|
|
break;
|
|
case '\n':
|
|
if ((cursor_pos = (cursor_pos / (cols * 2) + 1) * (cols * 2)) >= cols * rows * 2)
|
|
scroll();
|
|
break;
|
|
default:
|
|
VGA_BUFFER[cursor_pos++] = ch;
|
|
VGA_BUFFER[cursor_pos++] = color;
|
|
if (cursor_pos == (cols * rows * 2))
|
|
scroll();
|
|
}
|
|
}
|
|
|
|
void put_sz(uint8_t *sz) {
|
|
while (*sz)
|
|
put_char(*(sz++));
|
|
}
|
|
|
|
void put_16(uint16_t n) {
|
|
bool s = false;
|
|
if (n / 10000) {
|
|
put_char((uint8_t)'0' + n / 10000);
|
|
s = true;
|
|
}
|
|
n %= 10000;
|
|
if (n / 1000 || s) {
|
|
put_char((uint8_t)'0' + n / 1000);
|
|
s = true;
|
|
}
|
|
n %= 1000;
|
|
if (n / 100 || s) {
|
|
put_char((uint8_t)'0' + n / 100);
|
|
s = true;
|
|
}
|
|
n %= 100;
|
|
if (n / 10 || s) {
|
|
put_char((uint8_t)'0' + n / 10);
|
|
s = true;
|
|
}
|
|
put_char((uint8_t)'0' + n % 10);
|
|
}
|
|
|
|
void put_32_hex(uint32_t n) {
|
|
for (uint8_t i = 0; i < 4; ++i) {
|
|
put_char(n / 0xf0000000 + (n < 0xa0000000 ? (uint8_t)'0' : (uint8_t)'a' - 1));
|
|
n <<= 4;
|
|
}
|
|
put_char('_');
|
|
for (uint8_t i = 0; i < 4; ++i) {
|
|
put_char(n / 0xf0000000 + (n < 0xa0000000 ? (uint8_t)'0' : (uint8_t)'a' - 1));
|
|
n <<= 4;
|
|
}
|
|
}
|
|
|
|
void move_cursor(uint8_t col, uint8_t row) {
|
|
cursor_pos = (col + row * cols) * 2;
|
|
}
|
|
|
|
void set_color(uint8_t c) {
|
|
color = c;
|
|
}
|