【Linux 驱动】IMX6ULL eLCDIF驱动

1. eLCDIF设备树

lcdif: lcdif@021c8000 {compatible = "fsl,imx6ul-lcdif", "fsl,imx28-lcdif";    //属性reg = <0x021c8000 0x4000>;                             //起始地址 地址大小interrupts = <GIC_SPI 5 IRQ_TYPE_LEVEL_HIGH>;          //中断clocks = <&clks IMX6UL_CLK_LCDIF_PIX>,                 //时钟子系统<&clks IMX6UL_CLK_LCDIF_APB>,<&clks IMX6UL_CLK_DUMMY>;clock-names = "pix", "axi", "disp_axi";                //系统名称status = "disabled";                                   //状态};
* Freescale MXS LCD Interface (LCDIF)Required properties:
- compatible: Should be "fsl,<chip>-lcdif".  Supported chips includeimx23 and imx28.
- reg: Address and length of the register set for lcdif
- interrupts: Should contain lcdif interrupts
- display : phandle to display node (see below for details)* display nodeRequired properties:
- bits-per-pixel : <16> for RGB565, <32> for RGB888/666.
- bus-width : number of data lines.  Could be <8>, <16>, <18> or <24>.Required sub-node:
- display-timings : Refer to binding doc display-timing.txt for details.+----------+-------------------------------------+----------+-------+|          |        ↑                            |          |       ||          |        |vback_porch                 |          |       ||          |        ↓                            |          |       |+----------#######################################----------+-------+|          #        ↑                            #          |       ||          #        |                            #          |       ||  hback   #        |                            #  hfront  | hsync ||   porch  #        |       hactive              #  porch   |  len  ||<-------->#<-------+--------------------------->#<-------->|<----->||          #        |                            #          |       ||          #        |vactive                     #          |       ||          #        |                            #          |       ||          #        ↓                            #          |       |+----------#######################################----------+-------+|          |        ↑                            |          |       ||          |        |vfront_porch                |          |       ||          |        ↓                            |          |       |+----------+-------------------------------------+----------+-------+|          |        ↑                            |          |       ||          |        |vsync_len                   |          |       ||          |        ↓                            |          |       |+----------+-------------------------------------+----------+-------+Examples:&lcdif@80030000 {pinctrl-names = "default";                    //pinctrl 名字pinctrl-0 = <&pinctrl_lcdif_dat               //pinctrl 子系统&pinctrl_lcdif_ctrl&pinctrl_lcdif_reset>;display = <&display0>;                        status = "okay";                               //状态display: display {                             //子节点bits-per-pixel = <32>;                     //每个像素点大小bus-width = <24>;                          //总线宽度display-timings {                          //display-timing子节点native-mode = <&timing0>;timing0: timing0 {clock-frequency = <33500000>;      //时钟频率hactive = <800>;                   //水平显示分别率vactive = <480>;                   //垂直显示分别率hfront-porch = <164>;              //水平方向前肩hback-porch = <89>;                //水平方向后肩hsync-len = <10>;                  //水平脉冲宽度vback-porch = <23>;                //垂直方向后肩vfront-porch = <10>;               //垂直方向前肩 vsync-len = <10>;                  //垂直脉冲宽度hsync-active = <0>;                //水平方向有效脉冲极性vsync-active = <0>;                //垂直方向有效脉冲极性de-active = <1>;                   //使能有效极性pixelclk-active = <0>;             //像素时钟边沿采样有效性};};};
};

2. 相关数据结构

(1)struct  fb_info

Linux内核中使用fb_info结构体变量来描述一个framebuffer,在调用register_framebuffer接口注册framebuffer之前,必须要初始化其中的重要数据成员。

        struct  fb_info中成员众多,我们需要着重关注以下成员:

        fb_var_screeninfo:代表可修改的LCD显示参数,如分辨率和像素比特数等;

        fb_fix_screeninfo:代表不可修改的LCD属性参数,如显示内存的物理地址和长度等;

        fb_ops:LCD底层硬件操作接口集。

struct fb_info 
{int node;                       //用来表示该fb设备的次设备号int flags;                      //一个标志位struct mutex lock;              /* Lock for open/release/ioctl funcs */struct mutex mm_lock;           /* Lock for fb_mmap and smem_* fields */struct fb_var_screeninfo var;   /* Current var */        //  fb的可变参数struct fb_fix_screeninfo fix;   /* Current fix */        //  fb的不可变参数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_IOstruct delayed_work deferred_work;struct fb_deferred_io *fbdefio;
#endifstruct fb_ops *fbops;          //  该设备对应的操作方法struct device *device;         /* This is the parent */     //fb设备的父设备struct device *dev;            /* This is this fb device */ //本设备的deviceint class_flag;                /* private sysfs flags */
#ifdef CONFIG_FB_TILEBLITTINGstruct fb_tile_ops *tileops;   /* Tile Blitting */
#endifchar __iomem *screen_base;     /* Virtual address */        //LCD的显存地址(虚拟地址) unsigned long screen_size;     /* Amount of ioremapped VRAM or 0 *///  LCD显存的字节大小void *pseudo_palette;          /* Fake palette of 16 colors */ 
#define FBINFO_STATE_RUNNING    0
#define FBINFO_STATE_SUSPENDED    1u32 state;                     /* Hardware state i.e suspend */void *fbcon_par;               /* fbcon use-only private area *//* From here on everything is device dependent */void *par;/* we need the PCI or similiar aperture base/size notsmem_start/size as smem_start may just be an objectallocated inside the aperture so may not actually overlap */struct apertures_struct {unsigned int count;struct aperture {resource_size_t base;resource_size_t size;} ranges[0];} *apertures;
};struct fb_info

 (2)struct fb_var_screeninfo

Linux内核中使用struct fb_var_screeninfo来描述可修改的LCD显示参数,如分辨率和像素比特数等。

struct fb_var_screeninfo 
{__u32 xres;                  //   水平分辨率 __u32 yres;                  //   垂直分辨率__u32 xres_virtual;          //   虚拟水平分辨率__u32 yres_virtual;          //   虚拟垂直分辨率__u32 xoffset;               //  当前显存水平偏移量__u32 yoffset;               //  当前显存垂直偏移量__u32 bits_per_pixel;        //  像素深度__u32 grayscale;             /* != 0 Graylevels instead of colors */struct fb_bitfield red;      /* bitfield in fb mem if true color, */struct fb_bitfield green;    /* else only length is significant */struct fb_bitfield blue;struct fb_bitfield transp;   /* transparency*/    __u32 nonstd;           /* != 0 Non standard pixel format */__u32 activate;         /* see FB_ACTIVATE_* */__u32 height;           //  LCD的物理高度mm__u32 width;            // LCD的物理宽度mm__u32 accel_flags;      /* (OBSOLETE) see fb_info.flags *//* Timing: All values in pixclocks, except pixclock (of course) */__u32 pixclock;            /* pixel clock in ps (pico seconds)  */  //  像素时钟//下面这六个就是LCD的时序参数__u32 left_margin;        /* time from sync to picture   */__u32 right_margin;        /* time from picture to sync  */__u32 upper_margin;        /* time from sync to picture  */__u32 lower_margin;__u32 hsync_len;        /* length of horizontal sync     */__u32 vsync_len;        /* length of vertical sync       */__u32 sync;            /* see FB_SYNC_*        */__u32 vmode;            /* see FB_VMODE_*        */__u32 rotate;            /* angle we rotate counter clockwise */__u32 reserved[5];        /* Reserved for future compatibility */
};struct fb_var_screeninfo

(3)struct fb_fix_screeninfo

Linux内核中使用struct fb_fix_screeninfo来描述不可修改的LCD属性参数,如显示内存的物理地址和长度等。

struct fb_fix_screeninfo 
{char id[16];                /* identification string eg "TT Builtin" */unsigned long smem_start;   // LCD显存的起始地址(物理地址)/* (physical address) */__u32 smem_len;            /* Length of frame buffer mem *///  LCD显存的字节大小__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    *///  LCD一行的长度 (以字节为单位)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 */
};struct fb_fix_screeninfo

3. fbmem.c

3.1 入口函数

        FrameBuffer驱动是以模块的形式注册到系统中,在模块初始化时,创建FrameBuffer对应的设备文件及proc文件,并注册FrameBuffer设备操作接口函数fb_fops。

static int __init
fbmem_init(void)
{proc_create("fb", 0, NULL, &fb_proc_fops);    //向 proc 文件系统报告驱动状态和参数if (register_chrdev(FB_MAJOR,"fb",&fb_fops))    //注册字符设备驱动,主设备号是29 名称为fbprintk("unable to get major %d for fb devs\n", FB_MAJOR);fb_class = class_create(THIS_MODULE, "graphics");    //创建 /sys/class/graphics 设备类,配合 mdev生成设备文件if (IS_ERR(fb_class)) {printk(KERN_WARNING "Unable to create fb class; errno = %ld\n", PTR_ERR(fb_class));fb_class = NULL;}return 0;
}

3.2 fb_fops

        在linux设备驱动中,所有的显示缓存设备均由framebuffer子系统内部管理,即linux设备驱动框架只认识一个主设备号为29的framebuffer设备。应用层所有针对显示缓存(最多32个)的访问均会推送给fb_fops进行进一步分发操作。

static const struct file_operations fb_fops = {.owner =	THIS_MODULE,.read =		fb_read,.write =	fb_write,.unlocked_ioctl = fb_ioctl,
#ifdef CONFIG_COMPAT.compat_ioctl = fb_compat_ioctl,
#endif.mmap =		fb_mmap,.open =		fb_open,.release =	fb_release,
#ifdef HAVE_ARCH_FB_UNMAPPED_AREA.get_unmapped_area = get_fb_unmapped_area,
#endif
#ifdef CONFIG_FB_DEFERRED_IO.fsync =	fb_deferred_io_fsync,
#endif.llseek =	default_llseek,
};

3.2.1 fb_open

      

3.2.2 fb_mmap

3.2.3 fb_ioctl 

static long do_fb_ioctl(struct fb_info *info, unsigned int cmd,unsigned long arg)
{struct fb_ops *fb;struct fb_var_screeninfo var;struct fb_fix_screeninfo fix;struct fb_con2fbmap con2fb;struct fb_cmap cmap_from;struct fb_cmap_user cmap;struct fb_event event;void __user *argp = (void __user *)arg;long ret = 0;switch (cmd) {case FBIOGET_VSCREENINFO:        // 获取可变屏幕参数if (!lock_fb_info(info))return -ENODEV;var = info->var;unlock_fb_info(info);ret = copy_to_user(argp, &var, sizeof(var)) ? -EFAULT : 0;break;case FBIOPUT_VSCREENINFO:        // 设置可变屏幕参数if (copy_from_user(&var, argp, sizeof(var)))return -EFAULT;console_lock();if (!lock_fb_info(info)) {console_unlock();return -ENODEV;}info->flags |= FBINFO_MISC_USEREVENT;ret = fb_set_var(info, &var);info->flags &= ~FBINFO_MISC_USEREVENT;unlock_fb_info(info);console_unlock();if (!ret && copy_to_user(argp, &var, sizeof(var)))ret = -EFAULT;break;case FBIOGET_FSCREENINFO:        //获得固定的屏幕参数设置if (!lock_fb_info(info))return -ENODEV;fix = info->fix;unlock_fb_info(info);ret = copy_to_user(argp, &fix, sizeof(fix)) ? -EFAULT : 0;break;case FBIOPUTCMAP:                // 获得固定的屏幕参数设置if (copy_from_user(&cmap, argp, sizeof(cmap)))return -EFAULT;ret = fb_set_user_cmap(&cmap, info);break;case FBIOGETCMAP:                // 获得颜色表if (copy_from_user(&cmap, argp, sizeof(cmap)))return -EFAULT;if (!lock_fb_info(info))return -ENODEV;cmap_from = info->cmap;unlock_fb_info(info);ret = fb_cmap_to_user(&cmap_from, &cmap);break;case FBIOPAN_DISPLAY:            // 平移显示if (copy_from_user(&var, argp, sizeof(var)))return -EFAULT;console_lock();if (!lock_fb_info(info)) {console_unlock();return -ENODEV;}ret = fb_pan_display(info, &var);unlock_fb_info(info);console_unlock();if (ret == 0 && copy_to_user(argp, &var, sizeof(var)))return -EFAULT;break;case FBIO_CURSOR:ret = -EINVAL;break;case FBIOGET_CON2FBMAP:if (copy_from_user(&con2fb, argp, sizeof(con2fb)))return -EFAULT;if (con2fb.console < 1 || con2fb.console > MAX_NR_CONSOLES)return -EINVAL;con2fb.framebuffer = -1;event.data = &con2fb;if (!lock_fb_info(info))return -ENODEV;event.info = info;fb_notifier_call_chain(FB_EVENT_GET_CONSOLE_MAP, &event);unlock_fb_info(info);ret = copy_to_user(argp, &con2fb, sizeof(con2fb)) ? -EFAULT : 0;break;case FBIOPUT_CON2FBMAP:if (copy_from_user(&con2fb, argp, sizeof(con2fb)))return -EFAULT;if (con2fb.console < 1 || con2fb.console > MAX_NR_CONSOLES)return -EINVAL;if (con2fb.framebuffer >= FB_MAX)return -EINVAL;if (!registered_fb[con2fb.framebuffer])request_module("fb%d", con2fb.framebuffer);if (!registered_fb[con2fb.framebuffer]) {ret = -EINVAL;break;}event.data = &con2fb;console_lock();if (!lock_fb_info(info)) {console_unlock();return -ENODEV;}event.info = info;ret = fb_notifier_call_chain(FB_EVENT_SET_CONSOLE_MAP, &event);unlock_fb_info(info);console_unlock();break;case FBIOBLANK:console_lock();if (!lock_fb_info(info)) {console_unlock();return -ENODEV;}info->flags |= FBINFO_MISC_USEREVENT;ret = fb_blank(info, arg);info->flags &= ~FBINFO_MISC_USEREVENT;unlock_fb_info(info);console_unlock();break;default:if (!lock_fb_info(info))return -ENODEV;fb = info->fbops;if (fb->fb_ioctl)ret = fb->fb_ioctl(info, cmd, arg);elseret = -ENOTTY;unlock_fb_info(info);}return ret;
}

4. mxsfb.c

        一个标准的platform平台设备驱动

4.1 mxsfb_probe

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.xdnf.cn/news/1485980.html

如若内容造成侵权/违法违规/事实不符,请联系一条长河网进行投诉反馈,一经查实,立即删除!

相关文章

[路由器]IP-MAC的绑定与取消

背景&#xff1a;当公司的网络不想与外部人员进行共享&#xff0c;可以在路由器页面配置IP-MAC的绑定&#xff0c;让公司内部人员的手机和电脑的mac&#xff0c;才能接入到公司。第一步&#xff1a;在ARP防护中&#xff0c;启动IP-MAC绑定选项&#xff0c;必须启动仅允许IP-MAC…

Modbus转BACnet/IP网关快速对接Modbus协议设备与BA系统

摘要 在智能建筑和工业自动化领域&#xff0c;Modbus和BACnet/IP协议的集成应用越来越普遍。BA&#xff08;Building Automation&#xff0c;楼宇自动化&#xff09;系统作为现代建筑的核心&#xff0c;需要高效地处理来自不同协议的设备数据&#xff0c;负责监控和管理建筑内…

双边性:构建神经网络的新方法

正如承诺的那样&#xff0c;这是最近我遇到的最有趣的想法之一的第二部分。如果你错过了&#xff0c;请务必观看本系列的第一部分 - 神经科学家对改进神经网络的看法 - 我们讨论了双边性的生物学基础以及我们大脑的不对称性质如何带来更高的性能。 在这篇文章中&#xff0c;我…

吴恩达深度学习笔记1 Neural Networks and Deep Learning

参考视频&#xff1a;(超爽中英!) 2024公认最好的【吴恩达深度学习】教程&#xff01;附课件代码 Professionalization of Deep Learning_哔哩哔哩_bilibili Neural Networks and Deep Learning 1. 深度学习引言(Introduction to Deep Learning) 2. 神 经 网 络 的 编 程 基 础…

【RT摩拳擦掌】RT600 4路音频同步输入1路TDM输出方案

【RT摩拳擦掌】RT600 4路音频同步输入1路TDM输出方案 一&#xff0c; 文章简介二&#xff0c;硬件平台构建2.1 音频源板2.2 音频收发板2.3 双板硬件连接 三&#xff0c;软件方案与软件实现3.1 方案实现3.2 软件代码实现3.2.1 4路I2S接收3.2.2 I2S DMA pingpong配置3.2.3 音频数…

【BUG】已解决:AttributeError: ‘str‘ object has no attribute ‘read‘

AttributeError: ‘str‘ object has no attribute ‘read‘ 目录 AttributeError: ‘str‘ object has no attribute ‘read‘ 【常见模块错误】 【解决方案】 欢迎来到英杰社区https://bbs.csdn.net/topics/617804998https://bbs.csdn.net/topics/617804998 欢迎来到我的主…

【CSS】容器查询:@container

目录 什么是容器查询如何使用container实际应用场景浏览器支持 什么是容器查询 容器查询是一种CSS特性&#xff0c;允许开发者根据组件所在的容器的大小来应用样式&#xff0c;而不是整个视口的大小。这使得组件能够更加灵活地适应不同的布局环境&#xff0c;而不仅仅是依赖于…

IDEA缓存和索引

IDEA缓存和索引 —2020年06月10日 IntelliJ IDEA首次加载项目的时候。都会创建索引&#xff0c;而创建索引的时间根项目的文件多少成正比。 IntelliJ IDEA的缓存和索引主要是用来加快文件查询&#xff0c;从而加快各种查找、代码提示等操作的速度。 某些特殊情况下&#xf…

2024“钉耙编程”中国大学生算法设计超级联赛(2)

Rank Search Result (hdu.edu.cn) URL划分 - HDU 7451 - Virtual Judge (vjudge.net)​​​​​​​ 这题唯一要注意的就是后面只输出有等号的部分&#xff0c;然后模拟即可。 #include<bits/stdc.h> using lllong long; using ullunsigned long long; using PIIstd::…

《梦醒蝶飞:释放Excel函数与公式的力量》18.1 图表类型与设计

第18章&#xff1a;创建图表和数据可视化 18.1 图表类型与设计 Excel提供了多种图表类型&#xff0c;帮助用户以直观的方式展示数据。选择合适的图表类型和设计可以显著提高数据的可读性和理解度。以下将介绍常见的图表类型及其应用&#xff0c;并通过具体案例进行说明。 18.…

ML.Net 学习之使用经过训练的模型进行预测

什么是ML.Net&#xff1a;&#xff08;学习文档上摘的一段&#xff1a;ML.NET 文档 - 教程和 API 参考 | Microsoft Learn 【学习入口】&#xff09; 它使你能够在联机或脱机场景中将机器学习添加到 .NET 应用程序中。 借助此功能&#xff0c;可以使用应用程序的可用数据进行自…

【中项】系统集成项目管理工程师-第4章 信息系统架构-4.5技术架构

前言&#xff1a;系统集成项目管理工程师专业&#xff0c;现分享一些教材知识点。觉得文章还不错的喜欢点赞收藏的同时帮忙点点关注。 软考同样是国家人社部和工信部组织的国家级考试&#xff0c;全称为“全国计算机与软件专业技术资格&#xff08;水平&#xff09;考试”&…

【人工智能】使用Python的dlib库实现人脸识别技术

&#x1f525; 个人主页&#xff1a;空白诗 文章目录 一、引言二、传统人脸识别技术1. 基于几何特征的方法2. 基于模板匹配的方法3. 基于统计学习的方法 三、深度学习在脸识别中的应用1. 卷积神经网络&#xff08;CNN&#xff09;2. FaceNet和ArcFace 四、使用Python和dlib库实…

微服务:网关

网关 网关,即网络的关口,当一个网络传输到另一个网络时就需要经过网关来实现 数据的路由和转发 以及 数据安全的校验 网关技术实现 SpringCloudGateWay: 基于Spring的WebFlux技术,完全支持响应式编程,吞吐能力更强 SpringCloudGateWay 依赖 <!--网关--><depe…

Linux中的时间函数

参考&#xff1a; 几种取时间的方法&#xff08;附代码&#xff09; Linux中gmtime和localtime的区别(time_t格式转换为tm格式) C 库函数 - time() mktime和localtime_r能在多线程环境下使用么&#xff1f; Linux_C环境编程&#xff1a;时间日期函数总结 细说时间测量RDT…

乐鑫AWS IoT ExpressLink方案,简化物联网设备连接AWS IoT服务

在现代科技迅速发展的今天&#xff0c;物联网&#xff08;IoT&#xff09;已经成为连接物理世界与数字世界的重要桥梁&#xff0c;越来越多的设备开始接入网络&#xff0c;实现智能化控制。 在这个大背景下&#xff0c;乐鑫携手亚马逊&#xff0c;推出了AWS IoT ExpressLink方…

免费视频批量横转竖工具

简介 视频处理器 v1.3 是一款由是貔貅呀开发的视频编辑和处理工具&#xff0c;提供高效便捷的视频批量横转竖&#xff0c;主要功能&#xff1a; 导入与删除文件&#xff1a;轻松导入多个视频文件&#xff0c;删除不必要的文件。暂停与继续处理&#xff1a;随时暂停和继续处理。…

开源AI智能名片S2B2C商城小程序在社群团购模式中的应用与探索

摘要 本文深入探讨了开源AI智能名片S2B2C商城小程序在社群团购模式中的创新应用与未来发展。通过详细分析社群团购模式的特征、发展趋势及其面临的挑战&#xff0c;结合开源AI智能名片S2B2C商城小程序的技术优势与实际应用案例&#xff0c;本文提出了一系列旨在提升社群团购效…

centos中zabbix安装、卸载及遇到的问题

目录 Zabbix简介Zabbix5.0和Zabbix7.0的区别监控能力方面模板和 API 方面性能、速度方面 centos7安装Zabbix(5.0)安装zabbix遇到的问题卸载Zabbix Zabbix简介 Zabbix 是一个基于 WEB 界面的提供分布式系统监视以及网络监视功能的企业级的开源解决方案。zabbix 能监视各种网络参…

Android ContentResolver.loadThumbnail转Kotlin

Android ContentResolver.loadThumbnail转Kotlin loadThumbnail原先是Java实现的&#xff0c;现在抠出来转Kotlin实现。 private fun loadThumbnail(uri: Uri, size: Size, signal: CancellationSignal): Bitmap {return myLoadThumbnail(mContext?.contentResolver!!, uri, s…