new shell
This commit is contained in:
parent
143156f63e
commit
44d29a33df
18 changed files with 358 additions and 74 deletions
1
fs-skel/user/default.rc
Normal file
1
fs-skel/user/default.rc
Normal file
|
@ -0,0 +1 @@
|
||||||
|
set _path bin/
|
7
makefile
7
makefile
|
@ -74,14 +74,15 @@ obj/init.elf: obj/init/init.o obj/knob.so
|
||||||
obj/meminfo.elf: obj/meminfo/meminfo.o obj/knob.so
|
obj/meminfo.elf: obj/meminfo/meminfo.o obj/knob.so
|
||||||
ld -T src/user/elf.ld obj/meminfo/* obj/knob.so -o obj/meminfo.elf
|
ld -T src/user/elf.ld obj/meminfo/* obj/knob.so -o obj/meminfo.elf
|
||||||
|
|
||||||
obj/highway.elf: obj/highway/highway.o obj/knob.so
|
obj/highway.elf: obj/highway/main.o obj/highway/cmds.o obj/highway/line.o \
|
||||||
|
obj/highway/vars.o obj/knob.so
|
||||||
ld -T src/user/elf.ld obj/highway/* obj/knob.so -o obj/highway.elf
|
ld -T src/user/elf.ld obj/highway/* obj/knob.so -o obj/highway.elf
|
||||||
|
|
||||||
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/elf.ld obj/hello/* -o obj/hello.elf
|
||||||
|
|
||||||
obj/dumptext.elf: obj/dumptext/dumptext.o
|
obj/dumptext.elf: obj/dumptext/dumptext.o obj/knob.so
|
||||||
ld -T src/user/elf.ld obj/dumptext/* obj/knob.so -o obj/dumptext.elf
|
ld -T src/user/elf.ld obj/dumptext/* obj/knob.so -o obj/dumptext.elf
|
||||||
|
|
||||||
obj/dumphex.elf: obj/dumphex/dumphex.o
|
obj/dumphex.elf: obj/dumphex/dumphex.o obj/knob.so
|
||||||
ld -T src/user/elf.ld obj/dumphex/* obj/knob.so -o obj/dumphex.elf
|
ld -T src/user/elf.ld obj/dumphex/* obj/knob.so -o obj/dumphex.elf
|
46
src/user/highway/cmds.c
Normal file
46
src/user/highway/cmds.c
Normal file
|
@ -0,0 +1,46 @@
|
||||||
|
#include <knob/file.h>
|
||||||
|
#include <knob/user.h>
|
||||||
|
#include "line.h"
|
||||||
|
#include "vars.h"
|
||||||
|
|
||||||
|
void source(const char *path) {
|
||||||
|
struct file *f = open_file(path);
|
||||||
|
if (!f)
|
||||||
|
tell_user_sz("couldn't open file.\n");
|
||||||
|
char buf[128];
|
||||||
|
while (read_line_from_file(f, buf, 127))
|
||||||
|
run_line(buf);
|
||||||
|
close_file(f);
|
||||||
|
}
|
||||||
|
|
||||||
|
void set(const char *arg) {
|
||||||
|
const char *space = arg;
|
||||||
|
while (*space != ' ')
|
||||||
|
if (*space)
|
||||||
|
++space;
|
||||||
|
else {
|
||||||
|
struct no_null_sn vname = {
|
||||||
|
.data = arg,
|
||||||
|
.length = space - arg
|
||||||
|
};
|
||||||
|
del_var(vname);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
struct no_null_sn vname = {
|
||||||
|
.data = arg,
|
||||||
|
.length = space - arg
|
||||||
|
};
|
||||||
|
|
||||||
|
const char *vstart = space + 1,
|
||||||
|
*vend = vstart;
|
||||||
|
while (*vend)
|
||||||
|
++vend;
|
||||||
|
|
||||||
|
struct no_null_sn vval = {
|
||||||
|
.data = vstart,
|
||||||
|
.length = vend - vstart
|
||||||
|
};
|
||||||
|
|
||||||
|
set_var(vname, vval);
|
||||||
|
}
|
7
src/user/highway/cmds.h
Normal file
7
src/user/highway/cmds.h
Normal file
|
@ -0,0 +1,7 @@
|
||||||
|
#ifndef CMDS_H
|
||||||
|
#define CMDS_H
|
||||||
|
|
||||||
|
void source(const char *path);
|
||||||
|
void set(const char *arg);
|
||||||
|
|
||||||
|
#endif
|
|
@ -1,24 +0,0 @@
|
||||||
#include <knob/user.h>
|
|
||||||
#include <knob/task.h>
|
|
||||||
#include <knob/block.h>
|
|
||||||
|
|
||||||
//TODO: load a user environment file containing a PATH-like setting.
|
|
||||||
//TODO: have an active disk and/or directory
|
|
||||||
|
|
||||||
void main() {
|
|
||||||
char path_buf[1024 + 4] = "bin/";
|
|
||||||
char *const line_buf = path_buf + 4;
|
|
||||||
|
|
||||||
tell_user_sz("Highway, Portland Command Shell, started.\n"
|
|
||||||
"Type \"exit\" to quit.\n");
|
|
||||||
yield_task();
|
|
||||||
|
|
||||||
while (1) {
|
|
||||||
tell_user_sz("> ");
|
|
||||||
ask_user_line_sz(line_buf, 1023);
|
|
||||||
if (blockequ(line_buf, "exit", 5))
|
|
||||||
return;
|
|
||||||
if (!try_run_command_blocking(path_buf))
|
|
||||||
tell_user_sz("An error occured trying to run that command.\n");
|
|
||||||
}
|
|
||||||
}
|
|
100
src/user/highway/line.c
Normal file
100
src/user/highway/line.c
Normal file
|
@ -0,0 +1,100 @@
|
||||||
|
#include <knob/block.h>
|
||||||
|
#include <knob/quit.h>
|
||||||
|
#include <knob/task.h>
|
||||||
|
#include <knob/user.h>
|
||||||
|
#include "cmds.h"
|
||||||
|
#include "vars.h"
|
||||||
|
|
||||||
|
#define LINE_SIZE 4096
|
||||||
|
static char line[LINE_SIZE];
|
||||||
|
|
||||||
|
static void line_replace(const char *from) {
|
||||||
|
const char *fi = from;
|
||||||
|
char *ti = line;
|
||||||
|
while (*fi) {
|
||||||
|
if (ti == line + LINE_SIZE) {
|
||||||
|
tell_user_sz("Line too long.\n");
|
||||||
|
line[0] = '\0';
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
if (*fi != '$') {
|
||||||
|
*ti = *fi;
|
||||||
|
++ti;
|
||||||
|
++fi;
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
const char *var_start = ++fi;
|
||||||
|
const char *var_end = var_start;
|
||||||
|
while (*var_end != '$')
|
||||||
|
if (!*var_end++) {
|
||||||
|
tell_user_sz("Unterminated variable name.\n");
|
||||||
|
line[0] = '\0';
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
if (ti + (var_end - var_start) >= line + LINE_SIZE) {
|
||||||
|
tell_user_sz("Line too long.\n");
|
||||||
|
line[0] = '\0';
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
struct no_null_sn vname = {
|
||||||
|
.data = var_start,
|
||||||
|
.length = var_end - var_start
|
||||||
|
};
|
||||||
|
const struct no_null_sn *vval = get_var(vname);
|
||||||
|
if (!vval)
|
||||||
|
vval = &vname;
|
||||||
|
blockcpy(ti, vval->data, vval->length);
|
||||||
|
ti += vval->length;
|
||||||
|
fi = var_end + 1;
|
||||||
|
}
|
||||||
|
*ti = '\0';
|
||||||
|
}
|
||||||
|
|
||||||
|
void run_line(const char *original_line) {
|
||||||
|
line_replace(original_line);
|
||||||
|
if (!*line)
|
||||||
|
return;
|
||||||
|
const char *space;
|
||||||
|
for (space = line; *space && (*space != ' '); ++space)
|
||||||
|
;
|
||||||
|
if (blockequ(line, "source", space - line))
|
||||||
|
source(space + 1);
|
||||||
|
else if (blockequ(line, "set", space - line))
|
||||||
|
set(space + 1);
|
||||||
|
else if (blockequ(line, "echo", space - line)) {
|
||||||
|
tell_user_sz(space + 1);
|
||||||
|
tell_user_sz("\n");
|
||||||
|
}
|
||||||
|
else if (blockequ(line, "vars", space - line))
|
||||||
|
dump_vars();
|
||||||
|
else if (blockequ(line, "quit", space - line))
|
||||||
|
quit(space + 1);
|
||||||
|
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"
|
||||||
|
"of pseudo-commands. Variables are addressed by surrounding with \"$\". The\n"
|
||||||
|
"following list shows each of the pseudo-commands.\n\n"
|
||||||
|
" source FILE run each command in FILE\n"
|
||||||
|
" set VAR VALUE set $VAR$ to VALUE\n"
|
||||||
|
" echo STRING print STRING\n"
|
||||||
|
" vars dump variables\n"
|
||||||
|
" quit exit highway\n"
|
||||||
|
" help show this\n\n");
|
||||||
|
else if (!try_run_command_blocking(line)) {
|
||||||
|
struct no_null_sn arg = {
|
||||||
|
.data = "_path",
|
||||||
|
.length = 5
|
||||||
|
};
|
||||||
|
const struct no_null_sn *path = get_var(arg);
|
||||||
|
if (!path->length) {
|
||||||
|
tell_user_sz("Could not run command.\n");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
for (uint16_t to_i = LINE_SIZE - 1; to_i >= path->length; --to_i)
|
||||||
|
line[to_i] = line[to_i - path->length];
|
||||||
|
blockcpy(line, path->data, path->length);
|
||||||
|
if (!try_run_command_blocking(line)) {
|
||||||
|
tell_user_sz("Could not run command.\n");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
6
src/user/highway/line.h
Normal file
6
src/user/highway/line.h
Normal file
|
@ -0,0 +1,6 @@
|
||||||
|
#ifndef LINE_H
|
||||||
|
#define LINE_H
|
||||||
|
|
||||||
|
void run_line(const char *line);
|
||||||
|
|
||||||
|
#endif
|
16
src/user/highway/main.c
Normal file
16
src/user/highway/main.c
Normal file
|
@ -0,0 +1,16 @@
|
||||||
|
#include <knob/user.h>
|
||||||
|
#include <knob/task.h>
|
||||||
|
#include "cmds.h"
|
||||||
|
#include "line.h"
|
||||||
|
|
||||||
|
void main(const char *arg) {
|
||||||
|
source(*arg ? arg : "user/default.rc");
|
||||||
|
char cmd_buf[128];
|
||||||
|
yield_task();
|
||||||
|
tell_user_sz("Portland Highway\nType \"help\" for help.\n");
|
||||||
|
while (1) {
|
||||||
|
tell_user_sz("> ");
|
||||||
|
ask_user_line_sz(cmd_buf, 127);
|
||||||
|
run_line(cmd_buf);
|
||||||
|
}
|
||||||
|
}
|
98
src/user/highway/vars.c
Normal file
98
src/user/highway/vars.c
Normal file
|
@ -0,0 +1,98 @@
|
||||||
|
#include <knob/block.h>
|
||||||
|
#include <knob/heap.h>
|
||||||
|
#include <knob/user.h>
|
||||||
|
|
||||||
|
struct no_null_sn {
|
||||||
|
char *data;
|
||||||
|
uint32_t length;
|
||||||
|
};
|
||||||
|
|
||||||
|
struct var_dict_node {
|
||||||
|
struct var_dict_node *next;
|
||||||
|
struct var_dict_node *prev;
|
||||||
|
struct no_null_sn name;
|
||||||
|
struct no_null_sn value;
|
||||||
|
};
|
||||||
|
|
||||||
|
static struct var_dict_node *var_dict_start = 0;
|
||||||
|
|
||||||
|
__attribute__ ((pure))
|
||||||
|
static struct var_dict_node *get_node(struct no_null_sn name) {
|
||||||
|
for (struct var_dict_node *i = var_dict_start; i; i = i->next)
|
||||||
|
if ((i->name.length == name.length) &&
|
||||||
|
blockequ(i->name.data, name.data, name.length))
|
||||||
|
return i;
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
static struct var_dict_node *new_node() {
|
||||||
|
struct var_dict_node *node = get_block(sizeof(struct var_dict_node));
|
||||||
|
node->prev = 0;
|
||||||
|
node->next = var_dict_start;
|
||||||
|
if (var_dict_start)
|
||||||
|
var_dict_start->prev = node;
|
||||||
|
var_dict_start = node;
|
||||||
|
return node;
|
||||||
|
}
|
||||||
|
|
||||||
|
void set_var(struct no_null_sn name, struct no_null_sn value) {
|
||||||
|
struct var_dict_node *node = get_node(name);
|
||||||
|
if (node)
|
||||||
|
free_block(node->value.data);
|
||||||
|
else {
|
||||||
|
node = new_node();
|
||||||
|
node->name.length = name.length;
|
||||||
|
node->name.data = get_block(name.length);
|
||||||
|
blockcpy(node->name.data, name.data, name.length);
|
||||||
|
}
|
||||||
|
node->value.length = value.length;
|
||||||
|
node->value.data = get_block(value.length);
|
||||||
|
blockcpy(node->value.data, value.data, value.length);
|
||||||
|
}
|
||||||
|
|
||||||
|
__attribute__ ((pure))
|
||||||
|
const struct no_null_sn *get_var(struct no_null_sn name) {
|
||||||
|
struct var_dict_node *node = get_node(name);
|
||||||
|
return node ? &node->value : 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
void del_var(struct no_null_sn name) {
|
||||||
|
struct var_dict_node *node = get_node(name);
|
||||||
|
if (!node)
|
||||||
|
return;
|
||||||
|
free_block(node->name.data);
|
||||||
|
free_block(node->value.data);
|
||||||
|
if (node->prev)
|
||||||
|
node->prev->next = node->next;
|
||||||
|
if (node->next)
|
||||||
|
node->next->prev = node->prev;
|
||||||
|
if (node == var_dict_start)
|
||||||
|
var_dict_start = node->next;
|
||||||
|
free_block(node);
|
||||||
|
}
|
||||||
|
|
||||||
|
void dump_vars() {
|
||||||
|
for (struct var_dict_node *node = var_dict_start; node; node = node->next) {
|
||||||
|
tell_user_sz("$");
|
||||||
|
|
||||||
|
char *buf = get_block(node->name.length + 1);
|
||||||
|
blockcpy(buf, node->name.data, node->name.length);
|
||||||
|
buf[node->name.length] = '\0';
|
||||||
|
|
||||||
|
tell_user_sz(buf);
|
||||||
|
|
||||||
|
free_block(buf);
|
||||||
|
|
||||||
|
tell_user_sz("$ = ");
|
||||||
|
|
||||||
|
buf = get_block(node->value.length + 1);
|
||||||
|
blockcpy(buf, node->value.data, node->value.length);
|
||||||
|
buf[node->value.length] = '\0';
|
||||||
|
|
||||||
|
tell_user_sz(buf);
|
||||||
|
|
||||||
|
free_block(buf);
|
||||||
|
|
||||||
|
tell_user_sz("\n");
|
||||||
|
}
|
||||||
|
}
|
17
src/user/highway/vars.h
Normal file
17
src/user/highway/vars.h
Normal file
|
@ -0,0 +1,17 @@
|
||||||
|
#ifndef VARS_H
|
||||||
|
#define VARS_H
|
||||||
|
|
||||||
|
#include <stdint.h>
|
||||||
|
|
||||||
|
struct no_null_sn {
|
||||||
|
const char *data;
|
||||||
|
uint32_t length;
|
||||||
|
};
|
||||||
|
|
||||||
|
void set_var(struct no_null_sn name, struct no_null_sn value);
|
||||||
|
const struct no_null_sn *get_var(struct no_null_sn name) __attribute__ ((pure));
|
||||||
|
void del_var(struct no_null_sn name);
|
||||||
|
|
||||||
|
void dump_vars();
|
||||||
|
|
||||||
|
#endif
|
|
@ -1,7 +1,10 @@
|
||||||
#ifndef KNOB_BLOCK_H
|
#ifndef KNOB_BLOCK_H
|
||||||
#define KNOB_BLOCK_H
|
#define KNOB_BLOCK_H
|
||||||
|
|
||||||
|
#include <stdbool.h>
|
||||||
|
#include <stdint.h>
|
||||||
|
|
||||||
void blockcpy(void *to, const void *from, uint32_t size);
|
void blockcpy(void *to, const void *from, uint32_t size);
|
||||||
bool blockequ(void *a, void *b, uint32_t size) __attribute__ ((__pure__));
|
bool blockequ(const void *a, const void *b, uint32_t size) __attribute__ ((__pure__));
|
||||||
|
|
||||||
#endif
|
#endif
|
|
@ -9,8 +9,11 @@ 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
|
||||||
|
uint32_t read_line_from_file(struct file *f, char *sz, uint32_t max_length);
|
||||||
uint32_t seek_file_to(struct file *f, uint32_t to);
|
uint32_t seek_file_to(struct file *f, uint32_t to);
|
||||||
int32_t seek_file_by(struct file *f, int32_t by);
|
int32_t seek_file_by(struct file *f, int32_t by);
|
||||||
|
|
||||||
|
|
|
@ -4,7 +4,6 @@
|
||||||
#include <stdint.h>
|
#include <stdint.h>
|
||||||
|
|
||||||
void tell_user_sz(const char *sz);
|
void tell_user_sz(const char *sz);
|
||||||
void tell_user_n(uint32_t n);
|
|
||||||
|
|
||||||
//return value and max_length don't include null terminator
|
//return value and max_length don't include null terminator
|
||||||
//returns the real length of the string
|
//returns the real length of the string
|
||||||
|
|
|
@ -2,19 +2,6 @@
|
||||||
#include <knob/file.h>
|
#include <knob/file.h>
|
||||||
#include <knob/task.h>
|
#include <knob/task.h>
|
||||||
|
|
||||||
void start(const char *cmd) {
|
|
||||||
tell_user_sz("[init] Starting ");
|
|
||||||
tell_user_sz(cmd);
|
|
||||||
tell_user_sz(": ");
|
|
||||||
tell_user_sz(
|
|
||||||
run_command(cmd)
|
|
||||||
? "Succeded.\n"
|
|
||||||
: "Failed.\n"
|
|
||||||
);
|
|
||||||
tell_user_sz("[init] Yielding.\n");
|
|
||||||
yield_task();
|
|
||||||
}
|
|
||||||
|
|
||||||
#define STARTUP_FILE_PATH "sys/startup.rc"
|
#define STARTUP_FILE_PATH "sys/startup.rc"
|
||||||
|
|
||||||
void main() {
|
void main() {
|
||||||
|
@ -26,22 +13,17 @@ void main() {
|
||||||
|
|
||||||
tell_user_sz("[init] Reading from " STARTUP_FILE_PATH ".\n");
|
tell_user_sz("[init] Reading from " STARTUP_FILE_PATH ".\n");
|
||||||
|
|
||||||
char buf[1024];
|
char cmdbuf[128];
|
||||||
char *bufp = buf;
|
while (read_line_from_file(f, cmdbuf, 127)) {
|
||||||
while (read_from_file(f, 1, bufp)) {
|
tell_user_sz("[init] Starting ");
|
||||||
if (*bufp == '\n') {
|
tell_user_sz(cmdbuf);
|
||||||
if (bufp == buf)
|
tell_user_sz(": ");
|
||||||
continue;
|
if (run_command(cmdbuf)) {
|
||||||
*bufp = '\0';
|
tell_user_sz("Succeded.\n");
|
||||||
start(buf);
|
yield_task();
|
||||||
bufp = buf;
|
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
++bufp;
|
tell_user_sz("Failed.\n");
|
||||||
}
|
|
||||||
if (bufp != buf) {
|
|
||||||
*bufp = '\0';
|
|
||||||
start(buf);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
close_file(f);
|
close_file(f);
|
||||||
|
|
|
@ -9,7 +9,7 @@ void blockcpy(void *to, const void *from, uint32_t size) {
|
||||||
|
|
||||||
//unsophisticated, should check by dwords wheere available
|
//unsophisticated, should check by dwords wheere available
|
||||||
__attribute__ ((__pure__))
|
__attribute__ ((__pure__))
|
||||||
bool blockequ(void *a, void *b, uint32_t size) {
|
bool blockequ(const void *a, const void *b, uint32_t size) {
|
||||||
for (uint32_t i = 0; i < size; ++i)
|
for (uint32_t i = 0; i < size; ++i)
|
||||||
if (*(uint8_t *)(a++) != *(uint8_t *)(b++))
|
if (*(uint8_t *)(a++) != *(uint8_t *)(b++))
|
||||||
return false;
|
return false;
|
||||||
|
|
|
@ -3,7 +3,21 @@
|
||||||
#include <knob/heap.h>
|
#include <knob/heap.h>
|
||||||
#include <knob/env.h>
|
#include <knob/env.h>
|
||||||
|
|
||||||
|
struct ofl_node {
|
||||||
|
struct ofl_node *next;
|
||||||
|
struct ofl_node *prev;
|
||||||
|
_file_handle_t handle;
|
||||||
|
};
|
||||||
|
|
||||||
|
static struct ofl_node *head_ofl_node = 0;
|
||||||
|
|
||||||
|
void _close_all_files() {
|
||||||
|
for (struct ofl_node *i = head_ofl_node; i; i = i->next)
|
||||||
|
_close_file(i->handle);
|
||||||
|
}
|
||||||
|
|
||||||
struct file {
|
struct file {
|
||||||
|
struct ofl_node *node;
|
||||||
_file_handle_t handle;
|
_file_handle_t handle;
|
||||||
uint32_t position;
|
uint32_t position;
|
||||||
uint32_t length;
|
uint32_t length;
|
||||||
|
@ -38,7 +52,16 @@ struct file *open_file(const char *path) {
|
||||||
if (!h)
|
if (!h)
|
||||||
return 0;
|
return 0;
|
||||||
|
|
||||||
|
struct ofl_node *new_node = get_block(sizeof(struct ofl_node));
|
||||||
|
new_node->next = head_ofl_node;
|
||||||
|
new_node->prev = 0;
|
||||||
|
new_node->handle = h;
|
||||||
|
if (head_ofl_node)
|
||||||
|
head_ofl_node->prev = new_node;
|
||||||
|
head_ofl_node = new_node;
|
||||||
|
|
||||||
struct file *f = get_block(sizeof(struct file));
|
struct file *f = get_block(sizeof(struct file));
|
||||||
|
f->node = new_node;
|
||||||
f->handle = h;
|
f->handle = h;
|
||||||
f->position = 0;
|
f->position = 0;
|
||||||
f->length = _file_size(h);
|
f->length = _file_size(h);
|
||||||
|
@ -48,6 +71,14 @@ struct file *open_file(const char *path) {
|
||||||
|
|
||||||
void close_file(struct file *f) {
|
void close_file(struct file *f) {
|
||||||
_close_file(f->handle);
|
_close_file(f->handle);
|
||||||
|
struct ofl_node *n = f->node;
|
||||||
|
if (n->next)
|
||||||
|
n->next->prev = n->prev;
|
||||||
|
if (n->prev)
|
||||||
|
n->prev->next = n->next;
|
||||||
|
if (n == head_ofl_node)
|
||||||
|
head_ofl_node = n->next;
|
||||||
|
free_block(n);
|
||||||
free_block(f);
|
free_block(f);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -61,6 +92,19 @@ uint32_t read_from_file(struct file *f, uint32_t max, void *buf) {
|
||||||
return read;
|
return read;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
//return value and max_length don't include null terminator
|
||||||
|
uint32_t read_line_from_file(struct file *f, char *sz, uint32_t max_length) {
|
||||||
|
uint8_t i;
|
||||||
|
for (i = 0; i < max_length; ++i) {
|
||||||
|
char byte;
|
||||||
|
if (!read_from_file(f, 1, &byte) || (byte == '\n'))
|
||||||
|
break;
|
||||||
|
sz[i] = byte;
|
||||||
|
}
|
||||||
|
sz[i] = '\0';
|
||||||
|
return i;
|
||||||
|
}
|
||||||
|
|
||||||
uint32_t seek_file_to(struct file *f, uint32_t to) {
|
uint32_t seek_file_to(struct file *f, uint32_t to) {
|
||||||
if (to > f->length)
|
if (to > f->length)
|
||||||
to = f->length;
|
to = f->length;
|
||||||
|
|
|
@ -1,6 +1,8 @@
|
||||||
#include <pland/syscall.h>
|
#include <pland/syscall.h>
|
||||||
|
#include <knob/file.h>
|
||||||
|
|
||||||
__attribute__ ((noreturn))
|
__attribute__ ((noreturn))
|
||||||
void quit() {
|
void quit() {
|
||||||
|
_close_all_files();
|
||||||
_exit_task();
|
_exit_task();
|
||||||
}
|
}
|
|
@ -134,23 +134,6 @@ void tell_user_sz(const char *sz) {
|
||||||
_log_string(sz);
|
_log_string(sz);
|
||||||
}
|
}
|
||||||
|
|
||||||
void tell_user_n(uint32_t n) {
|
|
||||||
char buf[11];
|
|
||||||
char *buf_ptr = buf;
|
|
||||||
bool zero_yet = false;
|
|
||||||
for (uint32_t d = 1000000000U; d; d /= 10) {
|
|
||||||
uint8_t v = (n / d ) % 10;
|
|
||||||
if (v || zero_yet) {
|
|
||||||
zero_yet = true;
|
|
||||||
*buf_ptr++ = v | '0';
|
|
||||||
}
|
|
||||||
}
|
|
||||||
if (buf_ptr == buf)
|
|
||||||
*buf_ptr++ = '0';
|
|
||||||
*buf_ptr = '\0';
|
|
||||||
tell_user_sz(buf);
|
|
||||||
}
|
|
||||||
|
|
||||||
//return value and max_length don't include null terminator
|
//return value and max_length don't include null terminator
|
||||||
uint32_t ask_user_line_sz(char *sz, uint32_t max_length) {
|
uint32_t ask_user_line_sz(char *sz, uint32_t max_length) {
|
||||||
char log_buf[2];
|
char log_buf[2];
|
||||||
|
|
Reference in a new issue