10-S3C2440驱动学习(四)嵌入式linux-LCD驱动程序

发布者:tnzph488最新更新时间:2023-02-14 来源: eefocus关键字:S3C2440  linux 手机看文章 扫描二维码
随时随地手机看文章

核心层的代码以fbmem.c为主,核心层包括许多与具体硬件无关的代码,并且提供了API给用户空间。用户空间使用系统调用,系统调用会使用相应的API函数,最后会调用驱动层实现功能。最终操作到硬件,对于不同的设备,驱动层的代码将有所不同。



一、LCD内核驱动实现分析


内核中包含了LCD驱动程序S3c2410fb.c,通过platform平台驱动框架实现,参考其现在自己写。


字符设备驱动编写往往包括,那么LCD也不例外:

应用程序open的时候,会调用fbmem里面file_operation的open,这个open里面会调用硬件注册进来的结构体的一些函数和属性。


1、fbmem.c分析(内核写好的LCD驱动框架,里面实现一些接口,硬件平台来使用)


(1)入口函数fbmem_init;


fbmem_init(void)

{

create_proc_read_entry("fb", 0, NULL, fbmem_read_proc, NULL);

 

if (register_chrdev(FB_MAJOR,"fb",&fb_fops))

printk("unable to get major %d for fb devsn", FB_MAJOR);

 

fb_class = class_create(THIS_MODULE, "graphics");

if (IS_ERR(fb_class)) {

printk(KERN_WARNING "Unable to create fb class; errno = %ldn", PTR_ERR(fb_class));

fb_class = NULL;

}

return 0;

}


fbmem_init里面注册字符设备,其主设备号为29.并且创建了类class_create,但是没创建设备节点


fbmem_init--》register_chrdev--》#define FB_MAJOR             29

(2)假设


app: open("/dev/fb0", ...)   主设备号: 29, 次设备号: 0  //应用程序打开 /dev/fb0,主设备号29 次设备号0



--------------------------------------------------------------


kernel:


        fb_open    


             int fbidx = iminor(inode);                                // 会调用到fb_open ,里面得到次设备号,


             struct fb_info *info = =registered_fb[0];         //fb_info 这个结构体等于registered_fb数组里面的此设备号检索出来,fb_info 有open的话会调用其open函数




app: read()                                                                 //应用程序read的时候


---------------------------------------------------------------


kernel:


              fb_read             //最终调用到内核的fb_read函数 


                     intfbidx = iminor(inode);                                 //得到次设备号0


                     struct   fb_info *info = registered_fb[fbidx];    //在registered_fb数组里得到一个fb_info 结构体


                     if(info->fbops->fb_read)


                            returninfo->fbops->fb_read(info, buf, count, ppos);          //有读函数就调用


                     src= (u32 __iomem *) (info->screen_base + p);                      //没有从screen_base显存基地址读


                     dst= buffer;


                     *dst++= fb_readl(src++);


                     copy_to_user(buf,buffer, c)                                                     //copy_to_user返回给应用程序


因此:


open read都依赖fb_info结构体,从registered_fb数组中得到fb_info结构体。也就是说内核中主设备号为29的设备可能有很多,open的时候根据次设备号从registered_fb数组得到一个fb_info。registered_fb数组是硬件注册来完成初始化的。


(3)registered_fb在哪里被设置?

搜索后发现在:


register_framebuffer(Fbmem.c (driversvideo))



register_framebuffer(struct fb_info*fb_info)


{undefined


       registered_fb[i]= fb_info;


}


S3c2410fb.c里面有使用register_framebuffer。register_framebuffer是供给硬件设备驱动里面掉用。框图如下

到此已经可以很清晰看出LCD的框架。首先内核帮助我们实现了一个主设备号为29的设备,此时只创建了类,并没有在类下创建设备节点。当我们的下层硬件驱动掉用注册函数的时候,会初始化registered_fb结构体,并创建设备节点。此时应用程序可以来打开一个设备节点了,比如open("/dev/fb0", ...),最终会调用到fbmem核心层提供的open函数,这个open函数中根据次设备号,在registered_fb数组中取出硬件注册进来的结构体,调用里面的open函数,或者使用一些属性。这样内核可以方便管理类似的设备了。


注意fbmem核心层和platform_device是无关的,内核注册了fbmem,并且实现了一个platform_device完成初始化,S3c2410fb入口函数,注册平台设备,由于系统中有同名设备,所以.probe= s3c2410fb_probe会被调用,这个probe函数中实现向上fbmem核心层进行注册。


(4)内核已经帮我们实现好了LCD驱动框架的一部分,以及供硬件驱动掉用的接口函数,怎么写LCD硬件部分的驱动程序呢?根据S3c2410fb.c总结以下几部:


(上层fbmem内核已经写好,并完成上层驱动注册。我们要做的是写出硬件部分的函数,来初始化registered_fb)


1. 分配一个fb_info结构体: 怎么分配:framebuffer_alloc

2. 设置fb_info里面的相关参数

3. 注册:register_framebuffer

4. 硬件相关的设置

(5)参考S3c2410fb.c来分析如何写硬件部分的驱动程序。


内核中是通过平台总线驱动来实现的,现在我们独立出LCD驱动来编写。


module_init(s3c2410fb_init);


int __devinit s3c2410fb_init(void)


{undefined


       returnplatform_driver_register(&s3c2410fb_driver);


}


static struct platform_drivers3c2410fb_driver = {undefined


       .probe           = s3c2410fb_probe,


       .remove         = s3c2410fb_remove,


       .suspend = s3c2410fb_suspend,


       .resume         = s3c2410fb_resume,


       .driver            = {undefined


              .name     = "s3c2410-lcd",


              .owner    = THIS_MODULE,


       },


};


static int __init s3c2410fb_probe(structplatform_device *pdev)


{undefined


       structs3c2410fb_info *info;


       structfb_info    *fbinfo;


       fbinfo= framebuffer_alloc(sizeof(struct s3c2410fb_info), &pdev->dev);.


设置fbinfo。。。


       ret= register_framebuffer(fbinfo);


}


s3c2410fb_probe –>register_framebuffer(fbinfo);


fbmem.c系统实现并抽象出来的,使用的时候依赖于底层框架实现,如s3c2410fb_probe里面的内容决定了LCD的具体操作。


(6)设置了LCD的一些信息,现在设置一些硬件的相关配置。如LCD寄存器配置等。

(7)以下引脚为触摸屏的:

(8)设置内容保包括:


每来一个时钟VCLK,打出一个像素点的颜色,染色由VD1-VD23的值决定。


水平垂直同步型号HSYNC VSYNC


显存:存着颜色像素,从里面取出值,打到LCD上。


VM有效的时候打出颜色,无效的时候只是移动不打出颜色

因此,需要设置哪些:


(1)设置LCD控制器


VCLK:发出合适时钟,看LCD手册


(2)分配显存,把地址告诉LCD控制器,告诉颜色格式,一个像素多少字节表示。


内存中分配显存,存储着要显示到屏幕上的像素值,通过从显存中取值,一个一个显示到LCD上。


(3)配置相关引脚为LCD管角。

因此LCD驱动程序涉及到的要点就是以上内容,和常见的字符设备驱动程序,并没有很大差别。


二、自己写LCD驱动程序过程


(1)参考内核自带的LCD驱动程序S3c2410fb.c (driversvideo):

(2)复制参考的头文件,写入口函数,出口函数,协议

(3)需要做哪些事情:



1. 分配一个fb_info结构体: 怎么分配:framebuffer_alloc

2. 设置fb_info里面的相关参数

3. 硬件相关的设置

4. 注册:register_framebuffer

       /* 1分配一个fb_info */

       s3c_lcd= framebuffer_alloc(0, NULL);


。。。


       /* 4注册 */


       register_framebuffer(s3c_lcd);


(4)设置fb_info里面的相关参数

        /*设置 */


        /* 1设置固定的参数 */


/* 2设置可变的参数 */


/* 3设置操作函数 */


/* 4其他的设置 */


(5)几个重要的结构体--fb_info结构体



struct fb_info {

int node;/*  序号索引值,/dev/fb0,/dev/fb1  其中0,1 就是从这里获得的*/

int flags;

struct fb_var_screeninfo var; /* Current var *//* 可变参数,很重要 */

struct fb_fix_screeninfo fix; /* Current fix *//* 固定参数,很重要 */

struct fb_monspecs monspecs; /* Current Monitor specs */

struct work_struct queue; /* Framebuffer event queue */

struct fb_pixmap pixmap; /* Image hardware mapper */

struct fb_pixmap sprite; /* Cursor hardware mapper */

struct fb_cmap cmap; /* Current cmap */

struct list_head modelist;      /* mode list */

struct fb_videomode *mode; /* current mode */

 

#ifdef CONFIG_FB_BACKLIGHT

/* assigned backlight device */

/* set before framebuffer registration, 

   remove after unregister */

struct backlight_device *bl_dev;

/* Backlight level curve */

struct mutex bl_curve_mutex;

u8 bl_curve[FB_BACKLIGHT_LEVELS];

#endif

#ifdef CONFIG_FB_DEFERRED_IO

struct delayed_work deferred_work;

struct fb_deferred_io *fbdefio;

#endif

struct fb_ops *fbops;/* 固定参数,很重要 */

struct device *device; /* This is the parent */

struct device *dev; /* This is this fb device */

int class_flag;                    /* private sysfs flags */

#ifdef CONFIG_FB_TILEBLITTING

struct fb_tile_ops *tileops;    /* Tile Blitting */

#endif

char __iomem *screen_base; /* Virtual address *//* "显存“的基地址 */

unsigned long screen_size; /* Amount of ioremapped VRAM or 0 */  /* ”显存“的大小 */ 

void *pseudo_palette; /* Fake palette of 16 colors */ /* 16位假的调色板 */ 

#define FBINFO_STATE_RUNNING 0

#define FBINFO_STATE_SUSPENDED 1

u32 state; /* Hardware state i.e suspend */

void *fbcon_par;                /* fbcon use-only private area */

/* From here on everything is device dependent */

void *par; /* 这个用来存放私有数据 */

};


fb_fix_screeninfo fix结构体

struct fb_fix_screeninfo {

char id[16]; /* identification string eg "TT Builtin" */

unsigned long smem_start; /* Start of frame buffer mem */

/* (physical address) */

__u32 smem_len; /* Length of frame buffer mem */

__u32 type; /* see FB_TYPE_* */

__u32 type_aux; /* Interleave for interleaved Planes */

__u32 visual; /* see FB_VISUAL_* */ 

__u16 xpanstep; /* zero if no hardware panning  */

__u16 ypanstep; /* zero if no hardware panning  */

__u16 ywrapstep; /* zero if no hardware ywrap    */

__u32 line_length; /* length of a line in bytes    */

unsigned long mmio_start; /* Start of Memory Mapped I/O   */

/* (physical address) */

__u32 mmio_len; /* Length of Memory Mapped I/O  */

__u32 accel; /* Indicate to driver which */

/*  specific chip/card we have */

__u16 reserved[3]; /* Reserved for future compatibility */

};


fb_var_screeninfo var结构体

struct fb_var_screeninfo {

__u32 xres; /* visible resolution */

__u32 yres;

__u32 xres_virtual; /* virtual resolution */

__u32 yres_virtual;

__u32 xoffset; /* offset from virtual to visible */

__u32 yoffset; /* resolution */

 

__u32 bits_per_pixel; /* guess what */

__u32 grayscale; /* != 0 Graylevels instead of colors */

 

struct fb_bitfield red; /* bitfield in fb mem if true color, */

[1] [2] [3]
关键字:S3C2440  linux 引用地址:10-S3C2440驱动学习(四)嵌入式linux-LCD驱动程序

上一篇:11-S3C2440驱动学习(五)嵌入式linux-网络设备驱动(一)虚拟网卡驱动程序
下一篇:09-S3C2440驱动学习(三)嵌入式linux-platform平台总线驱动程序及分离分层构建驱动框架

推荐阅读最新更新时间:2024-11-12 13:21

Linux核心4.9出炉,Linus保证史上最大改版!
Google的Ara模组化手机计画虽然停摆了,但是贡献了大量程式码到4.9新版,尤其是Ara计画的Greybus硬体协定也加入了Linux 4.9版。 问世满25周年的Linux,释出了史上最大一次的核心改版Linux 4.9新版,Google喊卡的模组化手机Ara专案,反而在这次4.9新版中贡献了大量程式码。Linux之父Linus Torvalds在11日释出了4.9版,并在发布邮件中兴奋地说:“我非常确定这次是史上最大一次版本发布,commit数是有史以来最多的一版。” 过去几次核心程式码大修改大多为了特定议题,如4.2版核心大量增加了AMD GPU的支援程式码,3.2版核心程式码大增是因为开发团队的组织重整,但4.
[手机便携]
S3C2440裸机学习[2] - LCD驱动原理及代码分析[一]
1. LCD工作的硬件需求: 要使一块LCD正常的显示文字或图像,不仅需要LCD驱动器,而且还需要相应的LCD控制器。在通常情况下,生产厂商把LCD驱动器会以COF/COG的 形式与LCD玻璃基板制作在一起,而LCD控制器则是由外部的电路来实现,现在很多的MCU内部都集成了LCD控制器,如S3C2410/2440等。通 过LCD控制器就可以产生LCD驱动器所需要的控制信号来控制STN/TFT屏了。 2. S3C2440内部LCD控制器结构图: 我们根据数据手册来描述一下这个集成在S3C2440内部的LCD控制器: a:LCD控制器由REGBANK、LCDCDMA、TIMEGEN、VIDPRCS寄存器组成; b:REGBAN
[单片机]
(mini2440)建立交叉编译环境+配置linux内核
系统ubuntu12.04(非虚拟机下) mini2440 CPU型号:S3C2440AL-40 Nandflash型号:K9F1G08 Norflash型号:SST39VF1601 LCD:统宝 240 x 320 $:普通账户 #:root账户 *当shell下输入路径时可使用tab键自动补全 (一)建立交叉编译环境 1.将mini2440光盘中的linux文件夹拷贝到 /home/lianghuiyong 并改名为Linux_share (其中两个文档为我后面添加进去的) 2.Ctrl+Alt+T打开shell 3.$ su - root (切换root权限) 4.# cd /home/lianghuiyong/
[单片机]
(mini2440)建立交叉编译环境+配置<font color='red'>linux</font>内核
基于FFmpeg的远程视频监控系统编解码
0 引言 随着视频编解码技术、计算机网络技术、数字信号处理技术和嵌入式系统的发展,以嵌入式网络视频服务器为核心的远程视频监控系统开始在市场上崭露头角。该系统把摄像机输出的模拟视频信号通过内置的嵌入式视频编码器直接转换成视频流,通过计算机网络传输出去。嵌入式网络视频服务器具备视频编码处理、网络通信、系统控制等强大功能,直接支持网络视频传输和网络管理,使得监控范围达到前所未有的广度。在远程视频监控系统中,摄像头获取的原始视频流在传输之前需要压缩,而FFmpeg可以将原始视频压缩为H264格式视频流,H264是一种被广泛使用的高精度视频的录制、压缩和发布格式,因此采用FFmpeg来实现。 1 系统方案 系统是在S3C2440平台上运行嵌
[单片机]
基于FFmpeg的远程视频监控系统编解码
基于S3C2440的LED背光源节电系统设计方案
引言 节能环保技术是当前世界所关注的焦点,在液晶显示模组中,背光源的功耗最高可占总功耗的50%以上。尤其在10in 以下显示产品如手机、PDA、MP3 等便携式设备中,基本采用电池供电,功耗问题尤为突出。为有效降低液晶显示器背光源的亮度,以达到节电目的,本文在ARM开发平台上实现了一种基于直方图变换的背光源调光方法,实验证明,本文提出的方法在失真度为5%的情况下可实现背光节电约35%. 1 背光源调光方案 以TFT 液晶面板结构为例,包括背光、偏光片、液晶阵列、彩色滤光片等部分,人眼所感知的显示图像为上述各部分的综合效果。假设背光亮度归一化后设为b(为 区间实数),0 对应于背光关闭情况,1 对应于背光发光亮度最大情况。若
[电源管理]
基于<font color='red'>S3C2440</font>的LED背光源节电系统设计方案
S3C2440的NAND Flash控制器
首先还是启动方式NAND Flash的过程,因为NAND Flash是不支持程序在NAND内部运行的所以要把NAND搬到内存中运行,由于S3C2440内部的SRAM只有4K,当程序远大于4K时,这种方法是行不通的,于是我们可以让NAND中4K之后的程序搬到SDRAM中来运行 正如数据手册上所说,NOR的价格较NAND来说比较昂贵,再加上NOR的容量远比NAND小的多,再比如NOR一页的字节数也比NAND大的很多,而要在某个地址写入数据,必须先把地址所在的页擦除,所以更多的时候NAND用来存储数据,NOR用来存储程序。 如上图模型,当程序的容量大于4K的时候,NAND的前4K的程序将会被硬件自动引导到2440的S
[单片机]
<font color='red'>S3C2440</font>的NAND Flash控制器
OpenCV2.0.0移植到ARM9(二)(JZ2440----S3c2440)
1、交叉编译libjpeg 为了使OpenCV能处理jpeg图像,我们必须事先交叉编译好libjpeg这里使用的版本是jpegsrc.v6b。 下载地址:https://sourceforge.net/projects/libjpeg/files/libjpeg/6b/ 这里使用的安装包:jpegsrc.v6b.tar.gz。 (1)解压、配置 jpegsrc.v6b.tar.gz放在/work/system/目录下. 解压:$tar –zvxf jpegsrc.v6b.tar.gz 进入目录:$cd jpeg-6b 在/work/system/目录下新建一个libjpeg-arm目录,命令为:$sudo mkdir
[单片机]
OpenCV2.0.0移植到ARM9(二)(JZ2440----S3c2440)
基于ARM Linux的无线音视频对讲系统分析
随着数字化和网络化时代的到来,尤其是宽带无线网络的发展,为音视频这样大数据量传输业务在无线网络上的应用提供了契机。同时由于音视频独特的感官特性,使其相关的应用需求也变得越来越迫切。无线多媒体是多媒体和移动通信这两个领域的技术相互融合的产物,成为当今通信领域的一个热点。鉴于Linux内核的开源性,采用其作为操作系统,从而使整个系统具有更好的实时性和稳定性。整个系统以ARM11为核心处理器、采用新一代视频编解码标准H.264进行编译码,并通过无线网络传输音视频。它充分利用S3C6410微处理器内部集成的多媒体编解码器(Multi-Formatvideo Codec,MFC),有效提高了系统的性价比。整个系统为无线多媒体音视频的传输提供
[单片机]
基于ARM <font color='red'>Linux</font>的无线音视频对讲系统分析

推荐帖子

Proteus 8.4的逆天行为--模型化软件开发已成趋势!
看看,这幅图,Proteus8.4的逆天了,直接用流程图的方式生成其代码!Proteus8.4的逆天行为--模型化软件开发已成趋势!:time:有点厉害啊:~o:~o:~o:~o:~o:~o:~o变plc咯这个功能从哪里启用?好用不简单的代码可以用然而并没有什么卵用呵呵,:time::time::time:
平湖秋月 微控制器 MCU
06、国产FPGA 正点原子DFPGL22G开发板测评【学习篇】IP核FIFO
1、简介根据FIFO工作的时钟域,可以将FIFO分为同步FIFO和异步FIFO。同步FIFO是指读时钟和写时钟为同一个时钟,在时钟沿来临时同时发生读写操作。异步FIFO是指读写时钟不一致,读写时钟是互相独立的。2、实验任务本节的实验任务是使用Pango生成FIFOIP核,并实现以下功能:当FIFO为空时,向FIFO中写入数据,写入的数据量和FIFO深度一致,即FIFO被写满;然后从FIFO中读出数据,直到FIF
1nnocent 国产芯片交流
基于低压电力线载波技术的病房呼叫系统
阐述了病房呼叫系统的基本组成以及一些相关的硬件设计和软件设计。基于MSC—51单片机多机通讯的设计思路,以SC1128扩频通信芯片为基础,辅以外围电路,设计了一个利用低压载波传输数据的病房呼叫系统;并基于PowerBuilder语言知识设计了病房呼叫系统的管理界面。基于低压电力线载波技术的病房呼叫系统顶................
fighting 嵌入式系统
时钟芯片ZC1302完全替代DS1302
BENEFITSANDFEATURES*CompletelyManagesAllTimekeepingFunctionsoReal-TimeClockCountsSeconds,Minutes,Hours,DateoftheMonth,Month,DayoftheWeek,andYearwithLeap-YearCompensationValidUpto2100o31x8Battery-BackedGeneral-Pur
四维电子 电源技术
AD的问题
自己建的集成库,画完原理图和封装后,也关联了,生成PCB时。为什么说封装找不到,,,,附文件,大神给看看AD的问题楼主,我看了下你的图,猜测是因为你没有把库文件添加到library里面来哦。。你先把库添加了然后生成的时候才能找得到哦。就像你要生成。但是没告知清楚人家地址在哪里,人家在他已知的范围内找,,,结果没找到。。 没有加到LIBRARY里怎么可能把我自己画的元件加到原理图中,,,,,,是没有封装,不是没有模型 。。。。没有加到library是可以添加元件到原理图的。。。{
wd5039 PCB设计
【2024 DigiKey 创意大赛】二月柳絮大作战项目——作品提交
二月柳絮大作战作者:Maker_kun一、作品简介二月柳絮大作战主控板使用乐鑫科技ESP32-S3-LCD-Ev-Board,该开发板集成ESP32S3WiFi模块,同时开发板具备480*480的LCD液晶屏幕,对于用户界面设计非常友好;传感器使用sensrion公司的SPS30激光颗粒物传感器,可以感知PM2.5PM10以及颗粒物粒子大小,传感器可连续工作6年之久;空气状态显示采用ws2812LED构成8*8构成的点阵屏幕,可以直观通过颜色展示当前空气质量;负载
Maker_kun DigiKey得捷技术专区
小广播
设计资源 培训 开发板 精华推荐

最新单片机文章
何立民专栏 单片机及嵌入式宝典

北京航空航天大学教授,20余年来致力于单片机与嵌入式系统推广工作。

换一换 更多 相关热搜器件
更多往期活动
随便看看

 
EEWorld订阅号

 
EEWorld服务号

 
汽车开发圈

电子工程世界版权所有 京B2-20211791 京ICP备10001474号-1 电信业务审批[2006]字第258号函 京公网安备 11010802033920号 Copyright © 2005-2024 EEWORLD.com.cn, Inc. All rights reserved