1
0
mirror of https://github.com/sgmarz/osblog.git synced 2024-11-24 02:16:19 +04:00
osblog/risc_v/userspace/fb.cpp

275 lines
5.2 KiB
C++

#include <cstdio>
#include <input-event-codes.h>
#define MAX_EVENTS 100
#define cos(x) table_cos(x)
// #define cos(x) taylor_cos(x)
#define min(x, y) ((x < y) ? x : y)
#define max(x, y) ((x > y) ? x : y)
using u8 = unsigned char;
using i8 = signed char;
using u16 = unsigned short;
using i16 = signed short;
using u32 = unsigned int;
using i32 = signed int;
using u64 = unsigned long;
using i64 = signed long;
using f64 = double;
using f32 = float;
struct Pixel {
u8 r;
u8 g;
u8 b;
u8 a;
};
struct Event {
u16 event_type;
u16 code;
u32 value;
} events[MAX_EVENTS];
void fill_rect(Pixel *fb, u32 x, u32 y, u32 width, u32 height, Pixel &color);
void stroke_rect(Pixel *fb, u32 x, u32 y, u32 width, u32 height, Pixel &color, u32 size);
void set_pixel(Pixel *fb, u32 x, u32 y, Pixel &color);
void draw_cosine(Pixel *fb, u32 x, u32 y, u32 width, u32 height, Pixel &color);
void draw_circle(Pixel *fb, u32 x, u32 y, f64 r, Pixel &color);
const u64 noevt_slptm = 10000;
const u64 evt_slptm = 10000;
struct Rect {
u32 x;
u32 y;
u32 width;
u32 height;
};
constexpr u32 lerp(u32 val, u32 mx1, u32 mx2) {
f64 r = val / static_cast<f64>(mx1);
return r * mx2;
}
int main()
{
bool pressed = false;
Pixel *fb = (Pixel *)syscall_get_fb(6);
Pixel white_color = {255, 255, 255, 255};
Pixel black_color = {0, 0, 0, 255};
Pixel current_color = {255, 150, 0, 255};
u32 x = 0;
u32 y = 0;
u32 num_events;
fill_rect(fb, 0, 0, 640, 480, white_color);
syscall_inv_rect(6, 0, 0, 640, 480);
do {
if ((num_events = syscall_get_key(events, MAX_EVENTS)) > 0) {
for (u32 z = 0;z < num_events;z++) {
Event &ev = events[z];
switch (ev.code) {
case BTN_MOUSE:
pressed = (ev.value & 1) == 1;
break;
case KEY_O:
current_color = Pixel {255, 150, 0, 255};
break;
case KEY_B:
current_color = Pixel {0, 0, 255, 255};
break;
case KEY_G:
current_color = Pixel {0, 255, 0, 255};
break;
case KEY_R:
current_color = Pixel {255, 0, 0, 255};
break;
case KEY_W:
if (ev.value == 0) { //released
fill_rect(fb, 0, 0, 640, 480, white_color);
syscall_inv_rect(6, 0, 0, 640, 480);
}
break;
case KEY_Q:
if (ev.value == 0) { // released
fill_rect(fb, 0, 0, 640, 480, black_color);
syscall_inv_rect(6, 0, 0, 640, 480);
}
break;
}
}
}
if ((num_events = syscall_get_abs(events, MAX_EVENTS)) < 1) {
syscall_sleep(noevt_slptm);
continue;
}
for (u32 z = 0;z < num_events;z++) {
Event &ev = events[z];
if (ev.code == ABS_X) {
x = lerp(ev.value & 0x7fff, 32767, 640);
}
else if (ev.code == ABS_Y) {
y = lerp(ev.value & 0x7fff, 32767, 480);
}
if (pressed) {
fill_rect(fb, x, y, 5, 5, current_color);
}
}
if (pressed) {
syscall_inv_rect(6, 0, 0, 640, 480);
}
} while (true);
return 0;
}
void set_pixel(Pixel *fb, u32 x, u32 y, Pixel &color) {
if (x < 640 && y < 480) {
fb[y * 640 + x] = color;
}
}
void fill_rect(Pixel *fb, u32 x, u32 y, u32 width, u32 height, Pixel &color) {
for (u32 row = y; row < (y+height);row++) {
for (u32 col = x; col < (x+width);col++) {
set_pixel(fb, col, row, color);
}
}
}
void stroke_rect(Pixel *fb, u32 x, u32 y, u32 width, u32 height, Pixel &color, u32 size) {
// Essentially fill the four sides.
// Top
fill_rect(fb, x, y, width, size, color);
// Bottom
fill_rect(fb, x, y + height, width, size, color);
// Left
fill_rect(fb, x, y, size, height, color);
// Right
fill_rect(fb, x + width, y, size, height + size, color);
}
f64 table_cos(f64 angle_degrees) {
const f64 COS_TABLE[] = {
1.0,
0.9962,
0.9848,
0.9659,
0.9397,
0.9063,
0.8660,
0.8191,
0.7660,
0.7071,
0.6428,
0.5736,
0.5000,
0.4226,
0.3420,
0.2558,
0.1736,
0.0872,
0.0,
-0.0872,
-0.1736,
-0.2558,
-0.3420,
-0.4226,
-0.5000,
-0.5736,
-0.6428,
-0.7071,
-0.7660,
-0.8191,
-0.8660,
-0.9063,
-0.9397,
-0.9659,
-0.9848,
-0.9962,
-1.0,
-0.9962,
-0.9848,
-0.9659,
-0.9397,
-0.9063,
-0.8660,
-0.8191,
-0.7660,
-0.7071,
-0.6428,
-0.5736,
-0.5000,
-0.4226,
-0.3420,
-0.2558,
-0.1736,
-0.0872,
0.0,
0.0872,
0.1736,
0.2558,
0.3420,
0.4226,
0.5000,
0.5736,
0.6428,
0.7071,
0.7660,
0.8191,
0.8660,
0.9063,
0.9397,
0.9659,
0.9848,
0.9962,
1.0,
};
u32 lookup_ang = angle_degrees / 5;
return COS_TABLE[lookup_ang];
}
f64 taylor_cos(f64 angle_degrees) {
f64 x = 3.14159265359 * angle_degrees / 180.0;
f64 result = 1.0;
f64 inter = 1.0;
f64 num = x * x;
for (int i = 1;i <= 6;i++) {
u64 comp = 2 * i;
u64 den = comp * (comp - 1);
inter *= num / den;
if ((i & 1) == 0) {
result += inter;
}
else {
result -= inter;
}
}
return result;
}
f64 sin(f64 angle_degrees) {
return cos(90.0 - angle_degrees);
}
void draw_cosine(Pixel *fb, u32 x, u32 y, u32 width, u32 height, Pixel &color) {
for (u32 i = 1; i <= width;i++) {
f64 fy = -cos(i % 360);
f64 yy = fy / 2.0 * height;
u32 nx = x + i;
u32 ny = yy + y;
// printf("Cos %u = %lf, x: %u, y: %u\n", (i % 360), fy, nx, ny);
fill_rect(fb, nx, ny, 2, 2, color);
}
}
void draw_circle(Pixel *fb, u32 x, u32 y, f64 r, Pixel &color)
{
}