当前位置:首页 > 嵌入式 > 嵌入式教程
[导读]高清嵌入式系统中的DVI驱动开发

摘要:以飞思卡尔半导体的MX51为系统硬件平台,给出了高清嵌入式产品中DVI视频显示接口的实现方案;通过处理器的LCD1接口,外扩TFP 410芯片实现DVI视频输出;分析了framebuffer的实现机制及其在驱动中的应用;详细讲述Linux2.6.28内核下基于framebuffer开发DVI驱动的方法及步骤。该设计应用于高清机顸盒等产品中,效果良好。
关键词:Linux;设备驱动;DVI;framebuffer;MX51

引言
    DVI是Digital VisualInterface(数字视频接口)的缩写。在嵌入式电子领域,像DVI这样的高清接口应用越来越多,很多嵌入式产品采用H.264视频编码技术,支持播放H.264格式的720P分辨率的视频文件,这就需要至少1024×768分辨率的显示输出设备。
    MX51是飞思卡尔半导体的基于ARM Cortex-A8内核的高端ARM嵌入式多媒体处理器,支持720P视频多种格式的硬解码,可以用来开发高清机顶盒、上网本等产品,很多情况下需要集成DVI这样的高清视频端子。
    在嵌入式电子产品中,Linux操作系统占有越来越多的市场份额。本文采用Linux2.6.28内核和MX51作为系统的软、硬件平台,详细论述了基于framebtffer技术开发DVI显示驱动程序的方法。

1 DVI概述
    DVI接口只在一些高端显示器上可以看到,一般常见的液晶显示器只有VGA接口。VGA接口显示的是模拟信号,而DVI接口显示的是数字信号,它传输没有经过压缩的数字信号,最高速率可达4.9 Gbps,对高清视频显示可以达到较好的保真度,减少模拟信号传输时的信号损失。
    DVI基于TMDS(Transition Minimized Differential Signaling,转换最小差分信号)技术来传输数字信号,TMDS运用先进的编码算法把8位数据(R、G、B中的每路基色信号)通过最小转换编码为10位数据(包含行场同步信息、时钟信息、数据DE、纠错等),经过DC平衡后,采用差分信号传输数据。DVI和LVDS、TTL相比有较好的电磁兼容性能,可以用低成本的专用电缆实现长距离、高质量的数字信号传输。

2 硬件接口
    本设计采用的硬件平台是基于飞思卡尔半导体的MX51多媒体应用处理器开发板。该处理器集成了多种外设接口,其中包括两个液晶显示控制器(LCDC)及其接口,可以连接各类LCD,分辨率最大支持1280×800像素。通过MX51的LCD1接口,外扩德州仪器公司的TFP410芯片实现DVI视频输出,MX51的高清720P视频解码能力需要较大分辨率的显示输出设备。图1为MX51的LCD1接口与TFP410的连接图。


    图1中的TX2±、TX1±、TX0±、TXC±信号是DVI视频输出信号4对,8个信号。DATA[23:0]是视频数据输入信号,对应MX51 LCD1的DATA[23:0];DE、VSYNC、HSYNC、IDCK±等时钟信号分别对应LCD1的相应的引脚。SCL、SDA是I2C总线时钟和数据信号,接MX51 I2C接口的2个引脚。以上硬件电路连接,可实现MX51输出高清视频到DVI芯片,再通过外接LCD显示。MX51处理器内部集成的LCD控制器包括如下主要寄存器:
    ①LSSAR寄存器。设置显示缓冲区的首地址。
    ②LSR寄存器。设置显示缓冲区的大小。
    ③LPCR寄存器。设置像素时钟频率PCD、同步时钟极性FB_SYNC_CLK_INVERT、OE信号极性FB_SYNC_OE_ACT_HIGH、垂直信号时钟极性FB_SY NC_VERT_HIGH_ACT、水平信号时钟极性FB_SYNC_HOR_HIGH_ACT。
    ④LHCR寄存器。设置行同步信号的hsync_len、left_margin和right_margin。
    ⑤LVCR寄存器。设置帧同步信号的vsync_len、upper_margin和lower_margin。
    ⑥LPCCR寄存器。设置屏幕的显示亮度,LPCCR的低8位控制PWM的脉冲高电平占空比,调节范围为0x00~0xFF。
[!--empirenews.page--]
3 Linux的帧缓冲设备
3.1 framebuffer机制
    framebuffer(帧缓冲)是出现在Linux2.2.xx之后版本内核的一种驱动程序接口,在Linux体系中它居于上层应用程序和底层显示设备之间。framebuffer屏蔽了不同显示设备间的差异,将显示设备抽象为帧缓冲区,它是一种供用户态实现直接写屏的抽象设备。framebuffer可
以看成是显存的一个映像,用户通过内存映射将其映射到进程的地址空间后,通过对显示缓冲区的读写操作可直接控制LCD的屏幕输出。
    frameBuffer设备驱动主要基于linux/include/linux/fb.h和linux/drivers/video/fbmem.c这两个文件。fb.h中包含了与帧缓冲设备相关的重要的数据结构。fbmem.c是framebuffer机制的核心程序,它为上层应用程序提供了通用接口,同时也为下层特定硬件提供了接口。其内的函数可对具体硬件进行操作,比如对寄存器进行设置,对显示缓冲进行映射等。
3.2 几个重要的数据结构
    (1)struct fb_info
    这个结构是Linux为帧缓冲区设备定义的驱动层接口,它包含了关于帧缓冲设备属性和操作的完整描述,部分成员定义如下:
   
    其中,var记录用户可以修改的显示控制器参数,包括屏幕分辨率和每个像素的位宽等;fix记录用户不能修改的显示控制器参数;cmap为当前的颜色表;fbops指向对底层硬件操作的函数集;dev表示帧缓冲设备;screen_base为I/O映射的虚拟基地址。
    (2)struct fb_ops
    该结构提供了指向底层操作的函数指针,其成员函数最终与LCD控制器硬件打交道,这些函数需要驱动开发者根据LCD控制器的硬件设置及LCD显示屏的硬件参数进行设计。该结构部分成员定义如下:
   
    其中,fb_check_var用于检查可变的屏幕参数,并调整其为硬件支持的值;fb_set_par根据屏幕参数设置具体读写LCD控制器的寄存器以使其进入相应的工作状态,fb_setcolreg设置color寄存器来实现伪颜色表和颜色表的填充。
    (3)struct fb_var_screeninfo
    这是fb_info的成员结构体。它记录了帧缓冲设备和指定显示模式的可修改信息,包括屏幕分辨率、每个像素的位宽、帧延时、行延时等。
    (4)struct fb_fix_screeninfo
    这是fb_info的成员结构体,它描述显示卡的属性,并且在系统运行时不能被修改,例如缓冲区的首地址、长度等。当一种模式被设定后,内存信息由显示卡硬件给出,内存的位置等信息就不可修改。
    MX51将LCD控制器直接嵌入到处理器芯片内部,这为嵌入式系统关于显示功能的扩展提供了直接接口。LCD控制器驱动是DVI设备驱动的核心,它是一个标准的framebuffer设备驱动。设计驱动程序,首要的是配置LCD控制器,设置帧缓冲区,这在很大程度上要依赖于上述的数据结构,驱动设计需填写相关结构体并完成系统指定的接口函数。
[!--empirenews.page--]
4 DVI驱动程序设计
4.1 平台驱动
    从Linux2.6起引入了一套新的驱动管理和注册机制:platform_device和platform_driver。
    设备用platform_device表示,驱动用platform_driver注册。平台设备包括基于端口的设备、外围总线和集成在片上系统中的大多数控制器,作为MX51片上的独立硬件模块。LCD控制器是一个平台设备,因此驱动设计中需包含平台驱动。平台驱动的任务是向系统注册用到的设备,此处包括MX51的LCD控制器和TFP410 DVI视频输出芯片,使得设备驱动加载时可以从系统中查询到相应的设备是已注册的状态,然后执行设备驱动程序中的probe函数。
    在arch/arm/mach-mx51/rex51_3stack.C中,没置platform_device结构变量mxc_fb_device和i2c_board_info。结构变量mxc_i2cl_ board_info定义LCD控制器和TFP410设备。
    调用函数platform_device_register(&mxc_fb_device)和i2c_register_board_info(1,mxc_i2cl_board_info,ARRAY_SIZE(mxc_i2cl_ board_info))向系统注册以上设备。
4.2 设备驱动
4.2.1 LCD控制器驱动
    LCD控制器驱动是一个标准的帧缓冲设备驱动。首先在drivers/video/mxc/mxc_ipuv3_fb.c中定义全局结构变量mxcfb_driver:
   
    然后,在驱动入口函数mxcfb_init(void)中调用platform_driver_register(&mxcfb_driver)注册驱动,当驱动加载成功后,会自动调用探测函数mxcfb_probe。
    mxcfb_probe是驱动设计中的重要函数。主要负责初始化硬件。申请中断、分配framebuffer所需的内存、注册帧缓冲设备等,以下是与framebuffer相关的操作。
    ①调用mxcfb_init_fbinfo(&pdev->dev,&mxcfb_ops)函数,在其内通过framebuffer_alloc函数,为mx51帧缓冲信息结构体struct mxcfb_info分配所需空间。参数mxcfb_ops的定义如下:
   
    mxcfb_ops定义了指向底层操作的一系列函数,这些函数针对MX51帧缓冲操作,是framebuffer核心驱动操作的具体实现。[!--empirenews.page--]
    ②初始化帧缓冲信息结构体fb_info的固定和可变参数,填充fb_var_screeninfo var和fb_fix_screeninfo fix成员。
定义fbi为struct fb_info类型的指针,通过fbi->fbops=&mxcfb_ops语句,将已定义的文件操作接口mxcfb_ops赋予fb-info结构的fbops成员。
    调用mxcfb_check_var(&fbi->var,fbi)函数,检查和调整fb_info结构中变量var的值。var是一个struct fb_var_screeninfo类型的变量,表示显示控制器参数,其中与显示输出状态有关的信息,如屏幕分辨率等将在后面的DVI驱动中设置。
    调用mxcfb_set_fix(fbi)函数,用于填充一个struct fb_fix_screeninfo结构变量fbi->fix,它描述了显示输出设备自身的属性。
    ③调用register_framebuffer(fbi)函数,注册帧缓冲驱动程序,该函数只有一个参数,即前面已定义的、指向struct fb_info结构的指针fbi。
4.2.2 DVI设备驱动
    LCD控制器将DVI芯片作为它所连接的显示外设,在完成LCD控制器驱动后还需编写DVI设备驱动。在文件drivers/video/mxc/mxcfb_ dvi.c中定义驱动结构体:
   
    然后,在外设驱动入口函数dvi_init(void)中调用platform_driver_regtster(&dvi_driver)注册DVI驱动,驱动加载后,系统自动调用探测函数dvi_probe,该函数主要实现以下操作:一是指定framebuffer设备,由于MX51IPU(图像处理单元)支持多个framebuffer设备,此处要确定DVI究竟使用Mx51 IPU framebuffer的哪一个设备;二是填充fb_var_screeninfo结构变量var中有关显示输出状态的信息,如屏幕的显示分辨率、画面位置等,为此在程序中定义结构数组video_modes:
   
    结构struct fb_videomode用于描述显示输出状态,调用函数“fb_videomode_to_var(&var,&video_modes[0])”将屏幕显示参数转换为var结构变量的相关成员,由于var的部分成员值已在前面LCD控制器驱动中确定,此处完成了对var全部成员的设置。
    一个frambuffer设备由一个struct fb_info结构表示,本设计用fb_info结构的全局变量regtstered_fb表示系统注册的frambuffer设备,驱动程序的主要任务之一是填充这个结构变量。LCD控制器驱动与DVI外设驱动之间的信息传递,通过该全局变量实现。

5 DVI驱动测试
    首先,通过显示一幅图片测试DVI输出是否正常。通过转换工具(如Image21cd)把一幅1024×768大小的jpg图片转换为RGB 888分辨率、1024x 768的RGB格式的二进制图片。然后键入命令:cp pic.bin/dev/fb0,此时图片显示于屏幕上。接下来,再使用MX51的视频解码测试程序播放一个720P的视频H.264文件,可以看到视频播放清晰流畅,效果很好。

结语
    经测试,DVI驱动程序在MX51平台上成功实现。framebuffer是Linux提供给用户的一个直接面向显示缓冲区的接口,本设计是一个面向应用的framebuffer驱动,文中给出了DVI驱动的整体架构,对主要模块的设计思想和实现方法进行了详细的介绍。

本站声明: 本文章由作者或相关机构授权发布,目的在于传递更多信息,并不代表本站赞同其观点,本站亦不保证或承诺内容真实性等。需要转载请联系该专栏作者,如若文章内容侵犯您的权益,请及时联系本站删除。
换一批
延伸阅读

LED驱动电源的输入包括高压工频交流(即市电)、低压直流、高压直流、低压高频交流(如电子变压器的输出)等。

关键字: 驱动电源

在工业自动化蓬勃发展的当下,工业电机作为核心动力设备,其驱动电源的性能直接关系到整个系统的稳定性和可靠性。其中,反电动势抑制与过流保护是驱动电源设计中至关重要的两个环节,集成化方案的设计成为提升电机驱动性能的关键。

关键字: 工业电机 驱动电源

LED 驱动电源作为 LED 照明系统的 “心脏”,其稳定性直接决定了整个照明设备的使用寿命。然而,在实际应用中,LED 驱动电源易损坏的问题却十分常见,不仅增加了维护成本,还影响了用户体验。要解决这一问题,需从设计、生...

关键字: 驱动电源 照明系统 散热

根据LED驱动电源的公式,电感内电流波动大小和电感值成反比,输出纹波和输出电容值成反比。所以加大电感值和输出电容值可以减小纹波。

关键字: LED 设计 驱动电源

电动汽车(EV)作为新能源汽车的重要代表,正逐渐成为全球汽车产业的重要发展方向。电动汽车的核心技术之一是电机驱动控制系统,而绝缘栅双极型晶体管(IGBT)作为电机驱动系统中的关键元件,其性能直接影响到电动汽车的动力性能和...

关键字: 电动汽车 新能源 驱动电源

在现代城市建设中,街道及停车场照明作为基础设施的重要组成部分,其质量和效率直接关系到城市的公共安全、居民生活质量和能源利用效率。随着科技的进步,高亮度白光发光二极管(LED)因其独特的优势逐渐取代传统光源,成为大功率区域...

关键字: 发光二极管 驱动电源 LED

LED通用照明设计工程师会遇到许多挑战,如功率密度、功率因数校正(PFC)、空间受限和可靠性等。

关键字: LED 驱动电源 功率因数校正

在LED照明技术日益普及的今天,LED驱动电源的电磁干扰(EMI)问题成为了一个不可忽视的挑战。电磁干扰不仅会影响LED灯具的正常工作,还可能对周围电子设备造成不利影响,甚至引发系统故障。因此,采取有效的硬件措施来解决L...

关键字: LED照明技术 电磁干扰 驱动电源

开关电源具有效率高的特性,而且开关电源的变压器体积比串联稳压型电源的要小得多,电源电路比较整洁,整机重量也有所下降,所以,现在的LED驱动电源

关键字: LED 驱动电源 开关电源

LED驱动电源是把电源供应转换为特定的电压电流以驱动LED发光的电压转换器,通常情况下:LED驱动电源的输入包括高压工频交流(即市电)、低压直流、高压直流、低压高频交流(如电子变压器的输出)等。

关键字: LED 隧道灯 驱动电源
关闭