2019-01-14 14:31:49 +01:00
|
|
|
//----------------------------------------------------------------------------//
|
|
|
|
// GNU GPL OS/K //
|
|
|
|
// //
|
2019-02-06 14:07:38 +01:00
|
|
|
// Desc: VGA terminal functions //
|
2019-02-16 23:36:33 +01:00
|
|
|
// //
|
|
|
|
// //
|
|
|
|
// Copyright © 2018-2019 The OS/K Team //
|
|
|
|
// //
|
|
|
|
// This file is part of OS/K. //
|
|
|
|
// //
|
|
|
|
// OS/K is free software: you can redistribute it and/or modify //
|
|
|
|
// it under the terms of the GNU General Public License as published by //
|
|
|
|
// the Free Software Foundation, either version 3 of the License, or //
|
|
|
|
// any later version. //
|
|
|
|
// //
|
|
|
|
// OS/K is distributed in the hope that it will be useful, //
|
|
|
|
// but WITHOUT ANY WARRANTY//without even the implied warranty of //
|
|
|
|
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the //
|
|
|
|
// GNU General Public License for more details. //
|
|
|
|
// //
|
|
|
|
// You should have received a copy of the GNU General Public License //
|
|
|
|
// along with OS/K. If not, see <https://www.gnu.org/licenses/>. //
|
2019-01-14 14:31:49 +01:00
|
|
|
//----------------------------------------------------------------------------//
|
|
|
|
|
2019-02-16 23:36:33 +01:00
|
|
|
#include <kernel/term.h>
|
2019-01-14 14:31:49 +01:00
|
|
|
|
|
|
|
//----------------------------------------------------------//
|
|
|
|
// Internal functions for VGA terminals //
|
|
|
|
// These DO NOT check input correctness //
|
|
|
|
//----------------------------------------------------------//
|
|
|
|
|
|
|
|
//
|
|
|
|
// VGA-related macros
|
|
|
|
//
|
|
|
|
#define VGA_ComputeColorCode(fg, bg) ((fg) | (bg) << 4)
|
|
|
|
#define VGA_ComputeOffset(term, x, y) ((y) * (term)->width + (x))
|
|
|
|
#define VGA_ComputeEntry(ch, cl) (((ushort)(ch)) | (ushort)(cl) << 8)
|
|
|
|
|
|
|
|
//
|
2019-03-11 19:22:37 +01:00
|
|
|
// Clear terminal
|
2019-01-14 14:31:49 +01:00
|
|
|
//
|
|
|
|
error_t VGA_ClearTermUnlocked(Terminal_t *term)
|
|
|
|
{
|
|
|
|
const uchar color = VGA_ComputeColorCode(term->fgColor, term->bgColor);
|
2019-03-11 19:22:37 +01:00
|
|
|
const ushort filler = VGA_ComputeEntry(' ', color);
|
2019-01-14 14:31:49 +01:00
|
|
|
const size_t bufsize = term->width * term->height;
|
|
|
|
|
|
|
|
// Fill the buffer
|
|
|
|
memsetw((ushort *)term->data, filler, bufsize);
|
|
|
|
|
|
|
|
// XXX update cursor too
|
|
|
|
term->currentX = term->currentY = 0;
|
|
|
|
|
|
|
|
return EOK;
|
|
|
|
}
|
|
|
|
|
|
|
|
//
|
|
|
|
// Write a single character on the terminal
|
|
|
|
//
|
|
|
|
error_t VGA_PutOnTermUnlocked(Terminal_t *term, char ch)
|
|
|
|
{
|
|
|
|
uint i;
|
|
|
|
size_t prevY;
|
|
|
|
|
|
|
|
if (ch == '\r') {
|
|
|
|
term->currentX = 0;
|
|
|
|
return EOK;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Line feed first takes us to the very end of the line
|
|
|
|
// Later in this function we actually do the line feed
|
|
|
|
else if (ch == '\n') {
|
2019-03-11 19:22:37 +01:00
|
|
|
term->currentX = term->width - 1;
|
2019-01-14 14:31:49 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
// Tabulations account for "term->tabSize" spaces
|
|
|
|
else if (ch == '\t') {
|
2019-03-11 19:22:37 +01:00
|
|
|
prevY = term->currentX;
|
2019-01-14 14:31:49 +01:00
|
|
|
for (i = 0; i < term->tabSize; i++) {
|
|
|
|
// Make sure tabulations can't spread over two lines
|
2019-03-11 19:22:37 +01:00
|
|
|
if (term->currentX == prevY) {
|
2019-01-14 14:31:49 +01:00
|
|
|
VGA_PutOnTermUnlocked(term, ' ');
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
else {
|
|
|
|
ushort *buffer = (ushort *)term->data;
|
2019-03-11 19:22:37 +01:00
|
|
|
const size_t offset = VGA_ComputeOffset(term, term->currentX, term->currentY);
|
2019-01-14 14:31:49 +01:00
|
|
|
buffer[offset] = VGA_ComputeEntry(ch, VGA_ComputeColorCode(term->fgColor, term->bgColor));
|
|
|
|
}
|
|
|
|
|
|
|
|
// Did we reach the end of line?
|
|
|
|
if (++term->currentX == term->width) {
|
|
|
|
term->currentX = 0;
|
|
|
|
|
|
|
|
// Did we reach the buffer's end?
|
|
|
|
if (++term->currentY == term->height) {
|
|
|
|
// XXX scroll up
|
|
|
|
term->currentY = 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Nothing can go wrong
|
|
|
|
return EOK;
|
|
|
|
}
|
|
|
|
|
|
|
|
//
|
|
|
|
// Print string on terminal
|
|
|
|
//
|
|
|
|
error_t VGA_PrintOnTermUnlocked(Terminal_t *term, const char *str)
|
|
|
|
{
|
|
|
|
error_t retcode = EOK;
|
|
|
|
|
|
|
|
while (*str && retcode == EOK) {
|
|
|
|
retcode = term->PutOnTermUnlocked(term, *str++);
|
|
|
|
}
|
|
|
|
|
|
|
|
return retcode;
|
|
|
|
}
|
|
|
|
|
|
|
|
//
|
|
|
|
// VGA output
|
2019-02-06 14:07:38 +01:00
|
|
|
// XXX custom sizes
|
2019-01-14 14:31:49 +01:00
|
|
|
//
|
2019-02-06 14:07:38 +01:00
|
|
|
Terminal_t VGA_Terminal = {
|
2019-01-14 14:31:49 +01:00
|
|
|
.initDone = FALSE,
|
|
|
|
.lock = INITLOCK(KLOCK_MUTEX),
|
|
|
|
|
|
|
|
.name = "VGA Output Terminal",
|
|
|
|
.type = "VGA",
|
|
|
|
|
|
|
|
.data = (void *)0xB8000,
|
|
|
|
.width = 80,
|
|
|
|
.height = 25,
|
|
|
|
.currentX = 0,
|
|
|
|
.currentY = 0,
|
|
|
|
|
|
|
|
.tabSize = KTABSIZE,
|
|
|
|
.fgColor = KTERM_COLOR_LGREY,
|
|
|
|
.bgColor = KTERM_COLOR_BLACK,
|
|
|
|
|
|
|
|
.ClearTermUnlocked = VGA_ClearTermUnlocked,
|
|
|
|
.PutOnTermUnlocked = VGA_PutOnTermUnlocked,
|
|
|
|
.PrintOnTermUnlocked = VGA_PrintOnTermUnlocked,
|
|
|
|
};
|
|
|
|
|
2019-02-06 14:07:38 +01:00
|
|
|
|
|
|
|
//
|
|
|
|
// Initialize VGA output
|
|
|
|
//
|
|
|
|
void VGA_Init(void)
|
|
|
|
{
|
|
|
|
KalAssert(VGA_Terminal.initDone != INITOK);
|
|
|
|
|
|
|
|
VGA_Terminal.initDone = INITOK;
|
|
|
|
}
|
|
|
|
|