file manager
This commit is contained in:
parent
20853582d5
commit
de20d7430d
30 changed files with 603 additions and 234 deletions
43
doc/ints.txt
43
doc/ints.txt
|
@ -24,19 +24,23 @@ invalid system call numbers change eax to -1, and have no other effect.
|
||||||
|
|
||||||
table 1:
|
table 1:
|
||||||
|
|
||||||
function | eax | eax out | ebx | ecx | edx | esi | edi
|
function | eax | eax out | ebx | ecx | edx | esi | edi
|
||||||
---------------|-----|---------------|---------------|-------------|-------------|--------|-----
|
----------------|-----|-----------------|----------------|-------------|-------------|--------|-----
|
||||||
open file | 0x0 | handle | drive number | path | | |
|
open file | 0x0 | handle | drive number | path | | |
|
||||||
close file | 0x1 | | handle | | | |
|
close file | 0x1 | | handle | | | |
|
||||||
file read | 0x2 | read | handle | file offset | count | buffer |
|
file read | 0x2 | read | handle | file offset | count | buffer |
|
||||||
get file size | 0x3 | size | handle | | | |
|
get file size | 0x3 | size | handle | | | |
|
||||||
start task | 0x4 | handle | drive number | path | passed sz | |
|
start task | 0x4 | handle | drive number | path | passed sz | |
|
||||||
log string | 0x5 | | sz string | | | |
|
log string | 0x5 | | sz string | | | |
|
||||||
get key | 0x6 | keycode | | | | |
|
get key | 0x6 | keycode | | | | |
|
||||||
allocate ram | 0x7 | start pointer | pages | | | |
|
allocate ram | 0x7 | start pointer | pages | | | |
|
||||||
memory info | 0x8 | see table 2 | see table 2 | | | |
|
memory info | 0x8 | see table 2 | see table 2 | | | |
|
||||||
wait for task | 0x9 | | handle | | | |
|
wait for task | 0x9 | | handle | | | |
|
||||||
enumerate dir | 0xa | count | drive number | path | see table 3 | max |
|
enumerate dir | 0xa | count | drive number | path | see table 3 | max |
|
||||||
|
vga print at | 0xb | | row << 8 | col | sz string | | |
|
||||||
|
count of dir | 0xc | number of files | drive number | path | | |
|
||||||
|
clear screen | 0xd | | | | | |
|
||||||
|
set color | 0xe | | VGA color code | | | |
|
||||||
|
|
||||||
|
|
||||||
table 2:
|
table 2:
|
||||||
|
@ -55,12 +59,7 @@ eax -1 indicates unrecognized ebx
|
||||||
table 3:
|
table 3:
|
||||||
edx of "enumerate dir" is a pointer to a buffer where an array of "directory info"s can be put.
|
edx of "enumerate dir" is a pointer to a buffer where an array of "directory info"s can be put.
|
||||||
esi is the maximum number of "directory info"s to be placed in the buffer.
|
esi is the maximum number of "directory info"s to be placed in the buffer.
|
||||||
the "directory info" structure is defined as follows:
|
the "directory info" structure is 128 bytes long. The first 100 bytes are a null-terminated string
|
||||||
struct {
|
indicating the name of the file represented by this entry. The next four bytes are a 32-bit integer
|
||||||
bool is_dir;
|
indicating the size of the file represented in bytes. The next byte's lowest bit is a one if this
|
||||||
char name[100];
|
entry is a directory, and a zero if this entry is not. The remainder of the structure is reserved.
|
||||||
uint32_t size;
|
|
||||||
}
|
|
||||||
"is_dir" here indicates whether or not the entry is another directory.
|
|
||||||
"name" is a null-terminated string, up to 99 characters, indicating the name of the entry.
|
|
||||||
"size" is the size of the file this entry refers to, in bytes.
|
|
68
makefile
68
makefile
|
@ -1,5 +1,6 @@
|
||||||
kgccargs = -Wall -Wsuggest-attribute=pure -Wsuggest-attribute=const -m32 -Og -ffreestanding -fno-asynchronous-unwind-tables
|
kgccargs = -Wall -Wsuggest-attribute=pure -Wsuggest-attribute=const -m32 -Og -ffreestanding -fno-asynchronous-unwind-tables
|
||||||
ugccargs = ${kgccargs} -Isrc/user/include
|
ugccargs = ${kgccargs} -Isrc/user/include
|
||||||
|
ugppargs = ${kgccargs} -Isrc/user/include/cpp -Isrc/user/include
|
||||||
nasmargs = -f elf32
|
nasmargs = -f elf32
|
||||||
partlink = -r -m elf_i386
|
partlink = -r -m elf_i386
|
||||||
|
|
||||||
|
@ -30,7 +31,7 @@ out/fs/bin/%: obj/%.elf
|
||||||
|
|
||||||
out/fs: out/fs/bin/init out/fs/bin/meminfo out/fs/bin/highway \
|
out/fs: out/fs/bin/init out/fs/bin/meminfo out/fs/bin/highway \
|
||||||
out/fs/bin/hello out/fs/bin/dumptext out/fs/bin/dumphex \
|
out/fs/bin/hello out/fs/bin/dumptext out/fs/bin/dumphex \
|
||||||
out/fs/bin/dirinfo
|
out/fs/bin/dirinfo out/fs/bin/fileman
|
||||||
mkdir -p out/fs
|
mkdir -p out/fs
|
||||||
cp -r fs-skel/* out/fs/
|
cp -r fs-skel/* out/fs/
|
||||||
|
|
||||||
|
@ -42,14 +43,14 @@ obj/kernel/%.kao: src/kernel/%.asm
|
||||||
mkdir -p $(shell dirname $@)
|
mkdir -p $(shell dirname $@)
|
||||||
nasm ${nasmargs} $< -o $@
|
nasm ${nasmargs} $< -o $@
|
||||||
|
|
||||||
out/kernel.bin: obj/kernel/drive.ko obj/kernel/fat.ko obj/kernel/ide.ko \
|
out/kernel.bin: obj/kernel/drive.ko obj/kernel/fat.ko obj/kernel/ide.ko \
|
||||||
obj/kernel/idt.ko obj/kernel/log.ko obj/kernel/main.ko \
|
obj/kernel/idt.ko obj/kernel/log.ko obj/kernel/main.ko \
|
||||||
obj/kernel/main2.ko obj/kernel/pmap.ko obj/kernel/paging.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/vga.ko obj/kernel/isrs.kao obj/kernel/kbd.ko \
|
||||||
obj/kernel/vga.ko obj/kernel/isrs.kao obj/kernel/kbd.ko
|
obj/kernel/pmap.ko obj/kernel/paging.ko
|
||||||
mkdir -p out
|
mkdir -p out
|
||||||
ld -T src/kernel/elf-link.ld obj/kernel/* -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
|
||||||
|
|
||||||
out/boot.bin: src/boot.asm
|
out/boot.bin: src/boot.asm
|
||||||
|
@ -64,29 +65,46 @@ obj/%.ao: src/user/%.asm
|
||||||
mkdir -p $(shell dirname $@)
|
mkdir -p $(shell dirname $@)
|
||||||
nasm ${nasmargs} $< -o $@
|
nasm ${nasmargs} $< -o $@
|
||||||
|
|
||||||
obj/knob.so: obj/knob/env.o obj/knob/file.o obj/knob/format.o \
|
obj/%.po: src/user/%.cpp
|
||||||
obj/knob/heap.o obj/knob/quit.o obj/knob/user.o \
|
mkdir -p $(shell dirname $@)
|
||||||
obj/knob/task.o obj/knob/entry.ao obj/knob/block.o
|
g++ ${ugppargs} -c $< -o $@
|
||||||
ld ${partlink} obj/knob/* -o obj/knob.so
|
|
||||||
|
|
||||||
obj/init.elf: obj/init/init.o obj/knob.so
|
obj/c.rto: obj/runtimes/c/entry.ao
|
||||||
ld -T src/user/elf.ld obj/init/* obj/knob.so -o obj/init.elf
|
ld ${partlink} obj/runtimes/c/* -o obj/c.rto
|
||||||
|
|
||||||
obj/meminfo.elf: obj/meminfo/meminfo.o obj/knob.so
|
obj/cpp.rto:
|
||||||
ld -T src/user/elf.ld obj/meminfo/* obj/knob.so -o obj/meminfo.elf
|
#TODO
|
||||||
|
|
||||||
|
obj/knob.so: obj/knob/env.o obj/knob/file.o obj/knob/format.o \
|
||||||
|
obj/knob/heap.o obj/knob/user.o obj/knob/task.o \
|
||||||
|
obj/knob/block.o
|
||||||
|
ld ${partlink} $^ -o $@
|
||||||
|
|
||||||
|
obj/cove.so: obj/cove/cove.po obj/cove/buttons.po obj/cove/label.po \
|
||||||
|
obj/cove/vlist.po
|
||||||
|
ld ${partlink} $^ -o $@
|
||||||
|
|
||||||
|
obj/init.elf: obj/init/init.o obj/knob.so obj/c.rto
|
||||||
|
ld -T src/user/runtimes/c/elf.ld $^ -o $@
|
||||||
|
|
||||||
|
obj/meminfo.elf: obj/meminfo/meminfo.o obj/knob.so obj/c.rto
|
||||||
|
ld -T src/user/runtimes/c/elf.ld $^ -o $@
|
||||||
|
|
||||||
obj/highway.elf: obj/highway/main.o obj/highway/cmds.o obj/highway/line.o \
|
obj/highway.elf: obj/highway/main.o obj/highway/cmds.o obj/highway/line.o \
|
||||||
obj/highway/vars.o obj/knob.so
|
obj/highway/vars.o obj/knob.so obj/c.rto
|
||||||
ld -T src/user/elf.ld obj/highway/* obj/knob.so -o obj/highway.elf
|
ld -T src/user/runtimes/c/elf.ld $^ -o $@
|
||||||
|
|
||||||
obj/hello.elf: obj/hello/hello.ao
|
obj/hello.elf: obj/hello/hello.ao
|
||||||
ld -T src/user/elf.ld obj/hello/* -o obj/hello.elf
|
ld -T src/user/runtimes/asm/elf.ld $^ -o $@
|
||||||
|
|
||||||
obj/dumptext.elf: obj/dumptext/dumptext.o obj/knob.so
|
obj/dumptext.elf: obj/dumptext/dumptext.o obj/knob.so obj/c.rto
|
||||||
ld -T src/user/elf.ld obj/dumptext/* obj/knob.so -o obj/dumptext.elf
|
ld -T src/user/runtimes/c/elf.ld $^ -o $@
|
||||||
|
|
||||||
obj/dumphex.elf: obj/dumphex/dumphex.o obj/knob.so
|
obj/dumphex.elf: obj/dumphex/dumphex.o obj/knob.so obj/c.rto
|
||||||
ld -T src/user/elf.ld obj/dumphex/* obj/knob.so -o obj/dumphex.elf
|
ld -T src/user/runtimes/c/elf.ld $^ -o $@
|
||||||
|
|
||||||
obj/dirinfo.elf: obj/dirinfo/dirinfo.o obj/knob.so
|
obj/dirinfo.elf: obj/dirinfo/dirinfo.o obj/knob.so obj/c.rto
|
||||||
ld -T src/user/elf.ld obj/dirinfo/* obj/knob.so -o obj/dirinfo.elf
|
ld -T src/user/runtimes/c/elf.ld $^ -o $@
|
||||||
|
|
||||||
|
obj/fileman.elf: obj/fileman/fileman.o obj/knob.so obj/c.rto
|
||||||
|
ld -T src/user/runtimes/c/elf.ld $^ -o $@
|
|
@ -36,6 +36,11 @@ static uint32_t unknown_enumerate_dir(const struct drive *d, const char *path, s
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
__attribute__ ((const))
|
||||||
|
static uint32_t unknown_n_dir_entries(const struct drive *d, const char *path) {
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
static inline void determine_fs(struct drive *d) {
|
static inline void determine_fs(struct drive *d) {
|
||||||
if (try_fat_init_drive(d))
|
if (try_fat_init_drive(d))
|
||||||
return;
|
return;
|
||||||
|
@ -46,6 +51,7 @@ static inline void determine_fs(struct drive *d) {
|
||||||
d->load_sector = &unknown_load_sector;
|
d->load_sector = &unknown_load_sector;
|
||||||
d->get_file_length = &unknown_get_file_length;
|
d->get_file_length = &unknown_get_file_length;
|
||||||
d->enumerate_dir = &unknown_enumerate_dir;
|
d->enumerate_dir = &unknown_enumerate_dir;
|
||||||
|
d->n_dir_entries = &unknown_n_dir_entries;
|
||||||
d->get_free_sectors = &unknown_get_free_sectors;
|
d->get_free_sectors = &unknown_get_free_sectors;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -13,10 +13,11 @@ typedef uint8_t drive_id_t;
|
||||||
#define DCI_NAME_LEN 100
|
#define DCI_NAME_LEN 100
|
||||||
|
|
||||||
struct directory_content_info {
|
struct directory_content_info {
|
||||||
bool is_dir;
|
|
||||||
char name[DCI_NAME_LEN];
|
char name[DCI_NAME_LEN];
|
||||||
uint32_t size;
|
uint32_t size;
|
||||||
};
|
bool is_dir;
|
||||||
|
uint8_t pad[23];
|
||||||
|
} __attribute__ ((packed));
|
||||||
|
|
||||||
struct drive {
|
struct drive {
|
||||||
char *drive_type;
|
char *drive_type;
|
||||||
|
@ -34,6 +35,7 @@ struct drive {
|
||||||
void (*load_sector) (const struct drive *d, file_id_t fid, uint32_t sector, void *at);
|
void (*load_sector) (const struct drive *d, file_id_t fid, uint32_t sector, void *at);
|
||||||
uint32_t (*get_file_length) (const struct drive *d, file_id_t fid);
|
uint32_t (*get_file_length) (const struct drive *d, file_id_t fid);
|
||||||
uint32_t (*enumerate_dir) (const struct drive *d, const char *path, struct directory_content_info *info, uint32_t max);
|
uint32_t (*enumerate_dir) (const struct drive *d, const char *path, struct directory_content_info *info, uint32_t max);
|
||||||
|
uint32_t (*n_dir_entries) (const struct drive *d, const char *path);
|
||||||
uint32_t (*get_free_sectors)(const struct drive *d);
|
uint32_t (*get_free_sectors)(const struct drive *d);
|
||||||
fs_id_t fs_id;
|
fs_id_t fs_id;
|
||||||
};
|
};
|
||||||
|
|
|
@ -365,6 +365,72 @@ static uint32_t fat_enumerate_dir(const struct drive *d, const char *path, struc
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static uint32_t n_root_entries(const struct drive *d) {
|
||||||
|
uint32_t sect = infos[d->drive_id].root_start - 1;
|
||||||
|
struct directory_entry *entry = (struct directory_entry *)(fat_driver_buffer + 512);
|
||||||
|
|
||||||
|
uint32_t count = 0;
|
||||||
|
|
||||||
|
while (true) {
|
||||||
|
if (entry == (struct directory_entry *)(fat_driver_buffer + 512)) {
|
||||||
|
entry = (struct directory_entry *)fat_driver_buffer;
|
||||||
|
++sect;
|
||||||
|
d->read_sectors(d, sect, 1, entry);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!*(uint8_t *)entry) {
|
||||||
|
d->done(d);
|
||||||
|
return count;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (entry-> attrib & FA_LABEL) {
|
||||||
|
++entry;
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
++entry;
|
||||||
|
++count;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static uint32_t fat_n_dir_entries(const struct drive *d, const char *path) {
|
||||||
|
d->ready(d);
|
||||||
|
|
||||||
|
if (!*path)
|
||||||
|
return n_root_entries(d);
|
||||||
|
|
||||||
|
if (!try_load_from_path(d, path)) {
|
||||||
|
d->done(d);
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
uint16_t cluster = cur_dir->first_cluster;
|
||||||
|
load_cluster(cluster, fat_driver_buffer);
|
||||||
|
struct directory_entry *entry = (struct directory_entry *)fat_driver_buffer;
|
||||||
|
|
||||||
|
uint32_t count = 0;
|
||||||
|
|
||||||
|
while (true) {
|
||||||
|
if (entry == (struct directory_entry *)(fat_driver_buffer + 512)) {
|
||||||
|
entry = (struct directory_entry *)fat_driver_buffer;
|
||||||
|
load_cluster(cluster = next_cluster(cluster), fat_driver_buffer);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!*(uint8_t *)entry) {
|
||||||
|
d->done(d);
|
||||||
|
return count;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (check_fat_names(entry->name, this_dir) || check_fat_names(entry->name, parent_dir)) {
|
||||||
|
++entry;
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
++entry;
|
||||||
|
++count;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
void init_fat() {
|
void init_fat() {
|
||||||
next_fi = allocate_kernel_pages(1);
|
next_fi = allocate_kernel_pages(1);
|
||||||
next_id = 0;
|
next_id = 0;
|
||||||
|
@ -388,6 +454,7 @@ bool try_fat_init_drive(struct drive *d) {
|
||||||
d->load_sector = &fat_load_sector;
|
d->load_sector = &fat_load_sector;
|
||||||
d->get_file_length = &fat_get_file_length;
|
d->get_file_length = &fat_get_file_length;
|
||||||
d->enumerate_dir = &fat_enumerate_dir;
|
d->enumerate_dir = &fat_enumerate_dir;
|
||||||
|
d->n_dir_entries = &fat_n_dir_entries;
|
||||||
d->get_free_sectors = &fat_get_free_sectors;
|
d->get_free_sectors = &fat_get_free_sectors;
|
||||||
|
|
||||||
d->fs_id = next_id;
|
d->fs_id = next_id;
|
||||||
|
|
|
@ -8,6 +8,7 @@
|
||||||
#include "paging.h"
|
#include "paging.h"
|
||||||
#include "pmap.h"
|
#include "pmap.h"
|
||||||
#include "kbd.h"
|
#include "kbd.h"
|
||||||
|
#include "vga.h"
|
||||||
|
|
||||||
enum {
|
enum {
|
||||||
IDT_PRESENT = 0x80,
|
IDT_PRESENT = 0x80,
|
||||||
|
@ -100,6 +101,10 @@ static uint32_t sc_enumerate_dir(uint32_t drive_number, const char *path, struct
|
||||||
return drives[drive_number].enumerate_dir(drives + drive_number, path, buffer, max_entries);
|
return drives[drive_number].enumerate_dir(drives + drive_number, path, buffer, max_entries);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static uint32_t sc_count_of_dir(uint32_t drive_number, const char *path) {
|
||||||
|
return drives[drive_number].n_dir_entries(drives + drive_number, path);
|
||||||
|
}
|
||||||
|
|
||||||
void const *syscall_table[] = {
|
void const *syscall_table[] = {
|
||||||
&sc_open_file,
|
&sc_open_file,
|
||||||
&sc_close_file,
|
&sc_close_file,
|
||||||
|
@ -111,7 +116,11 @@ void const *syscall_table[] = {
|
||||||
&sc_allocate_ram,
|
&sc_allocate_ram,
|
||||||
&sc_memory_info,
|
&sc_memory_info,
|
||||||
&sc_wait_for_task,
|
&sc_wait_for_task,
|
||||||
&sc_enumerate_dir
|
&sc_enumerate_dir,
|
||||||
|
&vga_print_at,
|
||||||
|
&sc_count_of_dir,
|
||||||
|
&vga_blank,
|
||||||
|
&vga_set_color
|
||||||
};
|
};
|
||||||
|
|
||||||
//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
|
||||||
|
|
|
@ -23,7 +23,7 @@ extern on_kbd_isr
|
||||||
extern make_sure_tasks
|
extern make_sure_tasks
|
||||||
extern exception_halt
|
extern exception_halt
|
||||||
|
|
||||||
n_syscalls equ 0xb
|
n_syscalls equ 0xf
|
||||||
|
|
||||||
;section .bss
|
;section .bss
|
||||||
;_debug_is_start_task resb 1
|
;_debug_is_start_task resb 1
|
||||||
|
|
|
@ -226,19 +226,34 @@ static const uint8_t codes_e0[] = {
|
||||||
0, 0, 0, 0, 0, 0, 0, 0
|
0, 0, 0, 0, 0, 0, 0, 0
|
||||||
};
|
};
|
||||||
|
|
||||||
|
#include "log.h"
|
||||||
|
|
||||||
|
uint8_t get_code_byte() {
|
||||||
|
for (uint32_t spin = 0; spin < 10000000; ++spin)
|
||||||
|
;
|
||||||
|
return inb(PS2_DATA);
|
||||||
|
}
|
||||||
|
|
||||||
void on_kbd_isr() {
|
void on_kbd_isr() {
|
||||||
while (inb(PS2_CMD) & PS2S_CODE_READY) {
|
while (inb(PS2_CMD) & PS2S_CODE_READY) {
|
||||||
uint8_t code = inb(PS2_DATA);
|
//char nbuf[11];
|
||||||
|
uint8_t code = get_code_byte();
|
||||||
|
//logsz("code: 0x");
|
||||||
|
//u8_hex(code, nbuf);
|
||||||
|
//logsz(nbuf);
|
||||||
if (code == 0xe1) {
|
if (code == 0xe1) {
|
||||||
code = inb(PS2_DATA);
|
code = get_code_byte();
|
||||||
|
//logsz(" 0x");
|
||||||
|
//u8_hex(code, nbuf);
|
||||||
|
//logsz(nbuf);
|
||||||
if (code == 0x1d) {
|
if (code == 0x1d) {
|
||||||
if (inb(PS2_DATA) != 0x45)
|
if (get_code_byte() != 0x45)
|
||||||
code = 0;
|
code = 0;
|
||||||
else
|
else
|
||||||
code = CODE_PAUSE;
|
code = CODE_PAUSE;
|
||||||
}
|
}
|
||||||
else if (code == 0x9d) {
|
else if (code == 0x9d) {
|
||||||
if (inb(PS2_DATA) != 0xc5)
|
if (get_code_byte() != 0xc5)
|
||||||
code = 0;
|
code = 0;
|
||||||
else
|
else
|
||||||
code = 0xff;
|
code = 0xff;
|
||||||
|
@ -247,17 +262,20 @@ void on_kbd_isr() {
|
||||||
code = 0;
|
code = 0;
|
||||||
}
|
}
|
||||||
else if (code == 0xe0) {
|
else if (code == 0xe0) {
|
||||||
code = inb(PS2_DATA);
|
code = get_code_byte();
|
||||||
|
//logsz(" 0x");
|
||||||
|
//u8_hex(code, nbuf);
|
||||||
|
//logsz(nbuf);
|
||||||
if (code == 0x2a) {
|
if (code == 0x2a) {
|
||||||
if ((inb(PS2_DATA) != 0xe0) ||
|
if ((get_code_byte() != 0xe0) ||
|
||||||
(inb(PS2_DATA) != 0x37))
|
(get_code_byte() != 0x37))
|
||||||
code = 0;
|
code = 0;
|
||||||
else
|
else
|
||||||
code = CODE_PRSCR;
|
code = CODE_PRSCR;
|
||||||
}
|
}
|
||||||
else if (code == 0xb7) {
|
else if (code == 0xb7) {
|
||||||
if ((inb(PS2_DATA) != 0xe0) ||
|
if ((get_code_byte() != 0xe0) ||
|
||||||
(inb(PS2_DATA) != 0xaa))
|
(get_code_byte() != 0xaa))
|
||||||
code = 0;
|
code = 0;
|
||||||
else
|
else
|
||||||
code = 0xff;
|
code = 0xff;
|
||||||
|
@ -268,6 +286,8 @@ void on_kbd_isr() {
|
||||||
else
|
else
|
||||||
code = codes[code];
|
code = codes[code];
|
||||||
|
|
||||||
|
//logch('\n');
|
||||||
|
|
||||||
if (!code)
|
if (!code)
|
||||||
PANIC("Unknown scancode.");
|
PANIC("Unknown scancode.");
|
||||||
|
|
||||||
|
|
|
@ -15,9 +15,6 @@
|
||||||
#include "vga.h"
|
#include "vga.h"
|
||||||
#include "kbd.h"
|
#include "kbd.h"
|
||||||
|
|
||||||
void reset_tree();
|
|
||||||
void tree(struct drive *d);
|
|
||||||
|
|
||||||
void _start_user_mode() __attribute__ ((noreturn));
|
void _start_user_mode() __attribute__ ((noreturn));
|
||||||
|
|
||||||
__attribute__ ((noreturn))
|
__attribute__ ((noreturn))
|
||||||
|
@ -35,6 +32,7 @@ void main() {
|
||||||
|
|
||||||
pci_init();
|
pci_init();
|
||||||
|
|
||||||
|
/*
|
||||||
u16_dec(n_pci_devices, nbuf);
|
u16_dec(n_pci_devices, nbuf);
|
||||||
logsz(nbuf);
|
logsz(nbuf);
|
||||||
logsz(" PCI device(s) found:\n");
|
logsz(" PCI device(s) found:\n");
|
||||||
|
@ -63,6 +61,7 @@ void main() {
|
||||||
}
|
}
|
||||||
|
|
||||||
logch('\n');
|
logch('\n');
|
||||||
|
*/
|
||||||
|
|
||||||
init_fat();
|
init_fat();
|
||||||
//other fs drivers
|
//other fs drivers
|
||||||
|
@ -72,6 +71,7 @@ void main() {
|
||||||
init_ide();
|
init_ide();
|
||||||
//other drive drivers
|
//other drive drivers
|
||||||
|
|
||||||
|
/*
|
||||||
u8_dec(n_drives, nbuf);
|
u8_dec(n_drives, nbuf);
|
||||||
logsz(nbuf);
|
logsz(nbuf);
|
||||||
logsz(" drive(s) found:\n");
|
logsz(" drive(s) found:\n");
|
||||||
|
@ -108,20 +108,7 @@ void main() {
|
||||||
}
|
}
|
||||||
|
|
||||||
logch('\n');
|
logch('\n');
|
||||||
|
*/
|
||||||
for (uint8_t n = 0; n < n_drives; ++n) {
|
|
||||||
if (drives[n].get_free_sectors(drives + n) == -1)
|
|
||||||
continue;
|
|
||||||
|
|
||||||
u32_dec(n, nbuf);
|
|
||||||
logsz("sd");
|
|
||||||
logsz(nbuf);
|
|
||||||
logsz(" tree:\n");
|
|
||||||
|
|
||||||
reset_tree();
|
|
||||||
tree(drives + n);
|
|
||||||
logch('\n');
|
|
||||||
}
|
|
||||||
|
|
||||||
if (BOOT_INFO->support_flags & BIS_PAE)
|
if (BOOT_INFO->support_flags & BIS_PAE)
|
||||||
logsz("Processor supports PAE (but Portland OS does not yet).\n");
|
logsz("Processor supports PAE (but Portland OS does not yet).\n");
|
||||||
|
|
|
@ -1,69 +0,0 @@
|
||||||
#include <stdint.h>
|
|
||||||
#include "drive.h"
|
|
||||||
#include "util.h"
|
|
||||||
#include "log.h"
|
|
||||||
|
|
||||||
static char nbuf2[11];
|
|
||||||
|
|
||||||
static char path_builder[200];
|
|
||||||
static uint8_t path_builder_len;
|
|
||||||
|
|
||||||
static char indent_builder[20];
|
|
||||||
static uint8_t indent_builder_len;
|
|
||||||
|
|
||||||
void reset_tree() {
|
|
||||||
path_builder[0] = '\0';
|
|
||||||
path_builder_len = 0;
|
|
||||||
|
|
||||||
indent_builder[0] = ' ';
|
|
||||||
indent_builder[1] = ' ';
|
|
||||||
indent_builder[2] = '\0';
|
|
||||||
indent_builder_len = 2;
|
|
||||||
}
|
|
||||||
|
|
||||||
void tree(struct drive *d) {
|
|
||||||
struct directory_content_info infos[100];
|
|
||||||
uint8_t n_infos = d->enumerate_dir(d, path_builder, infos, 100);
|
|
||||||
|
|
||||||
if (!n_infos) {
|
|
||||||
logsz(indent_builder);
|
|
||||||
logsz("(empty)\n");
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
for (uint8_t i = 0; i < n_infos; ++i) {
|
|
||||||
logsz(indent_builder);
|
|
||||||
logsz(infos[i].name);
|
|
||||||
|
|
||||||
if (infos[i].is_dir) {
|
|
||||||
logsz(":\n");
|
|
||||||
|
|
||||||
indent_builder[indent_builder_len] = ' ';
|
|
||||||
indent_builder[indent_builder_len + 1] = ' ';
|
|
||||||
indent_builder[indent_builder_len + 2] = '\0';
|
|
||||||
indent_builder_len += 2;
|
|
||||||
|
|
||||||
uint8_t name_length = 0;
|
|
||||||
while (infos[i].name[name_length])
|
|
||||||
++name_length;
|
|
||||||
|
|
||||||
memcpy(path_builder + path_builder_len, infos[i].name, name_length + 1);
|
|
||||||
path_builder_len += name_length;
|
|
||||||
|
|
||||||
tree(d);
|
|
||||||
|
|
||||||
path_builder_len -= name_length;
|
|
||||||
path_builder[path_builder_len] = '\0';
|
|
||||||
|
|
||||||
indent_builder_len -= 2;
|
|
||||||
indent_builder[indent_builder_len] = '\0';
|
|
||||||
}
|
|
||||||
|
|
||||||
else {
|
|
||||||
u32_dec(infos[i].size, nbuf2);
|
|
||||||
logsz(" (");
|
|
||||||
logsz(nbuf2);
|
|
||||||
logsz(" bytes)\n");
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
|
@ -2,8 +2,9 @@
|
||||||
#include <stdint.h>
|
#include <stdint.h>
|
||||||
|
|
||||||
#define VGA_COLUMNS 80
|
#define VGA_COLUMNS 80
|
||||||
|
#define VGA_ROWS 25
|
||||||
#define VGA_START (uint16_t *)0x000b8000
|
#define VGA_START (uint16_t *)0x000b8000
|
||||||
#define VGA_END (VGA_START + VGA_COLUMNS * 25)
|
#define VGA_END (VGA_START + VGA_COLUMNS * VGA_ROWS)
|
||||||
static uint16_t *cursor = VGA_START;
|
static uint16_t *cursor = VGA_START;
|
||||||
static uint16_t mask;
|
static uint16_t mask;
|
||||||
|
|
||||||
|
@ -25,7 +26,7 @@ void vga_blank() {
|
||||||
uint32_t *p = (uint32_t *)VGA_START;
|
uint32_t *p = (uint32_t *)VGA_START;
|
||||||
while (p < (uint32_t *)VGA_END)
|
while (p < (uint32_t *)VGA_END)
|
||||||
*p++ = f;
|
*p++ = f;
|
||||||
cursor = VGA_START;
|
cursor = VGA_END - VGA_COLUMNS;
|
||||||
}
|
}
|
||||||
|
|
||||||
void vga_printch(char ch) {
|
void vga_printch(char ch) {
|
||||||
|
@ -37,4 +38,10 @@ void vga_printch(char ch) {
|
||||||
*cursor++ = mask | (uint8_t)ch;
|
*cursor++ = mask | (uint8_t)ch;
|
||||||
if (cursor == VGA_END)
|
if (cursor == VGA_END)
|
||||||
vga_scroll();
|
vga_scroll();
|
||||||
|
}
|
||||||
|
|
||||||
|
void vga_print_at(uint16_t pos, const char *sz) {
|
||||||
|
cursor = VGA_START + (pos >> 8) * VGA_COLUMNS + (pos & 0xff);
|
||||||
|
while (*sz)
|
||||||
|
vga_printch(*sz++);
|
||||||
}
|
}
|
|
@ -3,6 +3,7 @@
|
||||||
|
|
||||||
#include <stdint.h>
|
#include <stdint.h>
|
||||||
|
|
||||||
|
void vga_print_at(uint16_t pos, const char *sz);
|
||||||
void vga_set_color(uint8_t color);
|
void vga_set_color(uint8_t color);
|
||||||
void vga_blank();
|
void vga_blank();
|
||||||
void vga_printch(char ch);
|
void vga_printch(char ch);
|
||||||
|
|
|
@ -13,7 +13,7 @@ void main(const char *arg) {
|
||||||
tell_user_sz(*arg ? arg : "drive root");
|
tell_user_sz(*arg ? arg : "drive root");
|
||||||
tell_user_sz("\n");
|
tell_user_sz("\n");
|
||||||
|
|
||||||
_dir_info_entry infos[MAX_DIR_ENTRIES];
|
_dir_info_entry_t infos[MAX_DIR_ENTRIES];
|
||||||
uint8_t count = _enumerate_dir(dn, path, infos, MAX_DIR_ENTRIES);
|
uint8_t count = _enumerate_dir(dn, path, infos, MAX_DIR_ENTRIES);
|
||||||
tell_user_sz(
|
tell_user_sz(
|
||||||
count == MAX_DIR_ENTRIES
|
count == MAX_DIR_ENTRIES
|
||||||
|
|
130
src/user/fileman/fileman.c
Normal file
130
src/user/fileman/fileman.c
Normal file
|
@ -0,0 +1,130 @@
|
||||||
|
#include <pland/syscall.h>
|
||||||
|
#include <knob/block.h>
|
||||||
|
#include <knob/file.h>
|
||||||
|
#include <knob/heap.h>
|
||||||
|
#include <knob/task.h>
|
||||||
|
|
||||||
|
char path_sz[1024] = "";
|
||||||
|
uint32_t path_len = 0;
|
||||||
|
|
||||||
|
_dir_info_entry_t *dir_info;
|
||||||
|
uint32_t dir_info_len;
|
||||||
|
|
||||||
|
uint32_t scroll;
|
||||||
|
uint32_t selected;
|
||||||
|
#define MAX_PER_SCREEN 18
|
||||||
|
|
||||||
|
void paint_screen() {
|
||||||
|
_clear_screen();
|
||||||
|
|
||||||
|
_set_color(0x70);
|
||||||
|
_print_at(0, 0, path_sz);
|
||||||
|
_log_string("/");
|
||||||
|
_set_color(0x07);
|
||||||
|
|
||||||
|
for (uint32_t i = scroll, row = 2; (i < dir_info_len) && (row < MAX_PER_SCREEN + 2); ++i, ++row) {
|
||||||
|
if (i == selected)
|
||||||
|
_set_color(0x17);
|
||||||
|
_print_at(row, 0, dir_info[i].name);
|
||||||
|
if (dir_info[i].is_dir)
|
||||||
|
_log_string("/");
|
||||||
|
if (i == selected)
|
||||||
|
_set_color(0x07);
|
||||||
|
}
|
||||||
|
|
||||||
|
_print_at(21, 0, "keybindings:\n"
|
||||||
|
"UP, DOWN, PAGE UP, PAGE DOWN, HOME, END: move selection\n"
|
||||||
|
"ENTER: enter selected directory, run selected program\n"
|
||||||
|
"ESC: go up a directory Q: quit");
|
||||||
|
}
|
||||||
|
|
||||||
|
void load_dir() {
|
||||||
|
free_block(dir_info);
|
||||||
|
dir_info = get_directory_info(path_sz, &dir_info_len);
|
||||||
|
scroll = 0;
|
||||||
|
selected = 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
void main(const char *sz) {
|
||||||
|
path_len = strcpy(path_sz, sz);
|
||||||
|
dir_info = get_directory_info(sz, &dir_info_len);
|
||||||
|
|
||||||
|
paint_screen();
|
||||||
|
|
||||||
|
while (true) {
|
||||||
|
_key_code_t key;
|
||||||
|
switch (key = _get_key()) {
|
||||||
|
case 0:
|
||||||
|
_yield_task();
|
||||||
|
continue;
|
||||||
|
case '\n':
|
||||||
|
{
|
||||||
|
uint32_t old_len = path_len;
|
||||||
|
if (path_len)
|
||||||
|
path_sz[path_len++] = '/';
|
||||||
|
path_len += strcpy(path_sz + path_len, dir_info[selected].name);
|
||||||
|
if (dir_info[selected].is_dir)
|
||||||
|
load_dir();
|
||||||
|
else {
|
||||||
|
try_run_command_blocking(path_sz);
|
||||||
|
//TODO: handle error
|
||||||
|
path_sz[old_len] = '\0';
|
||||||
|
path_len = old_len;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
case _KEY_ESC:
|
||||||
|
{
|
||||||
|
char *cutoff = path_sz;
|
||||||
|
for (char *i = path_sz + path_len - 1; i >= path_sz; --i)
|
||||||
|
if (*i == '/') {
|
||||||
|
cutoff = i;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
*cutoff = '\0';
|
||||||
|
path_len = cutoff - path_sz;
|
||||||
|
load_dir();
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
case _KEY_HOME:
|
||||||
|
scroll = 0;
|
||||||
|
selected = 0;
|
||||||
|
break;
|
||||||
|
case _KEY_END:
|
||||||
|
selected = dir_info_len - 1;
|
||||||
|
scroll = dir_info_len >= MAX_PER_SCREEN
|
||||||
|
? dir_info_len - MAX_PER_SCREEN : 0;
|
||||||
|
break;
|
||||||
|
case _KEY_UP:
|
||||||
|
if (!selected)
|
||||||
|
continue;
|
||||||
|
if (--selected < scroll)
|
||||||
|
scroll = selected;
|
||||||
|
break;
|
||||||
|
case _KEY_DOWN:
|
||||||
|
if (selected == dir_info_len - 1)
|
||||||
|
continue;
|
||||||
|
if (++selected >= scroll + MAX_PER_SCREEN)
|
||||||
|
++scroll;
|
||||||
|
break;
|
||||||
|
case _KEY_PUP:
|
||||||
|
selected = selected >= MAX_PER_SCREEN
|
||||||
|
? selected - MAX_PER_SCREEN : 0;
|
||||||
|
scroll = scroll >= MAX_PER_SCREEN
|
||||||
|
? scroll - MAX_PER_SCREEN : 0;
|
||||||
|
break;
|
||||||
|
case _KEY_PDOWN:
|
||||||
|
if ((selected += MAX_PER_SCREEN) >= dir_info_len)
|
||||||
|
selected = dir_info_len - 1;
|
||||||
|
if ((scroll += MAX_PER_SCREEN) > selected)
|
||||||
|
scroll = selected;
|
||||||
|
break;
|
||||||
|
case 'q':
|
||||||
|
_clear_screen();
|
||||||
|
return;
|
||||||
|
default:
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
paint_screen();
|
||||||
|
}
|
||||||
|
}
|
|
@ -1,5 +1,5 @@
|
||||||
|
#include <pland/pcrt.h>
|
||||||
#include <knob/block.h>
|
#include <knob/block.h>
|
||||||
#include <knob/quit.h>
|
|
||||||
#include <knob/task.h>
|
#include <knob/task.h>
|
||||||
#include <knob/user.h>
|
#include <knob/user.h>
|
||||||
#include "cmds.h"
|
#include "cmds.h"
|
||||||
|
@ -68,7 +68,7 @@ void run_line(const char *original_line) {
|
||||||
else if (blockequ(line, "vars", space - line))
|
else if (blockequ(line, "vars", space - line))
|
||||||
dump_vars();
|
dump_vars();
|
||||||
else if (blockequ(line, "quit", space - line))
|
else if (blockequ(line, "quit", space - line))
|
||||||
quit(space + 1);
|
__pcrt_quit();
|
||||||
else if (blockequ(line, "help", space - line))
|
else if (blockequ(line, "help", space - line))
|
||||||
tell_user_sz("Highway is a command shell for Portland OS. It includes variables and a couple\n"
|
tell_user_sz("Highway is a command shell for Portland OS. It includes variables and a couple\n"
|
||||||
"of pseudo-commands. Variables are addressed by surrounding with \"$\". The\n"
|
"of pseudo-commands. Variables are addressed by surrounding with \"$\". The\n"
|
||||||
|
@ -78,7 +78,7 @@ void run_line(const char *original_line) {
|
||||||
" echo STRING print STRING\n"
|
" echo STRING print STRING\n"
|
||||||
" vars dump variables\n"
|
" vars dump variables\n"
|
||||||
" quit exit highway\n"
|
" quit exit highway\n"
|
||||||
" help show this\n\n");
|
" help show this\n");
|
||||||
else if (!try_run_command_blocking(line)) {
|
else if (!try_run_command_blocking(line)) {
|
||||||
struct no_null_sn arg = {
|
struct no_null_sn arg = {
|
||||||
.data = "_path",
|
.data = "_path",
|
||||||
|
|
|
@ -9,7 +9,7 @@ void main(const char *arg) {
|
||||||
yield_task();
|
yield_task();
|
||||||
tell_user_sz("Portland Highway\nType \"help\" for help.\n");
|
tell_user_sz("Portland Highway\nType \"help\" for help.\n");
|
||||||
while (1) {
|
while (1) {
|
||||||
tell_user_sz("> ");
|
tell_user_sz("\n> ");
|
||||||
ask_user_line_sz(cmd_buf, 127);
|
ask_user_line_sz(cmd_buf, 127);
|
||||||
run_line(cmd_buf);
|
run_line(cmd_buf);
|
||||||
}
|
}
|
||||||
|
|
|
@ -7,4 +7,7 @@
|
||||||
void blockcpy(void *to, const void *from, uint32_t size);
|
void blockcpy(void *to, const void *from, uint32_t size);
|
||||||
bool blockequ(const void *a, const void *b, uint32_t size) __attribute__ ((__pure__));
|
bool blockequ(const void *a, const void *b, uint32_t size) __attribute__ ((__pure__));
|
||||||
|
|
||||||
|
//returns length without null-terminator
|
||||||
|
uint32_t strcpy(char *to, const char *from);
|
||||||
|
|
||||||
#endif
|
#endif
|
|
@ -2,6 +2,7 @@
|
||||||
#define KNOB_FILE_H
|
#define KNOB_FILE_H
|
||||||
|
|
||||||
#include <stdint.h>
|
#include <stdint.h>
|
||||||
|
#include <pland/syscall.h>
|
||||||
|
|
||||||
struct file;
|
struct file;
|
||||||
|
|
||||||
|
@ -9,7 +10,6 @@ const char *remove_prefix(const char *path, uint8_t *dn_out);
|
||||||
|
|
||||||
struct file *open_file(const char *path);
|
struct file *open_file(const char *path);
|
||||||
void close_file(struct file *f);
|
void close_file(struct file *f);
|
||||||
void _close_all_files();
|
|
||||||
|
|
||||||
uint32_t read_from_file(struct file *f, uint32_t max, void *buf);
|
uint32_t read_from_file(struct file *f, uint32_t max, void *buf);
|
||||||
//return value and max_length don't include null terminator
|
//return value and max_length don't include null terminator
|
||||||
|
@ -19,4 +19,7 @@ int32_t seek_file_by(struct file *f, int32_t by);
|
||||||
|
|
||||||
uint32_t file_size(struct file *f) __attribute__ ((pure));
|
uint32_t file_size(struct file *f) __attribute__ ((pure));
|
||||||
|
|
||||||
|
//return value must be manually freed, unless it is a null pointer
|
||||||
|
_dir_info_entry_t *get_directory_info(const char *path, uint32_t *count_out);
|
||||||
|
|
||||||
#endif
|
#endif
|
|
@ -1,7 +0,0 @@
|
||||||
#ifndef KNOB_QUIT_H
|
|
||||||
#define KNOB_QUIT_H
|
|
||||||
|
|
||||||
void on_quit(void (*run_f)());
|
|
||||||
void quit() __attribute__ ((noreturn));
|
|
||||||
|
|
||||||
#endif
|
|
|
@ -2,6 +2,9 @@
|
||||||
#define KNOB_USER_H
|
#define KNOB_USER_H
|
||||||
|
|
||||||
#include <stdint.h>
|
#include <stdint.h>
|
||||||
|
#include <pland/syscall.h>
|
||||||
|
|
||||||
|
char key_to_char(_key_code_t key) __attribute__ ((const));
|
||||||
|
|
||||||
void tell_user_sz(const char *sz);
|
void tell_user_sz(const char *sz);
|
||||||
|
|
||||||
|
|
16
src/user/include/pland/pcrt.h
Normal file
16
src/user/include/pland/pcrt.h
Normal file
|
@ -0,0 +1,16 @@
|
||||||
|
#ifndef PLAND_PCRT_H
|
||||||
|
#define PLAND_PCRT_H
|
||||||
|
|
||||||
|
#define BEFORE_MAIN(f) \
|
||||||
|
__attribute__ ((section (".__pcrt_before_main"))) \
|
||||||
|
__attribute__ ((unused)) \
|
||||||
|
void (*const __pcrt_bm_##f)() = &f;
|
||||||
|
|
||||||
|
#define BEFORE_QUIT(f) \
|
||||||
|
__attribute__ ((section (".__pcrt_before_quit"))) \
|
||||||
|
__attribute__ ((unused)) \
|
||||||
|
void (*const __pcrt_bq_##f)() = &f;
|
||||||
|
|
||||||
|
void __pcrt_quit() __attribute__ ((noreturn));
|
||||||
|
|
||||||
|
#endif
|
|
@ -7,14 +7,9 @@
|
||||||
typedef uint32_t _file_handle_t;
|
typedef uint32_t _file_handle_t;
|
||||||
typedef uint32_t _task_handle_t;
|
typedef uint32_t _task_handle_t;
|
||||||
typedef uint32_t _drive_number_t;
|
typedef uint32_t _drive_number_t;
|
||||||
typedef uint32_t _pages_t;
|
|
||||||
typedef uint32_t _process_handle_t;
|
typedef uint32_t _process_handle_t;
|
||||||
|
|
||||||
typedef enum {
|
typedef enum {
|
||||||
_KEY_BACKSPACE = '\b',
|
|
||||||
_KEY_RETURN = '\n',
|
|
||||||
//etc.
|
|
||||||
|
|
||||||
_KEY_LSHIFT = 0x00000100,
|
_KEY_LSHIFT = 0x00000100,
|
||||||
_KEY_RSHIFT = 0x00000200,
|
_KEY_RSHIFT = 0x00000200,
|
||||||
_KEY_CAPS = 0x00000400,
|
_KEY_CAPS = 0x00000400,
|
||||||
|
@ -33,8 +28,103 @@ typedef enum {
|
||||||
_KEY_ALT = 0x0000c000,
|
_KEY_ALT = 0x0000c000,
|
||||||
_KEY_CTRL = 0x00030000,
|
_KEY_CTRL = 0x00030000,
|
||||||
_KEY_META = 0x000c0000,
|
_KEY_META = 0x000c0000,
|
||||||
|
|
||||||
|
_KEY_BEGIN_CAPS = 0x80,
|
||||||
|
_KEY_BEGIN_INSERT,
|
||||||
|
_KEY_BEGIN_NUM,
|
||||||
|
_KEY_BEGIN_SCROLL,
|
||||||
|
_KEY_BEGIN_LSHIFT,
|
||||||
|
_KEY_BEGIN_RSHIFT,
|
||||||
|
_KEY_BEGIN_LALT,
|
||||||
|
_KEY_BEGIN_RALT,
|
||||||
|
_KEY_BEGIN_LCTRL,
|
||||||
|
_KEY_BEGIN_RCTRL,
|
||||||
|
_KEY_BEGIN_LMETA,
|
||||||
|
_KEY_BEGIN_RMETA,
|
||||||
|
/* 0x8c - 0x97 reserved */
|
||||||
|
/* 0x98 - 0x9f unassigned */
|
||||||
|
_KEY_F1 = 0xa0,
|
||||||
|
_KEY_F2,
|
||||||
|
_KEY_F3,
|
||||||
|
_KEY_F4,
|
||||||
|
_KEY_F5,
|
||||||
|
_KEY_F6,
|
||||||
|
_KEY_F7,
|
||||||
|
_KEY_F8,
|
||||||
|
_KEY_F9,
|
||||||
|
_KEY_F10,
|
||||||
|
_KEY_F11,
|
||||||
|
_KEY_F12,
|
||||||
|
/* 0xac - 0xaf unassigned */
|
||||||
|
_KEY_NUM0 = 0xb0,
|
||||||
|
_KEY_NUM1,
|
||||||
|
_KEY_NUM2,
|
||||||
|
_KEY_NUM3,
|
||||||
|
_KEY_NUM4,
|
||||||
|
_KEY_NUM5,
|
||||||
|
_KEY_NUM6,
|
||||||
|
_KEY_NUM7,
|
||||||
|
_KEY_NUM8,
|
||||||
|
_KEY_NUM9,
|
||||||
|
_KEY_NTIMES,
|
||||||
|
_KEY_NPLUS,
|
||||||
|
_KEY_NENTER,
|
||||||
|
_KEY_NMINUS,
|
||||||
|
_KEY_NDOT,
|
||||||
|
_KEY_NSLASH,
|
||||||
|
/* 0xc0 unassigned */
|
||||||
|
_KEY_DELETE,
|
||||||
|
_KEY_HOME,
|
||||||
|
_KEY_END,
|
||||||
|
_KEY_PUP,
|
||||||
|
_KEY_PDOWN,
|
||||||
|
_KEY_UP,
|
||||||
|
_KEY_DOWN,
|
||||||
|
_KEY_LEFT,
|
||||||
|
_KEY_RIGHT,
|
||||||
|
_KEY_ESC,
|
||||||
|
_KEY_MENU,
|
||||||
|
_KEY_PAUSE,
|
||||||
|
_KEY_PRSCR,
|
||||||
|
/* 0xce - 0xef unassigned */
|
||||||
} _key_code_t;
|
} _key_code_t;
|
||||||
|
|
||||||
|
typedef enum {
|
||||||
|
_COLOR_FG_BLACK = 0x00,
|
||||||
|
_COLOR_FG_BLUE = 0x01,
|
||||||
|
_COLOR_FG_GREEN = 0x02,
|
||||||
|
_COLOR_FG_CYAN = 0x03,
|
||||||
|
_COLOR_FG_RED = 0x04,
|
||||||
|
_COLOR_FG_MAGENTA = 0x05,
|
||||||
|
_COLOR_FG_BROWN = 0x06,
|
||||||
|
_COLOR_FG_LGRAY = 0x07,
|
||||||
|
|
||||||
|
_COLOR_FG_DGRAY = 0x08,
|
||||||
|
_COLOR_FG_LBLUE = 0x09,
|
||||||
|
_COLOR_FG_LGREEN = 0x0a,
|
||||||
|
_COLOR_FG_LCYAN = 0x0b,
|
||||||
|
_COLOR_FG_LRED = 0x0c,
|
||||||
|
_COLOR_FG_PINK = 0x0d,
|
||||||
|
_COLOR_FG_YELLOW = 0x0e,
|
||||||
|
_COLOR_FG_WHITE = 0x0f,
|
||||||
|
|
||||||
|
_COLOR_BG_BLACK = 0x00,
|
||||||
|
_COLOR_BG_BLUE = 0x10,
|
||||||
|
_COLOR_BG_GREEN = 0x20,
|
||||||
|
_COLOR_BG_CYAN = 0x30,
|
||||||
|
_COLOR_BG_RED = 0x40,
|
||||||
|
_COLOR_BG_MAGENTA = 0x50,
|
||||||
|
_COLOR_BG_BROWN = 0x60,
|
||||||
|
_COLOR_BG_LGRAY = 0x70
|
||||||
|
} _vga_color_t;
|
||||||
|
|
||||||
|
typedef struct __attribute__ ((packed)) {
|
||||||
|
char name[100];
|
||||||
|
uint32_t size;
|
||||||
|
bool is_dir;
|
||||||
|
uint8_t pad[23];
|
||||||
|
} _dir_info_entry_t;
|
||||||
|
|
||||||
enum _scn {
|
enum _scn {
|
||||||
_SCN_OPEN_FILE,
|
_SCN_OPEN_FILE,
|
||||||
_SCN_CLOSE_FILE,
|
_SCN_CLOSE_FILE,
|
||||||
|
@ -46,72 +136,74 @@ enum _scn {
|
||||||
_SCN_ALLOCATE_RAM,
|
_SCN_ALLOCATE_RAM,
|
||||||
_SCN_MEMORY_INFO,
|
_SCN_MEMORY_INFO,
|
||||||
_SCN_WAIT_FOR_TASK,
|
_SCN_WAIT_FOR_TASK,
|
||||||
_SCN_ENUMERATE_DIR
|
_SCN_ENUMERATE_DIR,
|
||||||
|
_SCN_PRINT_AT,
|
||||||
|
_SCN_COUNT_OF_DIR,
|
||||||
|
_SCN_CLEAR_SCREEN,
|
||||||
|
_SCN_SET_COLOR
|
||||||
};
|
};
|
||||||
|
|
||||||
typedef struct {
|
|
||||||
bool is_dir;
|
|
||||||
char name[100];
|
|
||||||
uint32_t size;
|
|
||||||
} _dir_info_entry;
|
|
||||||
|
|
||||||
static inline uint32_t _sc0(enum _scn eax) {
|
static inline uint32_t _sc0(enum _scn eax) {
|
||||||
volatile uint32_t out;
|
uint32_t out;
|
||||||
asm (
|
asm volatile (
|
||||||
"int $0x30"
|
"int $0x30"
|
||||||
: "=a" (out) : "a" (eax) : "ecx", "edx");
|
: "=a" (out) : "a" (eax) : "ecx", "edx");
|
||||||
return out;
|
return out;
|
||||||
}
|
}
|
||||||
|
|
||||||
static inline uint32_t _sc1(enum _scn eax, uint32_t ebx) {
|
static inline uint32_t _sc1(enum _scn eax, uint32_t ebx) {
|
||||||
volatile uint32_t out;
|
uint32_t out;
|
||||||
asm (
|
asm volatile (
|
||||||
"int $0x30"
|
"int $0x30"
|
||||||
: "=a" (out) : "a" (eax), "b" (ebx) : "ecx", "edx");
|
: "=a" (out) : "a" (eax), "b" (ebx) : "ecx", "edx");
|
||||||
return out;
|
return out;
|
||||||
}
|
}
|
||||||
|
|
||||||
static inline uint32_t _sc2(enum _scn eax, uint32_t ebx, uint32_t ecx) {
|
static inline uint32_t _sc2(enum _scn eax, uint32_t ebx, uint32_t ecx) {
|
||||||
volatile uint32_t out;
|
uint32_t out;
|
||||||
asm (
|
uint32_t dummy;
|
||||||
|
asm volatile (
|
||||||
"int $0x30"
|
"int $0x30"
|
||||||
: "=a" (out) : "a" (eax), "b" (ebx), "c" (ecx) : "edx");
|
: "=a" (out), "=c" (dummy) : "a" (eax), "b" (ebx), "c" (ecx) : "edx");
|
||||||
return out;
|
return out;
|
||||||
}
|
}
|
||||||
|
|
||||||
static inline uint32_t _sc3(enum _scn eax, uint32_t ebx, uint32_t ecx, uint32_t edx) {
|
static inline uint32_t _sc3(enum _scn eax, uint32_t ebx, uint32_t ecx, uint32_t edx) {
|
||||||
volatile uint32_t out;
|
uint32_t out;
|
||||||
asm (
|
uint32_t dummy;
|
||||||
|
asm volatile (
|
||||||
"int $0x30"
|
"int $0x30"
|
||||||
: "=a" (out) : "a" (eax), "b" (ebx), "c" (ecx), "d" (edx));
|
: "=a" (out), "=c" (dummy), "=d" (dummy) : "a" (eax), "b" (ebx), "c" (ecx), "d" (edx));
|
||||||
return out;
|
return out;
|
||||||
}
|
}
|
||||||
|
|
||||||
static inline uint32_t _sc4(enum _scn eax, uint32_t ebx, uint32_t ecx, uint32_t edx, uint32_t esi) {
|
static inline uint32_t _sc4(enum _scn eax, uint32_t ebx, uint32_t ecx, uint32_t edx, uint32_t esi) {
|
||||||
volatile uint32_t out;
|
uint32_t out;
|
||||||
asm (
|
uint32_t dummy;
|
||||||
|
asm volatile (
|
||||||
"int $0x30"
|
"int $0x30"
|
||||||
: "=a" (out) : "a" (eax), "b" (ebx), "c" (ecx), "d" (edx), "S" (esi));
|
: "=a" (out), "=c" (dummy), "=d" (dummy) : "a" (eax), "b" (ebx), "c" (ecx), "d" (edx), "S" (esi));
|
||||||
return out;
|
return out;
|
||||||
}
|
}
|
||||||
|
|
||||||
static inline uint32_t _sc5(enum _scn eax, uint32_t ebx, uint32_t ecx, uint32_t edx, uint32_t esi, uint32_t edi) {
|
static inline uint32_t _sc5(enum _scn eax, uint32_t ebx, uint32_t ecx, uint32_t edx, uint32_t esi, uint32_t edi) {
|
||||||
volatile uint32_t out;
|
uint32_t out;
|
||||||
asm (
|
uint32_t dummy;
|
||||||
|
asm volatile (
|
||||||
"int $0x30"
|
"int $0x30"
|
||||||
: "=a" (out) : "a" (eax), "b" (ebx), "c" (ecx), "d" (edx), "S" (esi), "D" (edi));
|
: "=a" (out), "=c" (dummy), "=d" (dummy) : "a" (eax), "b" (ebx), "c" (ecx), "d" (edx), "S" (esi), "D" (edi));
|
||||||
return out;
|
return out;
|
||||||
}
|
}
|
||||||
|
|
||||||
static inline void _yield_task() {
|
static inline void _yield_task() {
|
||||||
asm (
|
asm volatile (
|
||||||
"int $0x39"
|
"int $0x39"
|
||||||
: : : "eax");
|
: : : "eax");
|
||||||
}
|
}
|
||||||
|
|
||||||
__attribute__ ((noreturn))
|
__attribute__ ((noreturn))
|
||||||
static inline void _exit_task() {
|
static inline void _exit_task() {
|
||||||
asm (
|
asm volatile (
|
||||||
"int $0x38"
|
"int $0x38"
|
||||||
);
|
);
|
||||||
__builtin_unreachable();
|
__builtin_unreachable();
|
||||||
|
@ -145,27 +237,27 @@ static inline _key_code_t _get_key() {
|
||||||
return _sc0(_SCN_GET_KEY);
|
return _sc0(_SCN_GET_KEY);
|
||||||
}
|
}
|
||||||
|
|
||||||
static inline void *_allocate_ram(_pages_t pages) {
|
static inline void *_allocate_ram(uint32_t pages) {
|
||||||
return (void *)_sc1(_SCN_ALLOCATE_RAM, pages);
|
return (void *)_sc1(_SCN_ALLOCATE_RAM, pages);
|
||||||
}
|
}
|
||||||
|
|
||||||
static inline _pages_t _kernel_dynamic_area_size() {
|
static inline uint32_t _kernel_dynamic_area_size() {
|
||||||
return _sc1(_SCN_MEMORY_INFO, 0x0);
|
return _sc1(_SCN_MEMORY_INFO, 0x0);
|
||||||
}
|
}
|
||||||
|
|
||||||
static inline _pages_t _kernel_dynamic_area_left() {
|
static inline uint32_t _kernel_dynamic_area_left() {
|
||||||
return _sc1(_SCN_MEMORY_INFO, 0x1);
|
return _sc1(_SCN_MEMORY_INFO, 0x1);
|
||||||
}
|
}
|
||||||
|
|
||||||
static inline _pages_t _total_userspace_size() {
|
static inline uint32_t _total_userspace_size() {
|
||||||
return _sc1(_SCN_MEMORY_INFO, 0x2);
|
return _sc1(_SCN_MEMORY_INFO, 0x2);
|
||||||
}
|
}
|
||||||
|
|
||||||
static inline _pages_t _total_userspace_left() {
|
static inline uint32_t _total_userspace_left() {
|
||||||
return _sc1(_SCN_MEMORY_INFO, 0x3);
|
return _sc1(_SCN_MEMORY_INFO, 0x3);
|
||||||
}
|
}
|
||||||
|
|
||||||
static inline _pages_t _this_process_memory_left() {
|
static inline uint32_t _this_process_memory_left() {
|
||||||
return _sc1(_SCN_MEMORY_INFO, 0x4);
|
return _sc1(_SCN_MEMORY_INFO, 0x4);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -173,8 +265,24 @@ static inline void _wait_for_task(_process_handle_t handle) {
|
||||||
_sc1(_SCN_WAIT_FOR_TASK, handle);
|
_sc1(_SCN_WAIT_FOR_TASK, handle);
|
||||||
}
|
}
|
||||||
|
|
||||||
static inline uint32_t _enumerate_dir(_drive_number_t drive_number, const char *path, _dir_info_entry *buffer, uint32_t max_count) {
|
static inline uint32_t _enumerate_dir(_drive_number_t drive_number, const char *path, _dir_info_entry_t *buffer, uint32_t max_count) {
|
||||||
return _sc4(_SCN_ENUMERATE_DIR, drive_number, (uint32_t)path, (uint32_t)buffer, max_count);
|
return _sc4(_SCN_ENUMERATE_DIR, drive_number, (uint32_t)path, (uint32_t)buffer, max_count);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static inline void _print_at(uint8_t row, uint8_t col, const char *sz) {
|
||||||
|
_sc2(_SCN_PRINT_AT, (row << 8) | col, (uint32_t)sz);
|
||||||
|
}
|
||||||
|
|
||||||
|
static inline uint32_t _count_of_dir(uint8_t drive_number, const char *path) {
|
||||||
|
return _sc2(_SCN_COUNT_OF_DIR, drive_number, (uint32_t)path);
|
||||||
|
}
|
||||||
|
|
||||||
|
static inline void _clear_screen() {
|
||||||
|
_sc0(_SCN_CLEAR_SCREEN);
|
||||||
|
}
|
||||||
|
|
||||||
|
static inline void _set_color(_vga_color_t color) {
|
||||||
|
_sc1(_SCN_SET_COLOR, color);
|
||||||
|
}
|
||||||
|
|
||||||
#endif
|
#endif
|
|
@ -14,4 +14,13 @@ bool blockequ(const void *a, const void *b, uint32_t size) {
|
||||||
if (*(uint8_t *)(a++) != *(uint8_t *)(b++))
|
if (*(uint8_t *)(a++) != *(uint8_t *)(b++))
|
||||||
return false;
|
return false;
|
||||||
return true;
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
//returns length without null-terminator
|
||||||
|
uint32_t strcpy(char *to, const char *from) {
|
||||||
|
uint32_t i = 0;
|
||||||
|
do
|
||||||
|
to[i] = from[i];
|
||||||
|
while (from[i++]);
|
||||||
|
return i - 1;
|
||||||
}
|
}
|
|
@ -1,23 +0,0 @@
|
||||||
bits 32
|
|
||||||
|
|
||||||
global _entry
|
|
||||||
|
|
||||||
extern main
|
|
||||||
extern quit
|
|
||||||
|
|
||||||
extern current_disk
|
|
||||||
|
|
||||||
section .text
|
|
||||||
_entry:
|
|
||||||
mov esp, stack
|
|
||||||
push edx
|
|
||||||
|
|
||||||
;TODO: determine current_disk
|
|
||||||
;any further needed initialization
|
|
||||||
|
|
||||||
push quit
|
|
||||||
jmp main
|
|
||||||
|
|
||||||
section .stack nobits alloc noexec write align=16
|
|
||||||
resb 4096
|
|
||||||
stack:
|
|
|
@ -1,4 +1,5 @@
|
||||||
#include <pland/syscall.h>
|
#include <pland/syscall.h>
|
||||||
|
#include <pland/pcrt.h>
|
||||||
#include <knob/format.h>
|
#include <knob/format.h>
|
||||||
#include <knob/heap.h>
|
#include <knob/heap.h>
|
||||||
#include <knob/env.h>
|
#include <knob/env.h>
|
||||||
|
@ -11,11 +12,13 @@ struct ofl_node {
|
||||||
|
|
||||||
static struct ofl_node *head_ofl_node = 0;
|
static struct ofl_node *head_ofl_node = 0;
|
||||||
|
|
||||||
void _close_all_files() {
|
static void _close_all_files() {
|
||||||
for (struct ofl_node *i = head_ofl_node; i; i = i->next)
|
for (struct ofl_node *i = head_ofl_node; i; i = i->next)
|
||||||
_close_file(i->handle);
|
_close_file(i->handle);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
BEFORE_QUIT(_close_all_files)
|
||||||
|
|
||||||
struct file {
|
struct file {
|
||||||
struct ofl_node *node;
|
struct ofl_node *node;
|
||||||
_file_handle_t handle;
|
_file_handle_t handle;
|
||||||
|
@ -121,4 +124,20 @@ int32_t seek_file_by(struct file *f, int32_t by) {
|
||||||
__attribute__ ((pure))
|
__attribute__ ((pure))
|
||||||
uint32_t file_size(struct file *f) {
|
uint32_t file_size(struct file *f) {
|
||||||
return f->length;
|
return f->length;
|
||||||
|
}
|
||||||
|
|
||||||
|
//return value must be manually freed, unless it is a null pointer
|
||||||
|
_dir_info_entry_t *get_directory_info(const char *path, uint32_t *count_out) {
|
||||||
|
uint8_t dn;
|
||||||
|
path = remove_prefix(path, &dn);
|
||||||
|
|
||||||
|
uint32_t count = _count_of_dir(dn, path);
|
||||||
|
if (!count) {
|
||||||
|
*count_out = 0;
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
_dir_info_entry_t *buffer = get_block(count * sizeof(_dir_info_entry_t));
|
||||||
|
*count_out = _enumerate_dir(dn, path, buffer, count);
|
||||||
|
return buffer;
|
||||||
}
|
}
|
|
@ -1,8 +0,0 @@
|
||||||
#include <pland/syscall.h>
|
|
||||||
#include <knob/file.h>
|
|
||||||
|
|
||||||
__attribute__ ((noreturn))
|
|
||||||
void quit() {
|
|
||||||
_close_all_files();
|
|
||||||
_exit_task();
|
|
||||||
}
|
|
|
@ -116,10 +116,8 @@ static const uint8_t shifted[] = {
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00
|
||||||
};
|
};
|
||||||
|
|
||||||
static char get_key_char() {
|
__attribute__ ((const))
|
||||||
_key_code_t key;
|
char key_to_char(_key_code_t key) {
|
||||||
while (!(key = _get_key()))
|
|
||||||
_yield_task();
|
|
||||||
return
|
return
|
||||||
key & _KEY_CAPS
|
key & _KEY_CAPS
|
||||||
? key & _KEY_SHIFT
|
? key & _KEY_SHIFT
|
||||||
|
@ -130,6 +128,13 @@ static char get_key_char() {
|
||||||
: key & 0xff;
|
: key & 0xff;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static char get_key_char() {
|
||||||
|
_key_code_t key;
|
||||||
|
while (!(key = _get_key()))
|
||||||
|
_yield_task();
|
||||||
|
return key_to_char(key);
|
||||||
|
}
|
||||||
|
|
||||||
void tell_user_sz(const char *sz) {
|
void tell_user_sz(const char *sz) {
|
||||||
_log_string(sz);
|
_log_string(sz);
|
||||||
}
|
}
|
||||||
|
|
22
src/user/runtimes/c/elf.ld
Normal file
22
src/user/runtimes/c/elf.ld
Normal file
|
@ -0,0 +1,22 @@
|
||||||
|
OUTPUT_FORMAT(elf32-i386)
|
||||||
|
OUTPUT_ARCH(i386)
|
||||||
|
ENTRY(__pcrt_entry)
|
||||||
|
|
||||||
|
MEMORY {
|
||||||
|
kernel (!a) : ORIGIN = 0x00000000, LENGTH = 0x08000000
|
||||||
|
user (awx) : ORIGIN = 0x08000000, LENGTH = 0xf8000000
|
||||||
|
}
|
||||||
|
|
||||||
|
SECTIONS {
|
||||||
|
.__pcrt_before_main : {
|
||||||
|
__pcrt_before_main_start = .;
|
||||||
|
*(.__pcrt_before_main)
|
||||||
|
__pcrt_before_main_end = .;
|
||||||
|
}
|
||||||
|
|
||||||
|
.__pcrt_before_quit : {
|
||||||
|
__pcrt_before_quit_start = .;
|
||||||
|
*(.__pcrt_before_quit)
|
||||||
|
__pcrt_before_quit_end = .;
|
||||||
|
}
|
||||||
|
}
|
42
src/user/runtimes/c/entry.asm
Normal file
42
src/user/runtimes/c/entry.asm
Normal file
|
@ -0,0 +1,42 @@
|
||||||
|
bits 32
|
||||||
|
|
||||||
|
global __pcrt_entry
|
||||||
|
global __pcrt_quit
|
||||||
|
|
||||||
|
extern main
|
||||||
|
extern __pcrt_before_main_start
|
||||||
|
extern __pcrt_before_main_end
|
||||||
|
extern __pcrt_before_quit_start
|
||||||
|
extern __pcrt_before_quit_end
|
||||||
|
|
||||||
|
section .text
|
||||||
|
__pcrt_entry:
|
||||||
|
mov esp, stack
|
||||||
|
push edx
|
||||||
|
|
||||||
|
mov ebx, __pcrt_before_main_start
|
||||||
|
.before_main_loop:
|
||||||
|
cmp ebx, __pcrt_before_main_end
|
||||||
|
je .call_main
|
||||||
|
call dword [ebx]
|
||||||
|
add ebx, 4
|
||||||
|
jmp .before_main_loop
|
||||||
|
|
||||||
|
.call_main:
|
||||||
|
call main
|
||||||
|
|
||||||
|
__pcrt_quit:
|
||||||
|
mov ebx, __pcrt_before_quit_start
|
||||||
|
.before_quit_loop:
|
||||||
|
cmp ebx, __pcrt_before_quit_end
|
||||||
|
je .end_task
|
||||||
|
call dword [ebx]
|
||||||
|
add ebx, 4
|
||||||
|
jmp .before_quit_loop
|
||||||
|
|
||||||
|
.end_task:
|
||||||
|
int 0x38
|
||||||
|
|
||||||
|
section .stack nobits alloc noexec write align=16
|
||||||
|
resb 4096
|
||||||
|
stack:
|
Reference in a new issue