dev: start implementing framebuffer devices
This commit is contained in:
25
dev/core.c
25
dev/core.c
@@ -19,6 +19,19 @@ static kern_status_t device_object_get_child_named(struct object *, const char *
|
|||||||
|
|
||||||
extern kern_status_t init_driver_tree(void);
|
extern kern_status_t init_driver_tree(void);
|
||||||
|
|
||||||
|
extern struct device_type_ops input_type_ops;
|
||||||
|
extern struct device_type_ops framebuffer_type_ops;
|
||||||
|
|
||||||
|
static struct device_type_ops *type_ops[] = {
|
||||||
|
[DEV_TYPE_UNKNOWN] = NULL,
|
||||||
|
[DEV_TYPE_BLOCK] = NULL,
|
||||||
|
[DEV_TYPE_CHAR] = NULL,
|
||||||
|
[DEV_TYPE_NET] = NULL,
|
||||||
|
[DEV_TYPE_INPUT] = &input_type_ops,
|
||||||
|
[DEV_TYPE_BUS] = NULL,
|
||||||
|
[DEV_TYPE_FRAMEBUFFER] = &framebuffer_type_ops,
|
||||||
|
};
|
||||||
|
|
||||||
static struct object_type device_type = {
|
static struct object_type device_type = {
|
||||||
.ob_name = "device",
|
.ob_name = "device",
|
||||||
.ob_size = sizeof(struct device),
|
.ob_size = sizeof(struct device),
|
||||||
@@ -60,7 +73,7 @@ kern_status_t device_init(void)
|
|||||||
struct bus_device *system_dev = bus_device_create();
|
struct bus_device *system_dev = bus_device_create();
|
||||||
struct device *system_dev_base = bus_device_base(system_dev);
|
struct device *system_dev_base = bus_device_base(system_dev);
|
||||||
snprintf(system_dev_base->dev_name, sizeof system_dev_base->dev_name, "system");
|
snprintf(system_dev_base->dev_name, sizeof system_dev_base->dev_name, "system");
|
||||||
__root_device = bus_device_base(system_dev);
|
set_root_device(bus_device_base(system_dev));
|
||||||
|
|
||||||
struct bus_device *misc_dev = bus_device_create();
|
struct bus_device *misc_dev = bus_device_create();
|
||||||
struct device *misc_dev_base = bus_device_base(misc_dev);
|
struct device *misc_dev_base = bus_device_base(misc_dev);
|
||||||
@@ -236,14 +249,12 @@ kern_status_t device_register(struct device *dev, struct driver *owner, struct d
|
|||||||
|
|
||||||
driver_add_device(owner, dev);
|
driver_add_device(owner, dev);
|
||||||
|
|
||||||
switch (dev->dev_type) {
|
if (type_ops[dev->dev_type] && type_ops[dev->dev_type]->register_device) {
|
||||||
case DEV_TYPE_INPUT:
|
status = type_ops[dev->dev_type]->register_device(dev);
|
||||||
status = input_device_register(INPUT_DEVICE(dev));
|
|
||||||
break;
|
|
||||||
default:
|
|
||||||
break;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/* TODO remove device if registration failed */
|
||||||
|
|
||||||
driver_unlock(owner);
|
driver_unlock(owner);
|
||||||
device_unlock_irqrestore(dev, flags);
|
device_unlock_irqrestore(dev, flags);
|
||||||
|
|
||||||
|
|||||||
47
dev/fb.c
Normal file
47
dev/fb.c
Normal file
@@ -0,0 +1,47 @@
|
|||||||
|
#include <socks/device.h>
|
||||||
|
#include <socks/bitmap.h>
|
||||||
|
#include <socks/libc/stdio.h>
|
||||||
|
|
||||||
|
static DECLARE_BITMAP(fb_device_ids, FRAMEBUFFER_DEVICE_MAX);
|
||||||
|
static spin_lock_t fb_device_ids_lock = SPIN_LOCK_INIT;
|
||||||
|
|
||||||
|
struct framebuffer_device *framebuffer_device_create(void)
|
||||||
|
{
|
||||||
|
struct device *dev = device_alloc();
|
||||||
|
if (!dev) {
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
dev->dev_type = DEV_TYPE_FRAMEBUFFER;
|
||||||
|
|
||||||
|
return FRAMEBUFFER_DEVICE(dev);
|
||||||
|
}
|
||||||
|
|
||||||
|
static kern_status_t generate_name(struct framebuffer_device *dev, char out[DEV_NAME_MAX])
|
||||||
|
{
|
||||||
|
snprintf(out, DEV_NAME_MAX, "fb%u", dev->fb_id);
|
||||||
|
return KERN_OK;
|
||||||
|
}
|
||||||
|
|
||||||
|
kern_status_t framebuffer_device_register(struct device *dev)
|
||||||
|
{
|
||||||
|
unsigned long flags;
|
||||||
|
spin_lock_irqsave(&fb_device_ids_lock, &flags);
|
||||||
|
unsigned int id = bitmap_lowest_clear(fb_device_ids, INPUT_DEVICE_MAX);
|
||||||
|
bitmap_set(fb_device_ids, id);
|
||||||
|
spin_unlock_irqrestore(&fb_device_ids_lock, flags);
|
||||||
|
|
||||||
|
struct framebuffer_device *fbdev = &dev->fb;
|
||||||
|
fbdev->fb_id = id;
|
||||||
|
|
||||||
|
char name[DEV_NAME_MAX];
|
||||||
|
generate_name(fbdev, name);
|
||||||
|
char path[OBJECT_PATH_MAX];
|
||||||
|
snprintf(path, sizeof path, "/dev/video/%s", name);
|
||||||
|
|
||||||
|
return object_namespace_create_link(global_namespace(), path, &dev->dev_base);
|
||||||
|
}
|
||||||
|
|
||||||
|
struct device_type_ops framebuffer_type_ops = {
|
||||||
|
.register_device = framebuffer_device_register,
|
||||||
|
};
|
||||||
14
dev/input.c
14
dev/input.c
@@ -61,7 +61,7 @@ static kern_status_t generate_name(struct input_device *dev, char out[DEV_NAME_M
|
|||||||
return KERN_OK;
|
return KERN_OK;
|
||||||
}
|
}
|
||||||
|
|
||||||
kern_status_t input_device_register(struct input_device *dev)
|
kern_status_t input_device_register(struct device *dev)
|
||||||
{
|
{
|
||||||
unsigned long flags;
|
unsigned long flags;
|
||||||
spin_lock_irqsave(&input_device_ids_lock, &flags);
|
spin_lock_irqsave(&input_device_ids_lock, &flags);
|
||||||
@@ -69,13 +69,17 @@ kern_status_t input_device_register(struct input_device *dev)
|
|||||||
bitmap_set(input_device_ids, id);
|
bitmap_set(input_device_ids, id);
|
||||||
spin_unlock_irqrestore(&input_device_ids_lock, flags);
|
spin_unlock_irqrestore(&input_device_ids_lock, flags);
|
||||||
|
|
||||||
dev->i_id = id;
|
struct input_device *inputdev = &dev->input;
|
||||||
|
inputdev->i_id = id;
|
||||||
|
|
||||||
char name[DEV_NAME_MAX];
|
char name[DEV_NAME_MAX];
|
||||||
generate_name(dev, name);
|
generate_name(inputdev, name);
|
||||||
char path[OBJECT_PATH_MAX];
|
char path[OBJECT_PATH_MAX];
|
||||||
snprintf(path, sizeof path, "/dev/input/%s", name);
|
snprintf(path, sizeof path, "/dev/input/%s", name);
|
||||||
|
|
||||||
struct device *base = input_device_base(dev);
|
return object_namespace_create_link(global_namespace(), path, &dev->dev_base);
|
||||||
return object_namespace_create_link(global_namespace(), path, &base->dev_base);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
struct device_type_ops input_type_ops = {
|
||||||
|
.register_device = input_device_register,
|
||||||
|
};
|
||||||
|
|||||||
@@ -6,6 +6,7 @@
|
|||||||
#include <socks/status.h>
|
#include <socks/status.h>
|
||||||
#include <socks/bitmap.h>
|
#include <socks/bitmap.h>
|
||||||
#include <socks/object.h>
|
#include <socks/object.h>
|
||||||
|
#include <socks/fb.h>
|
||||||
#include <socks/ringbuffer.h>
|
#include <socks/ringbuffer.h>
|
||||||
|
|
||||||
struct device;
|
struct device;
|
||||||
@@ -19,12 +20,14 @@ struct input_event;
|
|||||||
|
|
||||||
#define INPUT_DEVICE_EVENT_QUEUE_SIZE 128
|
#define INPUT_DEVICE_EVENT_QUEUE_SIZE 128
|
||||||
#define INPUT_DEVICE_MAX 4096
|
#define INPUT_DEVICE_MAX 4096
|
||||||
|
#define FRAMEBUFFER_DEVICE_MAX 4096
|
||||||
|
|
||||||
#define BLOCK_DEVICE(dev) ((dev)->dev_type == DEV_TYPE_BLOCK ? &(dev)->blk : NULL)
|
#define BLOCK_DEVICE(dev) ((dev)->dev_type == DEV_TYPE_BLOCK ? &(dev)->blk : NULL)
|
||||||
#define CHAR_DEVICE(dev) ((dev)->dev_type == DEV_TYPE_CHAR ? &(dev)->chr : NULL)
|
#define CHAR_DEVICE(dev) ((dev)->dev_type == DEV_TYPE_CHAR ? &(dev)->chr : NULL)
|
||||||
#define NET_DEVICE(dev) ((dev)->dev_type == DEV_TYPE_NET ? &(dev)->net : NULL)
|
#define NET_DEVICE(dev) ((dev)->dev_type == DEV_TYPE_NET ? &(dev)->net : NULL)
|
||||||
#define INPUT_DEVICE(dev) ((dev)->dev_type == DEV_TYPE_INPUT ? &(dev)->input : NULL)
|
#define INPUT_DEVICE(dev) ((dev)->dev_type == DEV_TYPE_INPUT ? &(dev)->input : NULL)
|
||||||
#define BUS_DEVICE(dev) ((dev)->dev_type == DEV_TYPE_BUS ? &(dev)->bus : NULL)
|
#define BUS_DEVICE(dev) ((dev)->dev_type == DEV_TYPE_BUS ? &(dev)->bus : NULL)
|
||||||
|
#define FRAMEBUFFER_DEVICE(dev) ((dev)->dev_type == DEV_TYPE_FRAMEBUFFER ? &(dev)->fb : NULL)
|
||||||
|
|
||||||
enum device_type {
|
enum device_type {
|
||||||
DEV_TYPE_UNKNOWN = 0,
|
DEV_TYPE_UNKNOWN = 0,
|
||||||
@@ -33,6 +36,11 @@ enum device_type {
|
|||||||
DEV_TYPE_NET,
|
DEV_TYPE_NET,
|
||||||
DEV_TYPE_INPUT,
|
DEV_TYPE_INPUT,
|
||||||
DEV_TYPE_BUS,
|
DEV_TYPE_BUS,
|
||||||
|
DEV_TYPE_FRAMEBUFFER,
|
||||||
|
};
|
||||||
|
|
||||||
|
struct device_type_ops {
|
||||||
|
kern_status_t(*register_device)(struct device *);
|
||||||
};
|
};
|
||||||
|
|
||||||
struct block_device_ops {
|
struct block_device_ops {
|
||||||
@@ -61,6 +69,10 @@ struct bus_device_ops {
|
|||||||
kern_status_t(*scan)(struct device *);
|
kern_status_t(*scan)(struct device *);
|
||||||
};
|
};
|
||||||
|
|
||||||
|
struct framebuffer_device_ops {
|
||||||
|
kern_status_t(*scan)(struct device *);
|
||||||
|
};
|
||||||
|
|
||||||
struct block_device {
|
struct block_device {
|
||||||
struct block_device_ops *b_ops;
|
struct block_device_ops *b_ops;
|
||||||
};
|
};
|
||||||
@@ -83,6 +95,13 @@ struct bus_device {
|
|||||||
struct bus_device_ops *b_ops;
|
struct bus_device_ops *b_ops;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
struct framebuffer_device {
|
||||||
|
unsigned int fb_id;
|
||||||
|
struct framebuffer_device_ops *fb_ops;
|
||||||
|
struct framebuffer_varinfo fb_varinfo;
|
||||||
|
struct framebuffer_fixedinfo fb_fixedinfo;
|
||||||
|
};
|
||||||
|
|
||||||
struct device {
|
struct device {
|
||||||
struct object dev_base;
|
struct object dev_base;
|
||||||
unsigned int dev_minor;
|
unsigned int dev_minor;
|
||||||
@@ -103,6 +122,7 @@ struct device {
|
|||||||
struct net_device net;
|
struct net_device net;
|
||||||
struct input_device input;
|
struct input_device input;
|
||||||
struct bus_device bus;
|
struct bus_device bus;
|
||||||
|
struct framebuffer_device fb;
|
||||||
};
|
};
|
||||||
};
|
};
|
||||||
|
|
||||||
@@ -160,6 +180,7 @@ extern struct block_device *block_device_create(void);
|
|||||||
extern struct net_device *net_device_create(void);
|
extern struct net_device *net_device_create(void);
|
||||||
extern struct input_device *input_device_create(void);
|
extern struct input_device *input_device_create(void);
|
||||||
extern struct bus_device *bus_device_create(void);
|
extern struct bus_device *bus_device_create(void);
|
||||||
|
extern struct framebuffer_device *framebuffer_device_create(void);
|
||||||
|
|
||||||
static inline struct device *char_device_base(struct char_device *dev)
|
static inline struct device *char_device_base(struct char_device *dev)
|
||||||
{
|
{
|
||||||
@@ -186,11 +207,15 @@ static inline struct device *bus_device_base(struct bus_device *dev)
|
|||||||
return (struct device *)((char *)dev - offsetof(struct device, bus));
|
return (struct device *)((char *)dev - offsetof(struct device, bus));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static inline struct device *framebuffer_device_base(struct framebuffer_device *dev)
|
||||||
|
{
|
||||||
|
return (struct device *)((char *)dev - offsetof(struct device, fb));
|
||||||
|
}
|
||||||
|
|
||||||
extern kern_status_t device_register(struct device *dev, struct driver *owner, struct device *parent);
|
extern kern_status_t device_register(struct device *dev, struct driver *owner, struct device *parent);
|
||||||
|
|
||||||
extern kern_status_t input_device_report_event(struct input_device *dev, const struct input_event *ev, bool noblock);
|
extern kern_status_t input_device_report_event(struct input_device *dev, const struct input_event *ev, bool noblock);
|
||||||
extern kern_status_t input_device_read(struct device *dev, void *buf, size_t size, size_t *bytes_read, socks_flags_t flags);
|
extern kern_status_t input_device_read(struct device *dev, void *buf, size_t size, size_t *bytes_read, socks_flags_t flags);
|
||||||
extern kern_status_t input_device_register(struct input_device *dev);
|
|
||||||
|
|
||||||
extern struct driver *driver_create(struct kext *self, const char *name);
|
extern struct driver *driver_create(struct kext *self, const char *name);
|
||||||
extern kern_status_t driver_destroy(struct driver *driver);
|
extern kern_status_t driver_destroy(struct driver *driver);
|
||||||
|
|||||||
33
include/socks/fb.h
Normal file
33
include/socks/fb.h
Normal file
@@ -0,0 +1,33 @@
|
|||||||
|
#ifndef SOCKS_FB_H_
|
||||||
|
#define SOCKS_FB_H_
|
||||||
|
|
||||||
|
#include <stdint.h>
|
||||||
|
|
||||||
|
enum framebuffer_flags {
|
||||||
|
FB_VGATEXT = 0x01u,
|
||||||
|
};
|
||||||
|
|
||||||
|
struct framebuffer_bitfield {
|
||||||
|
uint32_t b_offset;
|
||||||
|
uint16_t b_length;
|
||||||
|
};
|
||||||
|
|
||||||
|
struct framebuffer_varinfo {
|
||||||
|
enum framebuffer_flags fb_flags;
|
||||||
|
|
||||||
|
uint32_t fb_xres;
|
||||||
|
uint32_t fb_yres;
|
||||||
|
uint32_t fb_bpp;
|
||||||
|
uint32_t fb_stride;
|
||||||
|
|
||||||
|
struct framebuffer_bitfield fb_red;
|
||||||
|
struct framebuffer_bitfield fb_green;
|
||||||
|
struct framebuffer_bitfield fb_blue;
|
||||||
|
struct framebuffer_bitfield fb_alpha;
|
||||||
|
};
|
||||||
|
|
||||||
|
struct framebuffer_fixedinfo {
|
||||||
|
uint64_t fb_baseptr;
|
||||||
|
};
|
||||||
|
|
||||||
|
#endif
|
||||||
8
kexts/drivers/video/vgafb/extension.yaml
Normal file
8
kexts/drivers/video/vgafb/extension.yaml
Normal file
@@ -0,0 +1,8 @@
|
|||||||
|
name: vgafb
|
||||||
|
description: |
|
||||||
|
Generic VGA driver.
|
||||||
|
id: net.doorstuck.socks.vgafb
|
||||||
|
license: BSD-3-Clause
|
||||||
|
copyright: Copyright © Max Wash 2023
|
||||||
|
sources:
|
||||||
|
- main.c
|
||||||
36
kexts/drivers/video/vgafb/main.c
Normal file
36
kexts/drivers/video/vgafb/main.c
Normal file
@@ -0,0 +1,36 @@
|
|||||||
|
#include <socks/printk.h>
|
||||||
|
#include <socks/device.h>
|
||||||
|
#include <socks/kext.h>
|
||||||
|
#include <socks/libc/stdio.h>
|
||||||
|
|
||||||
|
static struct driver *vga_driver = NULL;
|
||||||
|
static struct framebuffer_device *vgafb = NULL;
|
||||||
|
|
||||||
|
static kern_status_t online(struct kext *self)
|
||||||
|
{
|
||||||
|
vga_driver = driver_create(self, "vgafb");
|
||||||
|
if (!vga_driver) {
|
||||||
|
return KERN_NO_MEMORY;
|
||||||
|
}
|
||||||
|
|
||||||
|
driver_register(vga_driver);
|
||||||
|
|
||||||
|
struct framebuffer_device *fb = framebuffer_device_create();
|
||||||
|
struct device *fb_base = framebuffer_device_base(fb);
|
||||||
|
snprintf(fb_base->dev_name, sizeof fb_base->dev_name, "vgafb");
|
||||||
|
|
||||||
|
kern_status_t status = device_register(fb_base, vga_driver, misc_device());
|
||||||
|
if (status != KERN_OK) {
|
||||||
|
driver_unregister(vga_driver);
|
||||||
|
driver_destroy(vga_driver);
|
||||||
|
return status;
|
||||||
|
}
|
||||||
|
|
||||||
|
vgafb = fb;
|
||||||
|
|
||||||
|
return KERN_OK;
|
||||||
|
}
|
||||||
|
|
||||||
|
DEFINE_KEXT("net.doorstuck.socks.vgafb",
|
||||||
|
online, NULL,
|
||||||
|
KEXT_NO_DEPENDENCIES);
|
||||||
Reference in New Issue
Block a user