rtc timestamp, knob rand, random terminal color

This commit is contained in:
Benji Dial 2021-02-16 22:26:47 -05:00
parent 2de07a2abe
commit ab4e1cfc8c
15 changed files with 154 additions and 26 deletions

View file

@ -57,6 +57,7 @@ table 1:
find unread ipc | 0x16 | sending task | | | | | find unread ipc | 0x16 | sending task | | | | |
wait ipc read | 0x17 | | reading task | | | | wait ipc read | 0x17 | | reading task | | | |
is task running | 0x18 | boolean | task handle | | | | is task running | 0x18 | boolean | task handle | | | |
get timestamp | 0x19 | secs since 2000 | | | | |
table 2: table 2:

View file

@ -1,3 +1 @@
set _path bin/ set _path bin/
set _color_fg 10
set _color_bg 61

View file

@ -53,7 +53,8 @@ out/kernel.bin: obj/kernel/drive.ko obj/kernel/fat.ko obj/kernel/ide.ko \
obj/kernel/panic.ko obj/kernel/pci.ko obj/kernel/elf.ko \ obj/kernel/panic.ko obj/kernel/pci.ko obj/kernel/elf.ko \
obj/kernel/serial.ko obj/kernel/task.ko obj/kernel/util.ko \ obj/kernel/serial.ko obj/kernel/task.ko obj/kernel/util.ko \
obj/kernel/window.ko obj/kernel/isrs.kao obj/kernel/kbd.ko \ obj/kernel/window.ko obj/kernel/isrs.kao obj/kernel/kbd.ko \
obj/kernel/pmap.ko obj/kernel/paging.ko obj/kernel/dump.ko obj/kernel/pmap.ko obj/kernel/paging.ko obj/kernel/dump.ko \
obj/kernel/cmos.ko
mkdir -p out mkdir -p out
ld -T src/kernel/elf-link.ld $^ -o obj/kernel.elf ld -T src/kernel/elf-link.ld $^ -o obj/kernel.elf
objcopy -O binary obj/kernel.elf out/kernel.bin objcopy -O binary obj/kernel.elf out/kernel.bin
@ -80,7 +81,7 @@ obj/c.rto: obj/runtimes/c/pcrt.ao
obj/cpp.rto: obj/cpp.rto:
#TODO #TODO
obj/knob.so: obj/knob/file.o obj/knob/format.o \ obj/knob.so: obj/knob/file.o obj/knob/format.o obj/knob/rand.o \
obj/knob/heap.o obj/knob/ipc.o obj/knob/task.o \ obj/knob/heap.o obj/knob/ipc.o obj/knob/task.o \
obj/knob/block.o obj/knob/key.o obj/knob/panic.o obj/knob/block.o obj/knob/key.o obj/knob/panic.o
ld ${partlink} $^ -o $@ ld ${partlink} $^ -o $@

34
src/kernel/cmos.c Normal file
View file

@ -0,0 +1,34 @@
#include "cmos.h"
#include "util.h"
static inline uint8_t read_cmos(uint8_t reg) {
outb(0x0070, reg);
return inb(0x0071);
}
//The idea of reading until you get the same value twice, and checking the status register
// is from the OSDev Wiki page at <https://wiki.osdev.org/CMOS#The_Real-Time_Clock>.
struct rtc_time get_rtc_time() {
uint8_t prev_sec = -1;
struct rtc_time ret;
get_sec:
while (read_cmos(0x0a) & 0x80)
;//spin while rtc is being updated
ret.seconds = read_cmos(0x00);
if (ret.seconds != prev_sec) {
prev_sec = ret.seconds;
goto get_sec;
}
ret.minutes = read_cmos(0x02);
ret.hours = read_cmos(0x04);
if (ret.hours & 0x80) {
ret.hours -= 0x80 - 12;
if (ret.hours == 24)
ret.hours = 0;
}
ret.day_of_week = read_cmos(0x06);
ret.day_of_month = read_cmos(0x07);
ret.month = read_cmos(0x08);
ret.year = read_cmos(0x09);
return ret;
}

18
src/kernel/cmos.h Normal file
View file

@ -0,0 +1,18 @@
#ifndef CMOS_H
#define CMOS_H
#include <stdint.h>
struct rtc_time {
uint8_t seconds;
uint8_t minutes;
uint8_t hours;
uint8_t day_of_week;
uint8_t day_of_month;
uint8_t month;
uint8_t year;
};
struct rtc_time get_rtc_time();
#endif

View file

@ -2,6 +2,7 @@
#include "window.h" #include "window.h"
#include "drive.h" #include "drive.h"
#include "panic.h" #include "panic.h"
#include "cmos.h"
#include "pmap.h" #include "pmap.h"
#include "task.h" #include "task.h"
#include "util.h" #include "util.h"
@ -141,6 +142,21 @@ uint32_t sc_is_task_running(uint32_t handle) {
return tasks[handle - 1].page_directory ? 1 : 0; return tasks[handle - 1].page_directory ? 1 : 0;
} }
static const uint16_t days_into_four_years_per_month[] = {
0, 31, 60, 91, 121, 152, 182, 213, 244, 274, 305, 335,
366, 366+31, 366+59, 366+90, 366+120, 366+151, 366+181, 366+212, 366+243, 366+273, 366+304, 366+334,
731, 731+31, 731+59, 731+90, 731+120, 731+151, 731+181, 731+212, 731+243, 731+273, 731+304, 731+334,
1096, 1096+31, 1096+59, 1096+90, 1096+120, 1096+151, 1096+181, 1096+212, 1096+243, 1096+273, 1096+304, 1096+334
};
//gregorian, assumes year is in 2001 to 2099
uint32_t sc_get_timestamp() {
const struct rtc_time time = get_rtc_time();
const uint32_t secs_into_month = time.seconds + time.minutes * 60 + time.hours * 3600 + (time.day_of_month - 1) * 86400;
const uint32_t days_to_month_into_four_years = days_into_four_years_per_month[time.month + (time.year % 4) * 12 - 1];
return secs_into_month + days_to_month_into_four_years * 86400 + (time.year / 4) * (365 * 4 + 1) * 86400;
}
void const *syscall_table[] = { void const *syscall_table[] = {
&sc_open_file, &sc_open_file,
&sc_close_file, &sc_close_file,
@ -166,7 +182,8 @@ void const *syscall_table[] = {
&sc_wait_any_ipc_sent, &sc_wait_any_ipc_sent,
&find_unread_ipc, &find_unread_ipc,
&sc_wait_ipc_read, &sc_wait_ipc_read,
&sc_is_task_running &sc_is_task_running,
&sc_get_timestamp
}; };
//these aren't really void ()'s, but gcc complains if we take an address of a void, so we give it a type //these aren't really void ()'s, but gcc complains if we take an address of a void, so we give it a type

View file

@ -25,7 +25,7 @@ extern exception_halt
extern pf_check_stack extern pf_check_stack
extern dump extern dump
n_syscalls equ 0x19 n_syscalls equ 0x1a
;section .bss ;section .bss
;_debug_is_start_task resb 1 ;_debug_is_start_task resb 1

View file

@ -4,16 +4,17 @@
#include "window.h" #include "window.h"
#include "panic.h" #include "panic.h"
#include "boot.h" #include "boot.h"
#include "cmos.h"
#include "pmap.h"
#include "task.h"
#include "util.h" #include "util.h"
#include "elf.h"
#include "fat.h" #include "fat.h"
#include "ide.h" #include "ide.h"
#include "idt.h" #include "idt.h"
#include "pmap.h"
#include "task.h"
#include "pci.h"
#include "elf.h"
#include "log.h"
#include "kbd.h" #include "kbd.h"
#include "log.h"
#include "pci.h"
void _start_user_mode() __attribute__ ((noreturn)); void _start_user_mode() __attribute__ ((noreturn));

View file

@ -17,12 +17,12 @@ static inline uint8_t hex_to_int(char ch) {
} }
void ensure_color() { void ensure_color() {
const struct no_null_sn *fg = get_var((struct no_null_sn){.data = "_color_fg", .length = 9}); const struct no_null_sn *color = get_var((struct no_null_sn){.data = "_color", .length = 6});
const struct no_null_sn *bg = get_var((struct no_null_sn){.data = "_color_bg", .length = 9}); //fgbg
if (fg && bg) if (color)
term_set_color( term_set_color(
(hex_to_int(fg->data[0]) << 4) | hex_to_int(fg->data[1]), (hex_to_int(color->data[0]) << 4) | hex_to_int(color->data[1]),
(hex_to_int(bg->data[0]) << 4) | hex_to_int(bg->data[1]) (hex_to_int(color->data[2]) << 4) | hex_to_int(color->data[3])
); );
} }
@ -100,13 +100,19 @@ void run_line(const char *original_line) {
term_clear(); term_clear();
term_paint(); term_paint();
} }
else if (blockequ(line, "color", 6)) {
new_color();
ensure_color();
term_paint();
}
else if (blockequ(line, "help", 5)) { else if (blockequ(line, "help", 5)) {
term_add_sz("Highway is a command shell for Portland OS. It includes variable support and a couple of pseudo-commands. Variables are addressed by surrounding with \"$\". The following list shows each of the pseudo-commands.\n\n" term_add_sz("Highway is a command shell for Portland OS. It includes variable support and a couple of pseudo-commands. Variables are addressed by surrounding with \"$\". The following list shows each of the pseudo-commands.\n\n"
" source FILE\t" "run each command in FILE\n" " source FILE\t" "run each command in FILE\n"
" clear\t\t\t" "clear the screen\n"
" echo STRING\t" "print STRING\n"
" set VAR VALUE\t" "set $VAR$ to VALUE\n" " set VAR VALUE\t" "set $VAR$ to VALUE\n"
" color\t\t\t" "new color scheme\n"
" clear\t\t\t" "clear the screen\n"
" vars\t\t\t" "dump variables\n" " vars\t\t\t" "dump variables\n"
" echo STRING\t" "print STRING\n"
" quit\t\t\t" "exit highway\n" " quit\t\t\t" "exit highway\n"
" help\t\t\t" "show this\n"); " help\t\t\t" "show this\n");
term_paint(); term_paint();

View file

@ -5,12 +5,10 @@
#include "cmds.h" #include "cmds.h"
#include "line.h" #include "line.h"
#include "vars.h"
void main(const char *arg) { void main(const char *arg) {
//syslogf(" this task: 0x%2h", this_task); new_color();
//syslogf(" stdio task: 0x%2h", stdio_task);
//syslogf("calling task: 0x%2h", calling_task);
source(*arg ? arg : "user/default.rc"); source(*arg ? arg : "user/default.rc");
ensure_color(); ensure_color();

View file

@ -3,6 +3,7 @@
#include <knob/format.h> #include <knob/format.h>
#include <knob/block.h> #include <knob/block.h>
#include <knob/heap.h> #include <knob/heap.h>
#include <knob/rand.h>
#include <pland/pcrt.h> #include <pland/pcrt.h>
@ -81,3 +82,23 @@ void dump_vars() {
term_paint(); term_paint();
} }
} }
static const char hex_digits[] = "0123456789abcdef";
static char color[] = {'1', '0', '.', '.'};
static const struct no_null_sn color_name = {
.data = "_color",
.length = 6
};
static const struct no_null_sn color_value = {
.data = color,
.length = 4
};
void new_color() {
const uint8_t bg = gen_rand() % 0x30 + 0x38;
color[2] = hex_digits[bg >> 4];
color[3] = hex_digits[bg & 0xf];
set_var(color_name, color_value);
}

View file

@ -13,5 +13,6 @@ const struct no_null_sn *get_var(struct no_null_sn name) __attribute__ ((pure));
void del_var(struct no_null_sn name); void del_var(struct no_null_sn name);
void dump_vars(); void dump_vars();
void new_color();
#endif #endif

View file

@ -0,0 +1,8 @@
#ifndef RAND_H
#define RAND_H
#include <stdint.h>
uint32_t gen_rand();
#endif

View file

@ -43,7 +43,8 @@ enum _scn {
_SCN_WAIT_ANY_IPC_SENT, _SCN_WAIT_ANY_IPC_SENT,
_SCN_FIND_UNREAD_IPC, _SCN_FIND_UNREAD_IPC,
_SCN_WAIT_IPC_READ, _SCN_WAIT_IPC_READ,
_SCN_IS_TASK_RUNNING _SCN_IS_TASK_RUNNING,
_SCN_GET_TIMESTAMP
}; };
static inline uint32_t _sc0(enum _scn eax) { static inline uint32_t _sc0(enum _scn eax) {
@ -228,4 +229,8 @@ static inline bool _is_task_running(_task_handle_t handle) {
return (bool)_sc1(_SCN_IS_TASK_RUNNING, handle); return (bool)_sc1(_SCN_IS_TASK_RUNNING, handle);
} }
static inline uint32_t _get_timestamp() {
return _sc0(_SCN_GET_TIMESTAMP);
}
#endif #endif

19
src/user/knob/rand.c Normal file
View file

@ -0,0 +1,19 @@
#include <pland/syscall.h>
#include <pland/pcrt.h>
#include <stdint.h>
static uint32_t r;
static void seed_rand() {
r = _get_timestamp();
}
BEFORE_MAIN(seed_rand)
uint32_t gen_rand() {
r ^= r << 13;
r ^= r >> 17;
r ^= r << 5;
return r;
}