LCD驱动分析(一)字符设备驱动框架分析

参考:S3C2440 LCD驱动(FrameBuffer)实例开发<一>

     S3C2440 LCD驱动(FrameBuffer)实例开发<二>

LCD驱动也是字符设备驱动,也遵循字符设备驱动的流程:

a. 分配主设备号

b. 构建file_operations结构体中的open,write,read...等函数

c. 调用register_chrdev()函数注册字符设备

d. 调用class_register()注册类

e. 调用device_create()创建设备,linux会在sysfs目录下自动创建字符设备。

以上的步骤同样适用于分析输入子系统,只不过上面的各个步骤可能分散在不同的文件与函数中完成。

1.linux/drivers/video/fbmem.c中的fbmem_init()函数完成a,b,c,d完成以上四步。

 1 static const struct file_operations fb_fops = {
 2     .owner =    THIS_MODULE,
 3     .read =        fb_read,
 4     .write =    fb_write,
 5     .unlocked_ioctl = fb_ioctl,
 6     .mmap =        fb_mmap,
 7     .open =        fb_open,
 8     .release =    fb_release,
 9     .llseek =    default_llseek,
10 };
11 static int __init fbmem_init(void)
12 {
13     proc_create("fb", 0, NULL, &fb_proc_fops);
14 
15     if (register_chrdev(FB_MAJOR,"fb",&fb_fops))
16         printk("unable to get major %d for fb devs
", FB_MAJOR);
17 
18     fb_class = class_create(THIS_MODULE, "graphics");
19     if (IS_ERR(fb_class)) {
20         printk(KERN_WARNING "Unable to create fb class; errno = %ld
", PTR_ERR(fb_class));
21         fb_class = NULL;
22     }
23     return 0;
24 }

2. 那么谁来调用device_create()创建设备?这里直接给出结论,后面再分析过程。

fbmem.c中定义的fb_info结构体的全局指针数组,fb_info结构体是LCD驱动的关键。register_framebuffer()函数中会调用device_create()函数创建设备节点。

struct fb_info *registered_fb[FB_MAX] __read_mostly; //FB_MAX=32

int register_framebuffer(struct fb_info *fb_info)

  -->ret = do_register_framebuffer(fb_info);

    -->找到空的registered_fb[i]

    -->fb_info->dev = device_create(fb_class, fb_info->device,       MKDEV(FB_MAJOR, i), NULL, "fb%d", i);

    -->设置fb_info

    -->registered_fb[i] = fb_info;

3. 如何打开设备?

fb_open(struct inode *inode, struct file *file)

  -->int fbidx = iminor(inode);

  -->struct fb_info *info=get_fb_info(fbidx);//根据此设备号找到fb_info结构体

  -->fb_info = registered_fb[fbidx];

  --> file->private_data = info;

  --> if (info->fbops->fb_open) res = info->fbops->fb_open(info,1);//如果结构体中定义了open函数,执行新的open函数

4.如何读取设备?这里的关键是info->screen_base,设置为LCD控制器帧缓冲区的起始地址。

fb_read(struct file *file, char __user *buf, size_t count, loff_t *ppos)

  -->struct fb_info *info = file_fb_info(file);//找到打开的fb_info结构体

  --> if (info->fbops->fb_read)//如果fb_info中定义的读函数,则执行新的读函数

    return info->fbops->fb_read(info, buf, count, ppos);

  //没有定义新的读函数则执行下面

  -->src = (u8 __iomem *) (info->screen_base + p);//设置源地址

  -->buffer = kmalloc((count > PAGE_SIZE) ? PAGE_SIZE : count, GFP_KERNEL);//设置目的地址

  -->dst = buffer;

  -->fb_memcpy_fromfb(dst, src, c);//读到用户空间

  -->copy_to_user(buf, buffer, c)

原文地址:https://www.cnblogs.com/yangjiguang/p/6080116.html