vbe support, truecolor window manager pixbufs

This commit is contained in:
Benji Dial 2021-02-18 11:56:08 -05:00
parent 9d8ce7688f
commit 00cc8736f1
23 changed files with 434 additions and 117 deletions

View file

@ -6,8 +6,12 @@
0x2 byte: PCI minor 0x2 byte: PCI minor
0x3 byte: PCI major 0x3 byte: PCI major
0x4 byte: last PCI bus 0x4 byte: last PCI bus
;0x5 byte
0x6 word: BIOS memory map length 0x6 word: BIOS memory map length
0x8 word: active VBE mode
0x0000.4200 - 0x0000.43ff (512): VBE2 info block
0x0000.4400 - 0x0000.44ff (256): active VBE mode info
0x0000.4f98 - 0x0000.4fff (104): TSS 0x0000.4f98 - 0x0000.4fff (104): TSS
0x0000.5000 - 0x0000.5fff (4k): kernel page directory 0x0000.5000 - 0x0000.5fff (4k): kernel page directory

View file

@ -24,6 +24,8 @@ the edx register of "start task" is a pointer to a null-terminated string.
the new task receives its own handle in ecx. the new task receives its own handle in ecx.
ipc operations return 0xffffffff if the specified task doesn't exist. ipc operations return 0xffffffff if the specified task doesn't exist.
"find unread ipc" system call returns 0 if there is no unread ipc. "find unread ipc" system call returns 0 if there is no unread ipc.
pixbufs are arrays of four-byte pixels, from left to right, then top to bottom.
pixels are a red byte, a green byte, a blue byte, and a padding byte.
invalid system call numbers change eax to -1, and have no other effect. invalid system call numbers change eax to -1, and have no other effect.

View file

@ -8,6 +8,11 @@ support_flags equ 0x4000
pci_hw_char equ 0x4001 pci_hw_char equ 0x4001
pci_ver equ 0x4002 pci_ver equ 0x4002
last_pci_bus equ 0x4004 last_pci_bus equ 0x4004
bios_mmap_len equ 0x4006
vbe_mode equ 0x4008
vbe_block equ 0x4200
vbe_mode_info equ 0x4400
pci_support equ 0x80 pci_support equ 0x80
pae_support equ 0x40 pae_support equ 0x40
@ -18,10 +23,7 @@ pae_support equ 0x40
xor ax, ax xor ax, ax
mov ss, ax mov ss, ax
mov sp, 0x7ffc mov sp, 0x8000
mov ax, 0x0013
int 0x10
mov ax, kernel_segment mov ax, kernel_segment
mov es, ax mov es, ax
@ -72,7 +74,95 @@ mmap_loop:
test ebx, ebx test ebx, ebx
jnz mmap_loop jnz mmap_loop
mmap_end: mmap_end:
mov word [0x4006], di mov word [bios_mmap_len], di
xor ax, ax
mov es, ax
mov di, vbe_block
mov ah, 0x4f
mov dword [di], 0x32454256;'VBE2'
int 0x10
cmp dword [di], 0x41534556;'VESA'
jne no_vbe
mov bx, word [di + 0x0e]
mov ax, word [di + 0x10]
mov fs, ax
xor esi, esi
mov bp, 0xffff
mov di, 0x5000
vbe_loop:
max_height equ 800
;fs:bx is mode pointer
;esi is highest resolution yet (width * height)
;bp is that mode's number
;uses 0x5000 - 0x50ff as mode info buffer
;skips those without byte-aligned pixels
mov cx, word [fs:bx]
cmp cx, 0xffff
je got_highest_vbe
add bx, 4
mov ax, 0x4f01
int 0x10
mov al, byte [di]
test al, 0x01
jz vbe_loop
test al, 0x10
jz vbe_loop
test al, 0x80
jz vbe_loop
mov dx, word [di + 0x14];height
cmp dx, max_height
jg vbe_loop
mov ax, word [di + 0x12];width
mul dx
shl edx, 16
mov dx, ax
xor eax, eax
mov al, byte [di + 0x19];bpp
test al, 0x07
jnz vbe_loop
mul edx
cmp eax, esi
jle vbe_loop
mov esi, eax
mov bp, cx
jmp vbe_loop
got_highest_vbe:
cmp bp, 0xffff
je no_vbe
mov cx, bp
mov ax, 0x4f01
mov di, vbe_mode_info
int 0x10
or ch, 0x40
mov word [vbe_mode], cx
mov ax, 0x4f02
mov bx, cx
int 0x10
cli cli
@ -84,6 +174,25 @@ mmap_end:
jmp 0x10:pmode jmp 0x10:pmode
no_vbe:
xor ax, ax
mov es, ax
mov ax, 0x1300
mov bx, 0x004f
mov cx, no_vbe_str.len
xor dx, dx
mov bp, no_vbe_str
int 0x10
real_halt:
hlt
jmp real_halt
no_vbe_str:
db "NO VBE2!"
.len equ $ - no_vbe_str
bits 32 bits 32
pmode: pmode:

View file

@ -52,8 +52,16 @@ void logf(enum log_level level, const char *format, ...) {
while (*s) while (*s)
logch(*(s++)); logch(*(s++));
break; break;
case 'n':;
uint32_t d = va_arg(args, uint32_t);
if (d >= 0x80000000) {
logch('-');
d = -d;
}
goto put_dec;
case 'd':; case 'd':;
const uint32_t d = va_arg(args, uint32_t); d = va_arg(args, uint32_t);
put_dec:
if (d == 0) { if (d == 0) {
logch('0'); logch('0');
break; break;

View file

@ -15,6 +15,7 @@
#include "kbd.h" #include "kbd.h"
#include "log.h" #include "log.h"
#include "pci.h" #include "pci.h"
#include "vbe.h"
void _start_user_mode() __attribute__ ((noreturn)); void _start_user_mode() __attribute__ ((noreturn));
@ -46,12 +47,16 @@ void main() {
logf(LOG_INFO, "Available kernel memory: %dk", kernel_pages_left * 4); logf(LOG_INFO, "Available kernel memory: %dk", kernel_pages_left * 4);
logf(LOG_INFO, "Available user memory: %dk", user_pages_left * 4); logf(LOG_INFO, "Available user memory: %dk", user_pages_left * 4);
logf(LOG_INFO, "");
logf(LOG_INFO, "PCI devices:"); logf(LOG_INFO, "PCI devices:");
for (uint16_t i = 0; i < n_pci_devices; ++i) { for (uint16_t i = 0; i < n_pci_devices; ++i) {
const struct pci_device *dev = nth_pci_device(i); const struct pci_device *dev = nth_pci_device(i);
logf(LOG_INFO, " %hw:%hw (%hb:%hb)", dev->id_vendor, dev->id_device, dev->class, dev->subclass); logf(LOG_INFO, " %hw:%hw (%hb:%hb)", dev->id_vendor, dev->id_device, dev->class, dev->subclass);
} }
logf(LOG_INFO, "");
logf(LOG_INFO, "Drives:"); logf(LOG_INFO, "Drives:");
for (uint8_t i = 0; i < n_drives; ++i) { for (uint8_t i = 0; i < n_drives; ++i) {
const struct drive *d = &drives[i]; const struct drive *d = &drives[i];
@ -59,6 +64,29 @@ void main() {
logf(LOG_INFO, " %s: %d%sk, %s (%d%sk free)", d->drive_type, d->n_sectors / 2, d->n_sectors % 2 ? ".5" : "", d->fs_type, free / 2, free % 2 ? ".5" : ""); logf(LOG_INFO, " %s: %d%sk, %s (%d%sk free)", d->drive_type, d->n_sectors / 2, d->n_sectors % 2 ? ".5" : "", d->fs_type, free / 2, free % 2 ? ".5" : "");
} }
logf(LOG_INFO, "");
logf(LOG_INFO, "VBE info:");
logf(LOG_INFO, " Implemention: %s", RM_PTR(char, VBE_INFO->oem_name));
logf(LOG_INFO, " Video memory: %dk", VBE_INFO->total_memory * 64);
logf(LOG_INFO, " Standard: %d.%d.%d", VBE_INFO->major_version, VBE_INFO->minor_version, VBE_INFO->version_rev);
logf(LOG_INFO, " Vendor: %s", RM_PTR(char, VBE_INFO->vendor_name));
logf(LOG_INFO, " Product: %s", RM_PTR(char, VBE_INFO->product_name));
logf(LOG_INFO, " Version: %s", RM_PTR(char, VBE_INFO->product_rev_name));
logf(LOG_INFO, "");
#define MASK(offset, length) ((~((1 << offset) - 1)) - (offset + length == 32 ? 0 : (~((1 << (offset + length)) - 1))))
logf(LOG_INFO, "Active video mode:");
logf(LOG_INFO, " Resolution: %dx%dx%d", VBE_MODE_INFO->width, VBE_MODE_INFO->height, VBE_MODE_INFO->bpp);
logf(LOG_INFO, " Red mask: 0x%h", MASK(VBE_MODE_INFO->red_off, VBE_MODE_INFO->red_len));
logf(LOG_INFO, " Green mask: 0x%h", MASK(VBE_MODE_INFO->green_off, VBE_MODE_INFO->green_len));
logf(LOG_INFO, " Blue mask: 0x%h", MASK(VBE_MODE_INFO->blue_off, VBE_MODE_INFO->blue_len));
logf(LOG_INFO, " Alpha mask: 0x%h", MASK(VBE_MODE_INFO->alpha_off, VBE_MODE_INFO->alpha_len));
logf(LOG_INFO, " Framebuffer address: 0x%h", VBE_MODE_INFO->frame_buf);
logf(LOG_INFO, "");
logf(LOG_INFO, "Loading init program."); logf(LOG_INFO, "Loading init program.");
if (!try_elf_run(drives, "bin/init", "", 0)) if (!try_elf_run(drives, "bin/init", "", 0))

71
src/kernel/vbe.h Normal file
View file

@ -0,0 +1,71 @@
#ifndef VBE_H
#define VBE_H
#include <stdint.h>
struct segoff {
uint16_t offset;
uint16_t segment;
} __attribute__ ((__packed__));
#define RM_PTR(type, segoff) ((type *)((segoff).segment * 16 + (segoff).offset))
enum vbe_capabilities {
VBE_CAP_SWITCHABLE_DAC = 0x1,
VBE_CAP_NON_VGA = 0x2,
VBE_CAP_WEIRD_RAMDAC = 0x4
};
#define VBE_INFO \
((struct { \
uint8_t signature[4]; \
uint8_t minor_version; \
uint8_t major_version; \
struct segoff oem_name; \
uint32_t capabilities; \
struct segoff mode_list; \
/*in units of 64kiB*/ \
uint16_t total_memory; \
uint16_t version_rev; \
struct segoff vendor_name; \
struct segoff product_name; \
struct segoff product_rev_name; \
} __attribute__ ((__packed__)) *)0x4200)
#define VBE_MODE_INFO \
((struct { \
uint16_t attribs; \
uint8_t wina_attribs; \
uint8_t winb_attribs; \
uint16_t win_gran; \
uint16_t win_size; \
uint16_t wina_seg; \
uint16_t winb_seg; \
struct segoff win_func; \
uint16_t pitch;/*in bytes*/ \
uint16_t width; \
uint16_t height; \
uint8_t char_width; \
uint8_t char_height; \
uint8_t n_planes; \
uint8_t bpp; \
uint8_t n_banks; \
uint8_t mem_model; \
uint8_t bank_size;/*in kiB*/\
uint8_t image_pages; \
uint8_t reserved; \
uint8_t red_len; \
uint8_t red_off; \
uint8_t green_len; \
uint8_t green_off; \
uint8_t blue_len; \
uint8_t blue_off; \
uint8_t alpha_len; \
uint8_t alpha_off; \
uint8_t color_attribs; \
void *frame_buf; \
void *off_screen; \
uint16_t off_screen_length; \
} __attribute__ ((__packed__)) *)0x4400)
#endif

View file

@ -7,18 +7,21 @@
#include "pmap.h" #include "pmap.h"
#include "elf.h" #include "elf.h"
#include "log.h" #include "log.h"
#include "vbe.h"
#define MAX_WINDOWS 64 #define MAX_WINDOWS 64
#define ACTION_BUFFER_PAGES 1 #define ACTION_BUFFER_PAGES 1
#define AB_END(buf) (buf + (ACTION_BUFFER_PAGES * 4096) / sizeof(struct window_action)) #define AB_END(buf) (buf + (ACTION_BUFFER_PAGES * 4096) / sizeof(struct window_action))
#define BACKGROUND(x, y) ((x / 16 + y / 16) % 2 ? 0x14 : 0x07) #define GSC(x, y) (x / 4 + y / 4 >= 256 ? 255 : x / 4 + y / 4)
#define BACKGROUND(x, y) ((struct pixel){.r = GSC(x / 2, y / 3), .g = GSC(x, y), .b = GSC(x, y / 2)})
#define BORDER_LIGHT 0x1c #define BORDER_COLOR ((struct pixel){.r = 0x11, .g = 0x11, .b = 0x11})
#define BORDER_DARK 0x08
#define MOVE_BY 5
static struct window { static struct window {
const volatile void *pixel_buffer_pma; const volatile struct pixel *pixel_buffer_pma;
uint16_t width; uint16_t width;
uint16_t height; uint16_t height;
@ -38,65 +41,74 @@ static struct window {
static struct window *bottom_window = 0; static struct window *bottom_window = 0;
static struct window *top_window = 0; static struct window *top_window = 0;
#define SCREEN_HEIGHT 200 uint32_t pix_clear_mask;
#define SCREEN_WIDTH 320
#define VGA_MEMORY ((uint8_t *)0xa0000)
static inline void draw_hz_line(uint16_t y, uint16_t xs, uint16_t xm, uint8_t color) { static inline void put_pix(uint16_t y, uint16_t x, struct pixel px) {
if (y >= SCREEN_HEIGHT) uint32_t *const fbp = VBE_MODE_INFO->frame_buf + y * VBE_MODE_INFO->pitch + x * VBE_MODE_INFO->bpp / 8;
return; //logf(LOG_INFO, "-- fbp = 0x%h", fbp);
if (xs >= SCREEN_WIDTH) //logf(LOG_INFO, "-- *fbp -> 0x%h", *fbp);
xs = 0; *fbp &= pix_clear_mask;
if (xm > SCREEN_WIDTH) *fbp |= (px.r >> (8 - VBE_MODE_INFO->red_len)) << VBE_MODE_INFO->red_off;
xm = SCREEN_WIDTH; *fbp |= (px.g >> (8 - VBE_MODE_INFO->green_len)) << VBE_MODE_INFO->green_off;
uint8_t *const line_start = VGA_MEMORY + y * SCREEN_WIDTH; *fbp |= (px.b >> (8 - VBE_MODE_INFO->blue_len)) << VBE_MODE_INFO->blue_off;
for (uint8_t *p = line_start + xs; p < line_start + xm; ++p) //logf(LOG_INFO, "-- *fbp <- 0x%h", *fbp);
*p = color;
} }
static inline void draw_vt_line(uint16_t x, uint16_t ys, uint16_t ym, uint8_t color) { static inline void draw_hz_line(uint16_t y, uint16_t xs, uint16_t xm, struct pixel color) {
if (x >= SCREEN_WIDTH) if (y >= VBE_MODE_INFO->height)
return; return;
if (ys >= SCREEN_HEIGHT) if (xs >= VBE_MODE_INFO->width)
xs = 0;
if (xm > VBE_MODE_INFO->width)
xm = VBE_MODE_INFO->width;
for (uint16_t x = xs; x < xm; ++x)
put_pix(y, x, color);
}
static inline void draw_vt_line(uint16_t x, uint16_t ys, uint16_t ym, struct pixel color) {
if (x >= VBE_MODE_INFO->width)
return;
if (ys >= VBE_MODE_INFO->height)
ys = 0; ys = 0;
if (ym > SCREEN_HEIGHT) if (ym > VBE_MODE_INFO->height)
ym = SCREEN_HEIGHT; ym = VBE_MODE_INFO->height;
uint8_t *const line_start = VGA_MEMORY + x; for (uint16_t y = ys; y < ym; ++y)
for (uint8_t *p = line_start + ys * SCREEN_WIDTH; p < line_start + ym * SCREEN_WIDTH; p += SCREEN_WIDTH) put_pix(y, x, color);
*p = color;
} }
static void paint_and_above(const struct window *w) { static void paint_and_above(const struct window *w) {
switch_to_kernel_cr3(); switch_to_kernel_cr3();
for (const struct window *i = w; i; i = i->above) { for (const struct window *i = w; i; i = i->above) {
const uint16_t ys = i->ypos < SCREEN_HEIGHT ? 0 : -i->ypos; const uint16_t ys = i->ypos < VBE_MODE_INFO->height ? 0 : -i->ypos;
const uint16_t xs = i->xpos < SCREEN_WIDTH ? 0 : -i->xpos; const uint16_t xs = i->xpos < VBE_MODE_INFO->width ? 0 : -i->xpos;
const uint16_t ym = i->ypos + i->height <= SCREEN_HEIGHT ? i->height : SCREEN_HEIGHT - i->ypos; const uint16_t ym = ((i->ypos + i->height) & 0xffff) <= VBE_MODE_INFO->height ? i->height : VBE_MODE_INFO->height - i->ypos;
const uint16_t xm = i->xpos + i->width <= SCREEN_WIDTH ? i->width : SCREEN_WIDTH - i->xpos; const uint16_t xm = ((i->xpos + i->width) & 0xffff) <= VBE_MODE_INFO->width ? i->width : VBE_MODE_INFO->width - i->xpos;
//logf(LOG_INFO, "y: %n .. %n", ys, ym - 1);
//logf(LOG_INFO, "x: %n .. %n", xs, xm - 1);
for (uint16_t y = ys; y < ym; ++y) for (uint16_t y = ys; y < ym; ++y)
for (uint16_t x = xs; x < xm; ++x) { for (uint16_t x = xs; x < xm; ++x)
const uint8_t pixel = ((uint8_t *)i->pixel_buffer_pma)[y * i->width + x]; put_pix(y + i->ypos, x + i->xpos, i->pixel_buffer_pma[y * i->width + x]);
if (pixel)
VGA_MEMORY[(y + i->ypos) * SCREEN_WIDTH + x + i->xpos] = pixel;
}
draw_hz_line(i->ypos - 2, i->xpos - 2, i->xpos + i->width + 2, BORDER_LIGHT); draw_hz_line(i->ypos - 2, i->xpos - 2, i->xpos + i->width + 2, BORDER_COLOR);
draw_vt_line(i->xpos - 2, i->ypos - 1, i->ypos + i->height + 2, BORDER_LIGHT); draw_hz_line(i->ypos - 1, i->xpos - 2, i->xpos + i->width + 2, BORDER_COLOR);
draw_hz_line(i->ypos - 1, i->xpos - 1, i->xpos + i->width + 2, BORDER_DARK); draw_hz_line(i->ypos + i->height, i->xpos - 2, i->xpos + i->width + 2, BORDER_COLOR);
draw_vt_line(i->xpos - 1, i->ypos, i->ypos + i->height + 2, BORDER_DARK); draw_hz_line(i->ypos + i->height + 1, i->xpos - 2, i->xpos + i->width + 2, BORDER_COLOR);
draw_hz_line(i->ypos + i->height, i->xpos, i->xpos + i->width + 1, BORDER_LIGHT); draw_vt_line(i->xpos - 2, i->ypos, i->ypos + i->height, BORDER_COLOR);
draw_vt_line(i->xpos + i->width, i->ypos, i->ypos + i->height, BORDER_LIGHT); draw_vt_line(i->xpos - 1, i->ypos, i->ypos + i->height, BORDER_COLOR);
draw_hz_line(i->ypos + i->height + 1, i->xpos, i->xpos + i->width + 2, BORDER_DARK); draw_vt_line(i->xpos + i->width, i->ypos, i->ypos + i->height, BORDER_COLOR);
draw_vt_line(i->xpos + i->width + 1, i->ypos, i->ypos + i->height + 1, BORDER_DARK); draw_vt_line(i->xpos + i->width + 1, i->ypos, i->ypos + i->height, BORDER_COLOR);
} }
switch_to_task_cr3(); switch_to_task_cr3();
} }
static void paint_bg() { static void paint_bg() {
for (uint16_t y = 0; y < SCREEN_HEIGHT; ++y) switch_to_kernel_cr3();
for (uint16_t x = 0; x < SCREEN_WIDTH; ++x) for (uint16_t y = 0; y < VBE_MODE_INFO->height; ++y)
VGA_MEMORY[y * SCREEN_WIDTH + x] = BACKGROUND(x, y); for (uint16_t x = 0; x < VBE_MODE_INFO->width; ++x)
put_pix(y, x, BACKGROUND(x, y));
switch_to_task_cr3();
} }
static void paint_all() { static void paint_all() {
@ -104,7 +116,7 @@ static void paint_all() {
paint_and_above(bottom_window); paint_and_above(bottom_window);
} }
struct window *new_window(uint16_t width, uint16_t height, const void *pixel_buffer) { struct window *new_window(uint16_t width, uint16_t height, const struct pixel *pixel_buffer) {
if (!pixel_buffer) { if (!pixel_buffer) {
logf(LOG_WARN, "Refusing to create window with null pixel buffer for task %s.", active_task->name); logf(LOG_WARN, "Refusing to create window with null pixel buffer for task %s.", active_task->name);
return 0; return 0;
@ -122,8 +134,8 @@ got_window:
w->pixel_buffer_pma = vma_to_pma(active_task->page_directory, pixel_buffer); w->pixel_buffer_pma = vma_to_pma(active_task->page_directory, pixel_buffer);
w->width = width; w->width = width;
w->height = height; w->height = height;
w->xpos = 10; w->xpos = (VBE_MODE_INFO->width / 2) - (width / 2);
w->ypos = 10; w->ypos = (VBE_MODE_INFO->height / 2) - (height / 2);
struct window_action *const ab = allocate_kernel_pages(ACTION_BUFFER_PAGES); struct window_action *const ab = allocate_kernel_pages(ACTION_BUFFER_PAGES);
w->action_buffer = ab; w->action_buffer = ab;
@ -174,7 +186,7 @@ void delete_any_windows_from(struct task_state *tstate) {
paint_all(); paint_all();
} }
void resize_window(struct window *w, uint16_t width, uint16_t height, const void *pixel_buffer) { void resize_window(struct window *w, uint16_t width, uint16_t height, const struct pixel *pixel_buffer) {
const bool smaller = (width < w->width) || (height < w->height); const bool smaller = (width < w->width) || (height < w->height);
w->width = width; w->width = width;
@ -187,7 +199,7 @@ void resize_window(struct window *w, uint16_t width, uint16_t height, const void
paint_and_above(w); paint_and_above(w);
} }
void reassign_pixel_buffer(struct window *w, const void *pixel_buffer) { void reassign_pixel_buffer(struct window *w, const struct pixel *pixel_buffer) {
w->pixel_buffer_pma = vma_to_pma(active_task->page_directory, pixel_buffer); w->pixel_buffer_pma = vma_to_pma(active_task->page_directory, pixel_buffer);
} }
@ -259,6 +271,14 @@ static char *run_command;
static char *run_command_pass = ""; static char *run_command_pass = "";
void init_win() { void init_win() {
#define MASK(offset, length) ((~((1 << offset) - 1)) - (offset + length == 32 ? 0 : (~((1 << (offset + length)) - 1))))
pix_clear_mask = ~(
MASK(VBE_MODE_INFO->red_off, VBE_MODE_INFO->red_len) |
MASK(VBE_MODE_INFO->green_off, VBE_MODE_INFO->green_len) |
MASK(VBE_MODE_INFO->blue_off, VBE_MODE_INFO->blue_len) |
MASK(VBE_MODE_INFO->alpha_off, VBE_MODE_INFO->alpha_len)
);
paint_bg(); paint_bg();
const file_id_t fid = drives->get_file(drives, RUN_COMMAND_FILE); const file_id_t fid = drives->get_file(drives, RUN_COMMAND_FILE);
@ -323,25 +343,25 @@ void on_action(struct window_action packet) {
break; break;
case WM_MOVE_LEFT: case WM_MOVE_LEFT:
if (top_window) { if (top_window) {
--top_window->xpos; top_window->xpos -= MOVE_BY;
paint_all(); paint_all();
} }
break; break;
case WM_MOVE_RIGHT: case WM_MOVE_RIGHT:
if (top_window) { if (top_window) {
++top_window->xpos; top_window->xpos += MOVE_BY;
paint_all(); paint_all();
} }
break; break;
case WM_MOVE_UP: case WM_MOVE_UP:
if (top_window) { if (top_window) {
--top_window->ypos; top_window->ypos -= MOVE_BY;
paint_all(); paint_all();
} }
break; break;
case WM_MOVE_DOWN: case WM_MOVE_DOWN:
if (top_window) { if (top_window) {
++top_window->ypos; top_window->ypos += MOVE_BY;
paint_all(); paint_all();
} }
break; break;

View file

@ -9,13 +9,20 @@
struct window; struct window;
struct pixel {
uint8_t r;
uint8_t g;
uint8_t b;
uint8_t pad;
} __attribute__ ((__packed__));
void init_win(); void init_win();
struct window *new_window(uint16_t width, uint16_t height, const void *pixel_buffer); struct window *new_window(uint16_t width, uint16_t height, const struct pixel *pixel_buffer);
void del_window(struct window *w); void del_window(struct window *w);
void resize_window(struct window *w, uint16_t width, uint16_t height, const void *pixel_buffer); void resize_window(struct window *w, uint16_t width, uint16_t height, const struct pixel *pixel_buffer);
void reassign_pixel_buffer(struct window *w, const void *pixel_buffer); void reassign_pixel_buffer(struct window *w, const struct pixel *pixel_buffer);
void push_window_paint(const struct window *w); void push_window_paint(const struct window *w);
struct window_action next_window_action(struct window *w); struct window_action next_window_action(struct window *w);
void wait_window_action(); void wait_window_action();

View file

@ -12,18 +12,33 @@
#define LINE_SIZE 4096 #define LINE_SIZE 4096
static char line[LINE_SIZE]; static char line[LINE_SIZE];
static inline uint8_t hex_to_int(char ch) { static inline uint8_t hex_digit_to_int(char ch) {
return ch - (ch <= '9' ? '0' : 'a' - 10); return ch - (ch <= '9' ? '0' : 'a' - 10);
} }
static inline uint8_t hex_byte_to_int(const char *ch) {
return (hex_digit_to_int(ch[0]) << 4) | hex_digit_to_int(ch[1]);
}
static inline _pixel_t hex_to_pixel(const char *ch) {
return (_pixel_t){
.r = hex_byte_to_int(ch),
.g = hex_byte_to_int(ch + 2),
.b = hex_byte_to_int(ch + 4)
};
}
void ensure_color() { void ensure_color() {
const struct no_null_sn *color = get_var((struct no_null_sn){.data = "_color", .length = 6}); const struct no_null_sn *color;
//fgbg get_color:
if (color) color = get_var((struct no_null_sn){.data = "_color", .length = 6});
term_set_color( if (color && (color->length == 12))
(hex_to_int(color->data[0]) << 4) | hex_to_int(color->data[1]), //rgb, fgbg
(hex_to_int(color->data[2]) << 4) | hex_to_int(color->data[3]) term_set_color(hex_to_pixel(color->data), hex_to_pixel(color->data + 6));
); else {
new_color();
goto get_color;
}
} }
static void line_replace(const char *from) { static void line_replace(const char *from) {

View file

@ -8,7 +8,6 @@
#include "vars.h" #include "vars.h"
void main(const char *arg) { void main(const char *arg) {
new_color();
source(*arg ? arg : "user/default.rc"); source(*arg ? arg : "user/default.rc");
ensure_color(); ensure_color();

View file

@ -84,7 +84,7 @@ void dump_vars() {
} }
static const char hex_digits[] = "0123456789abcdef"; static const char hex_digits[] = "0123456789abcdef";
static char color[] = {'1', '0', '.', '.'}; static char color[12] = "000000......";
static const struct no_null_sn color_name = { static const struct no_null_sn color_name = {
.data = "_color", .data = "_color",
@ -93,12 +93,46 @@ static const struct no_null_sn color_name = {
static const struct no_null_sn color_value = { static const struct no_null_sn color_value = {
.data = color, .data = color,
.length = 4 .length = 12
}; };
void new_color() { void new_color() {
const uint8_t bg = gen_rand() % 0x30 + 0x38; const uint16_t hue = gen_rand() % (0x33 * 6);
color[2] = hex_digits[bg >> 4]; _pixel_t bg = {.r = 0xff, .g = 0xcc, .b = 0xcc};
color[3] = hex_digits[bg & 0xf]; if (hue <= 0x33) {
bg.b += hue;
goto got_bg;
}
bg.b += 0x33;
if (hue <= 0x33 * 2) {
bg.r -= hue - 0x33;
goto got_bg;
}
bg.r -= 0x33;
if (hue <= 0x33 * 3) {
bg.g += hue - 0x33 * 2;
goto got_bg;
}
bg.g += 0x33;
if (hue <= 0x33 * 4) {
bg.b -= hue - 0x33 * 3;
goto got_bg;
}
bg.b -= 0x33;
if (hue <= 0x33 * 5) {
bg.r += hue - 0x33 * 4;
goto got_bg;
}
bg.r += 0x33;
bg.g -= hue - 0x33 * 5;
got_bg:
color[ 6] = hex_digits[bg.r / 16];
color[ 7] = hex_digits[bg.r % 16];
color[ 8] = hex_digits[bg.g / 16];
color[ 9] = hex_digits[bg.g % 16];
color[10] = hex_digits[bg.b / 16];
color[11] = hex_digits[bg.b % 16];
set_var(color_name, color_value); set_var(color_name, color_value);
} }

View file

@ -1,6 +1,8 @@
#ifndef LIBFONT_FONTS_H #ifndef LIBFONT_FONTS_H
#define LIBFONT_FONTS_H #define LIBFONT_FONTS_H
#include <pland/syscall.h>
#include <stdbool.h> #include <stdbool.h>
#include <stdint.h> #include <stdint.h>
@ -16,6 +18,7 @@ struct font_info {
struct font_info *get_font(const char *name); struct font_info *get_font(const char *name);
void put_char(const struct font_info *font, char ch, uint8_t *pb_ptr, uint32_t pb_pitch, uint8_t bg, uint8_t fg); //pitch is in pixels
void put_char(const struct font_info *font, char ch, _pixel_t *pb_ptr, uint32_t pb_pitch, _pixel_t bg, _pixel_t fg);
#endif #endif

View file

@ -37,8 +37,8 @@ struct terminal_command {
uint32_t x; uint32_t x;
} as_coords; } as_coords;
struct { struct {
uint8_t fg; _pixel_t fg;
uint8_t bg; _pixel_t bg;
} as_color; } as_color;
char as_char; char as_char;
uint32_t as_uint; uint32_t as_uint;

View file

@ -12,7 +12,7 @@ void term_get_dimensions(uint32_t *width, uint32_t *height);
void term_paint(); void term_paint();
void term_clear(); void term_clear();
void term_set_color(uint8_t fg, uint8_t bg); void term_set_color(_pixel_t fg, _pixel_t bg);
void term_set_cursor(uint32_t new_y, uint32_t new_x); void term_set_cursor(uint32_t new_y, uint32_t new_x);
void term_cursor_left(); void term_cursor_left();

View file

@ -18,6 +18,13 @@ typedef struct __attribute__ ((packed)) {
uint8_t pad[23]; uint8_t pad[23];
} _dir_info_entry_t; } _dir_info_entry_t;
typedef struct __attribute__ ((packed)) {
uint8_t r;
uint8_t g;
uint8_t b;
uint8_t pad;
} _pixel_t;
enum _scn { enum _scn {
_SCN_OPEN_FILE, _SCN_OPEN_FILE,
_SCN_CLOSE_FILE, _SCN_CLOSE_FILE,
@ -177,7 +184,7 @@ 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); return _sc2(_SCN_COUNT_OF_DIR, drive_number, (uint32_t)path);
} }
static inline _window_handle_t _new_window(uint16_t width, uint16_t height, void *pixel_buffer) { static inline _window_handle_t _new_window(uint16_t width, uint16_t height, _pixel_t *pixel_buffer) {
return (_window_handle_t)_sc3(_SCN_NEW_WINDOW, width, height, (uint32_t)pixel_buffer); return (_window_handle_t)_sc3(_SCN_NEW_WINDOW, width, height, (uint32_t)pixel_buffer);
} }
@ -185,11 +192,11 @@ static inline void _delete_window(_window_handle_t window) {
_sc1(_SCN_DELETE_WINDOW, (uint32_t)window); _sc1(_SCN_DELETE_WINDOW, (uint32_t)window);
} }
static inline void _resize_window(_window_handle_t window, uint16_t width, uint16_t height, const void *pixel_buffer) { static inline void _resize_window(_window_handle_t window, uint16_t width, uint16_t height, const _pixel_t *pixel_buffer) {
_sc4(_SCN_RESIZE_WINDOW, (uint32_t)window, width, height, (uint32_t)pixel_buffer); _sc4(_SCN_RESIZE_WINDOW, (uint32_t)window, width, height, (uint32_t)pixel_buffer);
} }
static inline void _reassign_pixbuf(_window_handle_t window, const void *pixel_buffer) { static inline void _reassign_pixbuf(_window_handle_t window, const _pixel_t *pixel_buffer) {
_sc2(_SCN_REASSIGN_PIXBUF, (uint32_t)window, (uint32_t)pixel_buffer); _sc2(_SCN_REASSIGN_PIXBUF, (uint32_t)window, (uint32_t)pixel_buffer);
} }

View file

@ -7,8 +7,8 @@
#include <stdarg.h> #include <stdarg.h>
void info_popup(struct popup *into, const char *text, uint8_t fg, uint8_t bg); void info_popup(struct popup *into, const char *text, _pixel_t fg, _pixel_t bg);
void info_popupf(struct popup *into, const char *text, uint8_t fg, uint8_t bg, ...); void info_popupf(struct popup *into, const char *text, _pixel_t fg, _pixel_t bg, ...);
void info_popupf_v(struct popup *into, const char *text, uint8_t fg, uint8_t bg, va_list args); void info_popupf_v(struct popup *into, const char *text, _pixel_t fg, _pixel_t bg, va_list args);
#endif #endif

View file

@ -5,7 +5,7 @@
struct popup { struct popup {
_window_handle_t handle; _window_handle_t handle;
uint8_t *pixbuf; _pixel_t *pixbuf;
bool has_quit; bool has_quit;
struct key_packet quit_as; struct key_packet quit_as;

View file

@ -56,7 +56,8 @@ struct font_info *get_font(const char *name) {
return 0; return 0;
} }
void put_char(const struct font_info *font, char ch, uint8_t *pb_ptr, uint32_t pb_pitch, uint8_t bg, uint8_t fg) { //pitch is in pixels
void put_char(const struct font_info *font, char ch, _pixel_t *pb_ptr, uint32_t pb_pitch, _pixel_t bg, _pixel_t fg) {
//char *const msg = format("put_char(font = 0x%x, ch = '%c', pb_ptr = 0x%x, pb_pitch = %u, bg = 0x%2x, fg = 0x%2x);", font, ch, pb_ptr, pb_pitch, bg, fg); //char *const msg = format("put_char(font = 0x%x, ch = '%c', pb_ptr = 0x%x, pb_pitch = %u, bg = 0x%2x, fg = 0x%2x);", font, ch, pb_ptr, pb_pitch, bg, fg);
//_system_log(msg); //_system_log(msg);
//free_block(msg); //free_block(msg);

View file

@ -48,7 +48,7 @@ void term_clear() {
try_send_command(&cmd); try_send_command(&cmd);
} }
void term_set_color(uint8_t fg, uint8_t bg) { void term_set_color(_pixel_t fg, _pixel_t bg) {
struct terminal_command cmd = { struct terminal_command cmd = {
.kind = SET_COLOR, .kind = SET_COLOR,
.as_color = { .as_color = {

View file

@ -15,7 +15,8 @@ redo:
"kernel memory free: %uk\n" "kernel memory free: %uk\n"
"userspace memory free: %uk / %uk\n" "userspace memory free: %uk / %uk\n"
"Escape to quit, F5 to refresh.", "Escape to quit, F5 to refresh.",
0x10, 0x07, (_pixel_t){.r = 0, .g = 0, .b = 0},
(_pixel_t){.r = 0xbf, .g = 0xbf, .b = 0xbf},
_kernel_dynamic_area_left() * 4, _kernel_dynamic_area_left() * 4,
_total_userspace_left() * 4, _total_userspace_left() * 4,
_total_userspace_size() * 4 _total_userspace_size() * 4

View file

@ -31,6 +31,6 @@ void main(const char *text) {
} }
struct popup p; struct popup p;
info_popup(&p, text, 0x10, 0x07); info_popup(&p, text, (_pixel_t){.r = 0, .g = 0, .b = 0}, (_pixel_t){.r = 0xbf, .g = 0xbf, .b = 0xbf});
make_modal(&p); make_modal(&p);
} }

View file

@ -18,7 +18,7 @@ static const struct key_packet info_quits[] = {
{ .key_id = 0 } { .key_id = 0 }
}; };
void info_popup(struct popup *into, const char *msg, uint8_t fg, uint8_t bg) { void info_popup(struct popup *into, const char *msg, _pixel_t fg, _pixel_t bg) {
if (!info_font) if (!info_font)
info_font = get_font(FONT); info_font = get_font(FONT);
@ -41,7 +41,7 @@ void info_popup(struct popup *into, const char *msg, uint8_t fg, uint8_t bg) {
const uint32_t pitch = info_font->space_width * w + 2 * PADDING; const uint32_t pitch = info_font->space_width * w + 2 * PADDING;
const uint32_t height = info_font->space_height * h + 2 * PADDING; const uint32_t height = info_font->space_height * h + 2 * PADDING;
uint8_t *const pixbuf = get_block(pitch * height); _pixel_t *const pixbuf = get_block(pitch * height * 4);
for (uint32_t y = 0; y < height; ++y) for (uint32_t y = 0; y < height; ++y)
for (uint32_t x = 0; x < pitch; ++x) for (uint32_t x = 0; x < pitch; ++x)
@ -63,13 +63,13 @@ void info_popup(struct popup *into, const char *msg, uint8_t fg, uint8_t bg) {
into->handle = _new_window(pitch, height, pixbuf); into->handle = _new_window(pitch, height, pixbuf);
} }
void info_popupf_v(struct popup *into, const char *text, uint8_t fg, uint8_t bg, va_list args) { void info_popupf_v(struct popup *into, const char *text, _pixel_t fg, _pixel_t bg, va_list args) {
char *const msg = format_v(text, args); char *const msg = format_v(text, args);
info_popup(into, msg, fg, bg); info_popup(into, msg, fg, bg);
free_block(msg); free_block(msg);
} }
void info_popupf(struct popup *into, const char *text, uint8_t fg, uint8_t bg, ...) { void info_popupf(struct popup *into, const char *text, _pixel_t fg, _pixel_t bg, ...) {
va_list args; va_list args;
va_start(args, bg); va_start(args, bg);
info_popupf_v(into, text, fg, bg, args); info_popupf_v(into, text, fg, bg, args);

View file

@ -11,22 +11,23 @@
#include <pland/pcrt.h> #include <pland/pcrt.h>
#define FONT_HARDCODE "fixed-10" #define FONT_HARDCODE "fixed-10"
#define PADDING 2
_window_handle_t window; _window_handle_t window;
uint8_t *pixbuf; _pixel_t *pixbuf;
char *termbuf; char *termbuf;
struct font_info *font; struct font_info *font;
uint32_t width; uint32_t width;
uint32_t height; uint32_t height;
uint32_t cols = 50; uint32_t cols = 100;
uint32_t rows = 15; uint32_t rows = 30;
uint32_t cursor_y = 0; uint32_t cursor_y = 0;
uint32_t cursor_x = 0; uint32_t cursor_x = 0;
uint8_t bg_color = 0x10; _pixel_t bg_color = (_pixel_t){.r = 0, .g = 0, .b = 0};
uint8_t fg_color = 0x07; _pixel_t fg_color = (_pixel_t){.r = 0xff, .g = 0xff, .b = 0xff};
struct waiting_for_key_record { struct waiting_for_key_record {
_task_handle_t task; _task_handle_t task;
@ -35,7 +36,7 @@ struct waiting_for_key_record {
static void draw_char(uint32_t y, uint32_t x, bool inverted) { static void draw_char(uint32_t y, uint32_t x, bool inverted) {
//syslogf("drawing 0x%2h%s at %u, %u", termbuf[y * cols + x], inverted ? " inverted" : "", y, x); //syslogf("drawing 0x%2h%s at %u, %u", termbuf[y * cols + x], inverted ? " inverted" : "", y, x);
put_char(font, termbuf[y * cols + x], pixbuf + y * font->space_height * width + x * font->space_width, width, inverted ? fg_color : bg_color, inverted ? bg_color : fg_color); put_char(font, termbuf[y * cols + x], pixbuf + (y * font->space_height + PADDING) * width + (x * font->space_width + PADDING), width, inverted ? fg_color : bg_color, inverted ? bg_color : fg_color);
} }
static void clear() { static void clear() {
@ -52,9 +53,9 @@ static void scroll_fw() {
for (; i < cols * rows; ++i) for (; i < cols * rows; ++i)
termbuf[i] = ' '; termbuf[i] = ' ';
const uint32_t row_height = font->space_height; const uint32_t row_height = font->space_height;
for (i = 0; i < width * (height - row_height); ++i) for (i = width * PADDING; i < width * (height - PADDING - row_height); ++i)
pixbuf[i] = pixbuf[i + width * row_height]; pixbuf[i] = pixbuf[i + width * row_height];
for (; i < width * height; ++i) for (; i < width * (height - PADDING); ++i)
pixbuf[i] = bg_color; pixbuf[i] = bg_color;
} }
@ -187,9 +188,9 @@ static void set_dimensions(uint32_t new_rows, uint32_t new_cols) {
cols = new_cols; cols = new_cols;
termbuf = get_block(rows * cols); termbuf = get_block(rows * cols);
width = cols * font->space_width; width = cols * font->space_width + PADDING * 2;
height = cols * font->space_height; height = cols * font->space_height + PADDING * 2;
pixbuf = get_block(width * height); pixbuf = get_block(width * height * 4);
cursor_y = 0; cursor_y = 0;
cursor_x = 0; cursor_x = 0;
@ -199,27 +200,34 @@ static void set_dimensions(uint32_t new_rows, uint32_t new_cols) {
} }
void draw_all() { void draw_all() {
for (uint32_t y = 0; y < PADDING; ++y)
for (uint32_t x = 0; x < width; ++x)
pixbuf[y * width + x] = bg_color;
for (uint32_t y = height - PADDING; y < height; ++y)
for (uint32_t x = 0; x < width; ++x)
pixbuf[y * width + x] = bg_color;
for (uint32_t x = 0; x < PADDING; ++x)
for (uint32_t y = PADDING; y < height - PADDING; ++y)
pixbuf[y * width + x] = bg_color;
for (uint32_t x = width - PADDING; x < width; ++x)
for (uint32_t y = PADDING; y < height - PADDING; ++y)
pixbuf[y * width + x] = bg_color;
for (uint32_t y = 0; y < rows; ++y) for (uint32_t y = 0; y < rows; ++y)
for (uint32_t x = 0; x < cols; ++x) for (uint32_t x = 0; x < cols; ++x)
draw_char(y, x, false); draw_char(y, x, false);
draw_char(cursor_y, cursor_x, true); draw_char(cursor_y, cursor_x, true);
} }
//#include <knob/format.h>
void main(const char *cmd) { void main(const char *cmd) {
//syslogf(" this task: 0x%2h", this_task);
//syslogf(" stdio task: 0x%2h", stdio_task);
//syslogf("calling task: 0x%2h", calling_task);
font = get_font(FONT_HARDCODE); font = get_font(FONT_HARDCODE);
if (!font) if (!font)
return; return;
termbuf = get_block(cols * rows); termbuf = get_block(cols * rows);
width = cols * font->space_width; width = cols * font->space_width + PADDING * 2;
height = rows * font->space_height; height = rows * font->space_height + PADDING * 2;
pixbuf = get_block(width * height); pixbuf = get_block(width * height * 4);
clear(); clear();
add_sz_ww("Portland Terminal\n"); add_sz_ww("Portland Terminal\n");
window = _new_window(width, height, pixbuf); window = _new_window(width, height, pixbuf);