Ich schreibe ein „falsches“ Framebuffer-Gerät, damit ich meinen E-Reader als externen Bildschirm für meinen Raspberry Pi verwenden kann. Der E-Reader-Teil funktioniert einwandfrei (https://github.com/llandsmeer/inkvt), aber jetzt möchte ich es als externen Bildschirm für Linux anzeigen lassen.
Ich dachte, ein virtueller Linux-Framebuffer wäre „am einfachsten“, aber vielleicht gibt es andere Möglichkeiten, Linux einen gefälschten Bildschirm hinzuzufügen (der in xrandr angezeigt wird)?
Wie dem auch sei, hier ist das aktuelle Kernelmodul. Es erstellt ein /dev/fbX
Gerät, das nach der Abfrage fbset
richtig konfiguriert zu sein scheint. Wenn ich jedoch versuche, es zu verwenden, schlägt es fehl und ich erhalte fbcat: Corrupted page table at address 7fd7d5c90000
von dmesg die Meldung.
Ich glaube, es liegt daran, dass ich den Speicherbereich ( char mem[]
) falsch konfiguriert habe. Ich habe versucht, kalloc
instread zu verwenden, aber das hat auch nicht funktioniert. Wie soll ich vorgehen?
#include <linux/init.h>
#include <linux/module.h>
#include <linux/kernel.h>
#include <linux/fb.h>
#define WIDTH 256
#define HEIGHT 256
static char mem[WIDTH*HEIGHT*3];
static void op_fillrect(struct fb_info * info, const struct fb_fillrect * rect) {
printk(KERN_ALERT "fillrect\n");
}
static void op_copyarea(struct fb_info * info, const struct fb_copyarea * area) {
printk(KERN_ALERT "copyarea\n");
}
static void op_imageblit(struct fb_info * info, const struct fb_image * image) {
printk(KERN_ALERT "imageblit\n");
}
static struct fb_info * info;
static struct fb_ops ops = {
.owner = THIS_MODULE,
.fb_fillrect = op_fillrect,
.fb_copyarea = op_copyarea,
.fb_imageblit = op_imageblit,
};
static const struct fb_fix_screeninfo fix = {
.id = "LennartFB",
.smem_start = (unsigned long)mem,
.smem_len = sizeof(mem),
.type = FB_TYPE_PACKED_PIXELS,
.visual = FB_VISUAL_TRUECOLOR,
.line_length = 256,
};
static const struct fb_var_screeninfo var = {
.xres = WIDTH,
.yres = HEIGHT,
.xres_virtual = WIDTH,
.yres_virtual = HEIGHT,
.bits_per_pixel = 24,
.height = 256,
.width = 256,
.vmode = FB_VMODE_NONINTERLACED,
.red = { .length = 8, .offset = 16, },
.green = { .length = 8, .offset = 8, },
.blue = { .length = 8, .offset = 0, }
};
static int mod_init(void) {
int err;
info = framebuffer_alloc(0, 0);
if (!info) return -1;
info->fbops = &ops;
info->fix = fix;
info->var = var;
info->screen_base = mem;
err = fb_alloc_cmap(&info->cmap, 2, 0);
if (err < 0) return -2;
err = register_framebuffer(info);
if (err < 0) return -3;
return 0;
}
static void mod_exit(void) {
return;
}
module_init(mod_init);
module_exit(mod_exit);
MODULE_LICENSE("GPL v3");
dmesg:
[ 4984.929380] fbcat: Corrupted page table at address 7fd7d5c90000
[ 4984.929385] PGD 8000000277709067 P4D 8000000277709067 PUD 3eb108067 PMD 2c732e067 PTE 800fffffc0f5d235
[ 4984.929391] Bad pagetable: 000d [#1] SMP PTI
[ 4984.929396] CPU: 11 PID: 8076 Comm: fbcat Tainted: P OE 5.3.0-46-generic #38~18.04.1-Ubuntu
[ 4984.929398] Hardware name: Acer Aspire A715-72G/Charmeleon_CFS, BIOS V1.19 07/13/2018
[ 4984.929403] RIP: 0033:0x55fa6cb22e98