1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
|
// MIT License, Copyright (c) 2021 Marvin Borner
#include <assert.h>
#include <cpu.h>
#include <def.h>
#include <errno.h>
#include <fb.h>
#include <io.h>
#include <mem.h>
#include <mm.h>
#include <multiboot.h>
#include <str.h>
#include <sys.h>
struct vbe_basic {
u8 stuff1[16];
u16 pitch;
u16 width;
u16 height;
u8 stuff2[18];
u8 *fb;
u8 stuff3[212];
};
PROTECTED static struct vbe_basic *vbe = NULL;
static u32 fb_map_buffer(struct page_dir *dir)
{
assert(vbe);
u32 size = vbe->height * vbe->pitch;
return virtual_alloc(dir, memory_range_around((u32)vbe->fb, size), MEMORY_USER).base;
}
static u32 fb_owner = 0;
static res fb_ioctl(u32 request, void *arg1, void *arg2, void *arg3)
{
UNUSED(arg2);
UNUSED(arg3);
switch (request) {
case IOCTL_FB_GET: {
if (!vbe)
return -ENOENT;
u32 size = MIN(sizeof(*vbe), (u32)arg2);
if (!memory_writable_range(memory_range(arg1, size)))
return -EFAULT;
if (fb_owner != 0 && proc_from_pid(fb_owner))
return -EBUSY;
else
fb_owner = proc_current()->pid;
u32 fb = fb_map_buffer(proc_current()->page_dir);
vbe->fb = (u8 *)fb;
memcpy_user(arg1, vbe, size);
return EOK;
}
default:
return -EINVAL;
}
}
CLEAR void fb_install(void)
{
vbe = (void *)multiboot_vbe();
struct io_dev *dev = zalloc(sizeof(*dev));
dev->control = fb_ioctl;
io_add(IO_FRAMEBUFFER, dev);
// Identity map framebuffer to kernel to prevent unwanted writing
u32 size = vbe->height * vbe->pitch;
memory_map_identity(virtual_kernel_dir(), memory_range_around((u32)vbe->fb, size),
MEMORY_CLEAR);
}
|