rtc timestamp, knob rand, random terminal color
This commit is contained in:
parent
2de07a2abe
commit
ab4e1cfc8c
15 changed files with 154 additions and 26 deletions
|
@ -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:
|
||||||
|
|
|
@ -1,3 +1 @@
|
||||||
set _path bin/
|
set _path bin/
|
||||||
set _color_fg 10
|
|
||||||
set _color_bg 61
|
|
9
makefile
9
makefile
|
@ -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,9 +81,9 @@ 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 $@
|
||||||
|
|
||||||
obj/libterm.so: obj/libterm/terminal.o obj/libterm/termtask.o obj/libterm/readline.o
|
obj/libterm.so: obj/libterm/terminal.o obj/libterm/termtask.o obj/libterm/readline.o
|
||||||
|
|
34
src/kernel/cmos.c
Normal file
34
src/kernel/cmos.c
Normal 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
18
src/kernel/cmos.h
Normal 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
|
|
@ -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
|
||||||
|
|
|
@ -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
|
||||||
|
|
|
@ -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));
|
||||||
|
|
||||||
|
|
|
@ -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();
|
||||||
|
|
|
@ -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();
|
||||||
|
|
||||||
|
|
|
@ -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>
|
||||||
|
|
||||||
|
@ -80,4 +81,24 @@ void dump_vars() {
|
||||||
term_addf_no_ww("$%ns$\t= %ns\n", node->name.length, node->name.data, node->value.length, node->value.data);
|
term_addf_no_ww("$%ns$\t= %ns\n", node->name.length, node->name.data, node->value.length, node->value.data);
|
||||||
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);
|
||||||
}
|
}
|
|
@ -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
|
8
src/user/include/knob/rand.h
Normal file
8
src/user/include/knob/rand.h
Normal file
|
@ -0,0 +1,8 @@
|
||||||
|
#ifndef RAND_H
|
||||||
|
#define RAND_H
|
||||||
|
|
||||||
|
#include <stdint.h>
|
||||||
|
|
||||||
|
uint32_t gen_rand();
|
||||||
|
|
||||||
|
#endif
|
|
@ -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
19
src/user/knob/rand.c
Normal 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;
|
||||||
|
}
|
Reference in a new issue