如何直接写入linux帧缓冲区?

发布于 2024-08-07 02:00:45 字数 23 浏览 10 评论 0原文

如何直接写入linux帧缓冲区?

How to write directly to linux framebuffer?

如果你对这篇内容有疑问,欢迎到本站社区发帖提问 参与讨论,获取更多帮助,或者扫码二维码加入 Web 技术交流群。

扫码二维码加入Web技术交流群

发布评论

需要 登录 才能够评论, 你可以免费 注册 一个本站的账号。

评论(4

最佳男配角 2024-08-14 02:00:46

看看 FBIOPUT_VSCREENINFO、ioctl 和 mmap

(我有代码,但在这台电脑上没有,抱歉)

编辑:这应该让你开始

  //open file descriptor and get info
inf fdScreen = open( "devicename", O_RDWR );
fb_var_screeninfo varInfo;
ioctl( fdScreen, FBIOGET_VSCREENINFO, &varInfo );

  //set resolution/dpi/color depth/.. in varInfo, then write it back
ioctl( fdScreen, FBIOPUT_VSCREENINFO, &varInfo );

  //get writable screen memory; unsigned short here for 16bit color
unsigned short* display = mmap( 0, nScreenSize,
                                PROT_READ | PROT_WRITE, MAP_SHARED, fdScreen, 0 );

look at FBIOPUT_VSCREENINFO, ioctl and mmap

(I have the code but not at this pc, sorry)

edit: this should get you started

  //open file descriptor and get info
inf fdScreen = open( "devicename", O_RDWR );
fb_var_screeninfo varInfo;
ioctl( fdScreen, FBIOGET_VSCREENINFO, &varInfo );

  //set resolution/dpi/color depth/.. in varInfo, then write it back
ioctl( fdScreen, FBIOPUT_VSCREENINFO, &varInfo );

  //get writable screen memory; unsigned short here for 16bit color
unsigned short* display = mmap( 0, nScreenSize,
                                PROT_READ | PROT_WRITE, MAP_SHARED, fdScreen, 0 );
臻嫒无言 2024-08-14 02:00:46

基本上你打开/dev/fb0,对其进行一些ioctls,然后mmap它。然后,您只需写入进程中的 mmap 区域即可。

Basically you open /dev/fb0, do some ioctls on it, then mmap it. Then you just write to the mmap'd area in your process.

随风而去 2024-08-14 02:00:46

您想编写设备驱动程序吗?如果是这样,请查看此操作指南

Are you looking to write a device driver? If so check out this HowTo guide

旧夏天 2024-08-14 02:00:46

在另一个 stackoverflow 问题上找到了这段代码。它使用帧缓冲区绘制一个粉红色阴影矩形到屏幕上,它立即对我有用。您必须从正确的文本终端运行它,而不是在图形终端内运行。

#include <stdlib.h>
 #include <unistd.h>
 #include <stdio.h>
 #include <fcntl.h>
 #include <linux/fb.h>
 #include <sys/mman.h>
 #include <sys/ioctl.h>

 int main()
 {
     int fbfd = 0;
     struct fb_var_screeninfo vinfo;
     struct fb_fix_screeninfo finfo;
     long int screensize = 0;
     char *fbp = 0;
     int x = 0, y = 0;
     long int location = 0;

 // Open the file for reading and writing
 fbfd = open("/dev/fb0", O_RDWR);
 if (fbfd == -1) {
     perror("Error: cannot open framebuffer device");
     exit(1);
 }
 printf("The framebuffer device was opened successfully.\n");

 // Get fixed screen information
 if (ioctl(fbfd, FBIOGET_FSCREENINFO, &finfo) == -1) {
     perror("Error reading fixed information");
     exit(2);
 }

 // Get variable screen information
 if (ioctl(fbfd, FBIOGET_VSCREENINFO, &vinfo) == -1) {
     perror("Error reading variable information");
     exit(3);
 }

 printf("%dx%d, %dbpp\n", vinfo.xres, vinfo.yres, vinfo.bits_per_pixel);

 // Figure out the size of the screen in bytes
 screensize = vinfo.xres * vinfo.yres * vinfo.bits_per_pixel / 8;

 // Map the device to memory
 fbp = (char *)mmap(0, screensize, PROT_READ | PROT_WRITE, MAP_SHARED,
                    fbfd, 0);
 if ((int)fbp == -1) {
     perror("Error: failed to map framebuffer device to memory");
     exit(4);
 }
 printf("The framebuffer device was mapped to memory successfully.\n");

 x = 300; y = 100;       // Where we are going to put the pixel

 // Figure out where in memory to put the pixel
 for (y = 100; y < 300; y++)
     for (x = 100; x < 300; x++) {

         location = (x+vinfo.xoffset) * (vinfo.bits_per_pixel/8) +
                    (y+vinfo.yoffset) * finfo.line_length;

         if (vinfo.bits_per_pixel == 32) {
             *(fbp + location) = 100;        // Some blue
             *(fbp + location + 1) = 15+(x-100)/2;     // A little green
             *(fbp + location + 2) = 200-(y-100)/5;    // A lot of red
             *(fbp + location + 3) = 0;      // No transparency
         } else  { //assume 16bpp
             int b = 10;
             int g = (x-100)/6;     // A little green
             int r = 31-(y-100)/16;    // A lot of red
             unsigned short int t = r<<11 | g << 5 | b;
             *((unsigned short int*)(fbp + location)) = t;
         }

     }
 munmap(fbp, screensize);
 close(fbfd);
 return 0;
 }

I found this code on this other stackoverflow question. It draws a shaded pink rectangle to screen using the framebuffer and it worked straight away for me. You have to run it from a proper text terminal, not inside a graphical terminal.

#include <stdlib.h>
 #include <unistd.h>
 #include <stdio.h>
 #include <fcntl.h>
 #include <linux/fb.h>
 #include <sys/mman.h>
 #include <sys/ioctl.h>

 int main()
 {
     int fbfd = 0;
     struct fb_var_screeninfo vinfo;
     struct fb_fix_screeninfo finfo;
     long int screensize = 0;
     char *fbp = 0;
     int x = 0, y = 0;
     long int location = 0;

 // Open the file for reading and writing
 fbfd = open("/dev/fb0", O_RDWR);
 if (fbfd == -1) {
     perror("Error: cannot open framebuffer device");
     exit(1);
 }
 printf("The framebuffer device was opened successfully.\n");

 // Get fixed screen information
 if (ioctl(fbfd, FBIOGET_FSCREENINFO, &finfo) == -1) {
     perror("Error reading fixed information");
     exit(2);
 }

 // Get variable screen information
 if (ioctl(fbfd, FBIOGET_VSCREENINFO, &vinfo) == -1) {
     perror("Error reading variable information");
     exit(3);
 }

 printf("%dx%d, %dbpp\n", vinfo.xres, vinfo.yres, vinfo.bits_per_pixel);

 // Figure out the size of the screen in bytes
 screensize = vinfo.xres * vinfo.yres * vinfo.bits_per_pixel / 8;

 // Map the device to memory
 fbp = (char *)mmap(0, screensize, PROT_READ | PROT_WRITE, MAP_SHARED,
                    fbfd, 0);
 if ((int)fbp == -1) {
     perror("Error: failed to map framebuffer device to memory");
     exit(4);
 }
 printf("The framebuffer device was mapped to memory successfully.\n");

 x = 300; y = 100;       // Where we are going to put the pixel

 // Figure out where in memory to put the pixel
 for (y = 100; y < 300; y++)
     for (x = 100; x < 300; x++) {

         location = (x+vinfo.xoffset) * (vinfo.bits_per_pixel/8) +
                    (y+vinfo.yoffset) * finfo.line_length;

         if (vinfo.bits_per_pixel == 32) {
             *(fbp + location) = 100;        // Some blue
             *(fbp + location + 1) = 15+(x-100)/2;     // A little green
             *(fbp + location + 2) = 200-(y-100)/5;    // A lot of red
             *(fbp + location + 3) = 0;      // No transparency
         } else  { //assume 16bpp
             int b = 10;
             int g = (x-100)/6;     // A little green
             int r = 31-(y-100)/16;    // A lot of red
             unsigned short int t = r<<11 | g << 5 | b;
             *((unsigned short int*)(fbp + location)) = t;
         }

     }
 munmap(fbp, screensize);
 close(fbfd);
 return 0;
 }
~没有更多了~
我们使用 Cookies 和其他技术来定制您的体验包括您的登录状态等。通过阅读我们的 隐私政策 了解更多相关信息。 单击 接受 或继续使用网站,即表示您同意使用 Cookies 和您的相关数据。
原文