动手写 framebuffer 画点、划线程序 (七)

文的copyright归yuweixian4230@163.com 所有,使用GPL发布,可以自由拷贝,转载。但转载请保持文档的完整性,注明原作者及原链接,严禁用于任何商业用途。
作者:yuweixian4230@163.com

博客:yuweixian4230.blog.chinaunix.net  

简单实现了 画点、划线、画矩形的程序编写,没有时间去编写其他的了,以后具体做到的这方面的事,在详细写吧。。
    
     没有写 颜色操作。

代码上传; framebuff-myself-press.rar   

  1. #include <unistd.h>
  2. #include <stdio.h>
  3. #include <fcntl.h>
  4. #include <linux/fb.h>
  5. #include <sys/mman.h>

  6. typedef struct fbdev{
  7.     int fdfd; //open "dev/fb0"
  8.     struct fb_var_screeninfo vinfo;
  9.     struct fb_fix_screeninfo finfo;
  10.     long int screensize;
  11.     char *map_fb; 
  12.     
  13. }FBDEV;

  14. void init_dev(FBDEV *dev)
  15. {
  16.     FBDEV *fr_dev=dev;

  17.     fr_dev->fdfd=open("/dev/fb0",O_RDWR);
  18.     printf("the framebuffer device was opended successfully.\n");

  19.     ioctl(fr_dev->fdfd,FBIOGET_FSCREENINFO,&(fr_dev->finfo)); //获取 固定参数

  20.     ioctl(fr_dev->fdfd,FBIOGET_VSCREENINFO,&(fr_dev->vinfo)); //获取可变参数

  21.     fr_dev->screensize=fr_dev->vinfo.xres*fr_dev->vinfo.yres*fr_dev->vinfo.bits_per_pixel/8; 

  22.     fr_dev->map_fb=(char *)mmap(NULL,fr_dev->screensize,PROT_READ|PROT_WRITE,MAP_SHARED,fr_dev->fdfd,0);

  23.     printf("init_dev successfully.\n");
  24. }

  25. void draw_dot(FBDEV *dev,int x,int y) //(x.y) 是坐标
  26. {
  27.     FBDEV *fr_dev=dev;
  28.     int *xx=&x;
  29.     int *yy=&y;    

  30.     long int location=0;
  31.     location=location=(*xx+fr_dev->vinfo.xoffset)*(fr_dev->vinfo.bits_per_pixel/8)+
  32.                  (*yy+fr_dev->vinfo.yoffset)*fr_dev->finfo.line_length;
  33.     int b=10;
  34.     int g=10;
  35.     int r=10;
  36.     unsigned short int t=r<<11|g<<5|b;
  37.     *((unsigned short int *)(fr_dev->map_fb+location))=t;
  38. }


  39. void draw_line(FBDEV *dev,int x1,int y1,int x2,int y2) 
  40. {
  41.     FBDEV *fr_dev=dev;
  42.     int *xx1=&x1;
  43.     int *yy1=&y1;
  44.     int *xx2=&x2;
  45.     int *yy2=&y2;

  46.     int i=0;
  47.     int j=0;
  48.     int tekxx=*xx2-*xx1;
  49.     int tekyy=*yy2-*yy1;

  50.     //if((*xx2>=*xx1)&&(*yy2>=*yy1))
  51.     if(*xx2>=*xx1)
  52.     {    
  53.         for(i=*xx1;i<=*xx2;i++)
  54.         {
  55.             j=(i-*xx1)*tekyy/tekxx+*yy1;
  56.             draw_dot(fr_dev,i,j);
  57.         }
  58.     }
  59.     else
  60.     {
  61.         //if(*xx2<*xx1)
  62.         for(i=*xx2;i<*xx1;i++)
  63.         {
  64.             j=(i-*xx2)*tekyy/tekxx+*yy2;
  65.             draw_dot(fr_dev,i,j);
  66.         }
  67.     }


  68. }

  69. void draw_rect(FBDEV *dev,int x1,int y1,int x2,int y2)
  70. {
  71.     FBDEV *fr_dev=dev;
  72.     int *xx1=&x1;
  73.     int *yy1=&y1;
  74.     int *xx2=&x2;
  75.     int *yy2=&y2;
  76.     int i=0,j=0;
  77.    
  78.     for(j=*yy1;j<*yy2;j++) //注意 这里要 xx1 < xx2
  79.         for(i=*xx1;i<*xx2;i++)
  80.         {

  81.             draw_dot(fr_dev,i,j);            
  82.         
  83.         }
  84.     
  85. }
  86. int main()
  87. {
  88.     FBDEV     fr_dev;
  89.     fr_dev.fdfd=-1;
  90.     init_dev(&fr_dev);
  91.    
  92.     draw_line(&fr_dev,0,0,100,100);
  93.     draw_line(&fr_dev,200,200,110,110);
  94.     draw_line(&fr_dev,10,200,150,100);
  95.     draw_line(&fr_dev,300,10,160,90);
  96.     draw_rect(&fr_dev,300,200,320,240);
  97.         
  98.     printf("bye the framebuffer\n");
  99.     munmap(fr_dev.map_fb,fr_dev.screensize);
  100.     close(fr_dev.fdfd);
  101.     
  102.     return 0;    
  103. }

### 回答1: 好的,以下是一段用C语言编的通过framebuffer控制LCD的程序示例: ```c #include <stdio.h> #include <stdlib.h> #include <fcntl.h> #include <unistd.h> #include <sys/ioctl.h> #include <linux/fb.h> #include <linux/kd.h> #define DEVICE_PATH "/dev/fb0" int main() { int fb_fd = open(DEVICE_PATH, O_RDWR); if (fb_fd < 0) { perror("Failed to open framebuffer device"); exit(EXIT_FAILURE); } struct fb_var_screeninfo vinfo; if (ioctl(fb_fd, FBIOGET_VSCREENINFO, &vinfo) < 0) { perror("Failed to get variable screen information"); exit(EXIT_FAILURE); } int xres = vinfo.xres; int yres = vinfo.yres; int bpp = vinfo.bits_per_pixel; size_t fb_size = xres * yres * bpp / 8; char *fbp = (char *) mmap(NULL, fb_size, PROT_READ | PROT_WRITE, MAP_SHARED, fb_fd, 0); if (fbp == MAP_FAILED) { perror("Failed to mmap framebuffer device"); exit(EXIT_FAILURE); } int x, y; for (y = 0; y < yres; y++) { for (x = 0; x < xres; x++) { long int location = (x + vinfo.xoffset) * (vinfo.bits_per_pixel / 8) + (y + vinfo.yoffset) * vinfo.bytes_per_line; if (bpp == 32) { // ARGB *(fbp + location + 0) = 0x00; // blue *(fbp + location + 1) = 0x00; // green *(fbp + location + 2) = 0xFF; // red *(fbp + location + 3) = 0x00; // alpha } else if (bpp == 16) { // RGB565 unsigned short int color = 0xF800; // red *((unsigned short int *)(fbp + location)) = color; } } } munmap(fbp, fb_size); close(fb_fd); return 0; } ``` 这段程序使用了Linux系统提供的framebuffer设备文件"/dev/fb0"来控制LCD显示。程序首先通过ioctl函数获取屏幕信息,包括分辨率、位深等。然后通过mmap函数将framebuffer映射到用户空间,之后就可以通过修改映射内存来改变LCD显示内容。 程序的示例操作是将屏幕上所有像素点设置为蓝色(ARGB格式)或红色(RGB565格式),可以根据实际需求修改。值得注意的是,操作framebuffer需要有足够的权限,通常需要以root用户身份运行。 ### 回答2: 使用C语言编通过framebuffer控制LCD的程序,可以通过以下步骤完成: 步骤1:打开framebuffer设备文件 首先,需要通过打开framebuffer设备文件与LCD屏幕进行交互。我们可以使用open()函数来打开设备文件,例如: int fbfd = open("/dev/fb0", O_RDWR); 步骤2:获取屏幕参数 获取LCD屏幕的信息,例如分辨率、色深等。可以使用ioctl()函数调用FBIOGET_VSCREENINFO命令获取,例如: struct fb_var_screeninfo vinfo; ioctl(fbfd, FBIOGET_VSCREENINFO, &vinfo); 步骤3:映射framebuffer内存到用户空间 使用mmap()函数将LCD屏幕数据的物理内存映射到应用程序的用户空间,以便实现对屏幕的直接读。例如: char *fbp = (char *)mmap(0, screensize, PROT_READ | PROT_WRITE, MAP_SHARED, fbfd, 0); 步骤4:绘制图像 使用C语言的图形绘制函数,例如直线、矩形、圆等,以及更高级的绘图库(如SDL、OpenGL等),在framebuffer上进行图像绘制。例如: for (y = 0; y < vinfo.yres; y++) { for (x = 0; x < vinfo.xres; x++) { location = (x+vinfo.xoffset) * (vinfo.bits_per_pixel/8) + (y+vinfo.yoffset) * finfo.line_length; *(fbp + location) = 255; // 设置像素点的红色分量 *(fbp + location + 1) = 0; // 设置像素点的绿色分量 *(fbp + location + 2) = 0; // 设置像素点的蓝色分量 *(fbp + location + 3) = 0; // 设置像素点的透明度 } } 步骤5:刷新屏幕 在完成图像绘制后,需要调用刷屏函数,将缓冲区的图像数据刷新到LCD屏幕上。例如: ioctl(fbfd, FBIOPAN_DISPLAY, &vinfo); 步骤6:关闭framebuffer设备文件 在程序结束时,需要关闭framebuffer设备文件,例如: munmap(fbp, screensize); close(fbfd); 通过上述步骤,使用C语言编程序即可通过framebuffer控制LCD屏幕,实现图像的显示和更新。请注意,具体实现可能需要根据不同的硬件平台和操作系统做相应的调整和修改。 ### 回答3: 使用C语言编通过framebuffer控制LCD的程序需要以下步骤: 1. 打开framebuffer设备:首先需要打开framebuffer设备文件,一般位于/dev/fb0。可以使用open()函数来打开该设备文件,并获取文件描述符。 2. 获取设备可变参数:通过调用ioctl()函数获取framebuffer设备的可变参数,包括分辨率、像素格式、颜色位数等信息。这些参数会存储在一个结构体变量,比如fb_var_screeninfo。 3. 分配内存缓冲区:根据设备分辨率和色彩格式,计算出所需的内存缓冲区大小,并使用malloc()函数分配足够的内存空间来存储图像数据。 4. 清空内存缓冲区:使用memset()函数将内存缓冲区清空为全黑色或全白色,以便在刚开始时显示正确的颜色。 5. 绘制图像数据:根据显示需求将需要显示的图像数据填充到内存缓冲区。可以通过修改内存缓冲区的相应像素值来绘制图像。 6. 将图像数据刷新到屏幕:通过write()函数将内存缓冲区的图像数据framebuffer设备文件,从而实现将图像显示到屏幕上。 7. 关闭framebuffer设备:完成显示操作后,使用close()函数关闭framebuffer设备文件。 需要注意的是,具体的控制LCD的方式和步骤可能因LCD型号和设备驱动的不同而有所差异,上述步骤仅为一般情况的简单示意。在实际开发,建议查阅相关LCD设备的文档和驱动代码,以便获取更准确的操作方法和参数设置。
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值