【嵌入式linux驱动开发】第十节 LCD 背光驱动代码架构分析(1)

                                                                                                                  

                                                           【嵌入式linux驱动开发】第十节LCD背光驱动代码架构分析(1)

                                                                                                    

   【 创 科 之 龙: 嵌入式开发经典系列教程】

安装环境:ubuntu 10.02

学习目标:主要教你理解mini2440LCD背光驱动代码架构分析

作者:aiku

QQ:1653687969

技术解答群:234945702(初学者可以加入一起学习交流,不懂都在群上解答)

时间:2015111

一、本节内容:

1.本节主要是用友善之臂mini2440为实验平台, 分析LCD背光驱动代码架构分析。

4.Aiku真心送你一些话:

当你的才华还撑不起你的野心的时候,那么你就应该静下心来好好学习;

当你的能力还驾驭不了你的目标时候,那么你就应该沉下心来,好好历练,历练;

学好嵌入式,是我们每个初学者的共同目标;拿到高薪,是我们初学者最终共同梦想。

梦想,不是浮躁,而是沉淀和积累,只有拼出来的美丽,没有等出来的辉煌,机会永远是留给最渴望的那个人。我们要常常问问自己,想要怎样的人生,静心学习,耐心沉淀。那么接下来我们就进入linux世界里沉淀和历练吧!

 

二、硬件原理:

我们知道任何一个牛逼的嵌入式工程师,他都离不开硬件,如果你不懂硬件,那么尽管你软件再牛逼,都是无济于事。

1.LCD 背光控制原理

在 mini2440/micro2440 开发板中,LCD 背光是通过 CPU 的LCD_PWR 引脚来控制的,从原理图中可以看出,它对应于 GPG4 。

当 LCD_PWR 输出为高电平"1"时,将打开背光;

当输出为低电平"0"时,将关闭背光

echo 0 > /dev/backlight  (关闭)

echo 1 > /dev/backlight    (打开)


那么硬件是如何实现背光开和关?

首先,我看一下,T35屏的数据手册("ZQ3506"是LQ035Q1DG06的兼容国产屏,它们的Datasheet是可以通用的。)


(注意:我们只是打开和关闭背光,而并没有背光亮度的调节作用)。如果要调节亮度应该如何写代码?


三、驱动架构分析:

现在,我们需要增加一个简单的背光驱动,以便能够通过软件便可简单的控制背光的开关。我们要达到的目的是:在命令终端通过向背光设备发送偶数比如"0"便可关闭背光,发送奇数比如"1"便可打开背光,这样使用起来就方便多了,而不需要专门的应用程序控制它,设备节点:LCD 背光设备文件:/dev/backlight

在命令行种输入:echo 0 > /dev/backlight 可以关闭 LCD 背光。

在命令行种输入:echo 1 > /dev/backlight 可以打开 LCD 背光。






为了实现这点,我们在 linux-2.6.32.2/drivers/video目录增加一个 mini2440_backlight.c

#include <linux/errno.h>
#include <linux/kernel.h>
#include <linux/module.h>
#include <linux/slab.h>
#include <linux/input.h>
#include <linux/init.h>
#include <linux/serio.h>
#include <linux/delay.h>
#include <linux/clk.h>
#include <linux/miscdevice.h>
#include <linux/gpio.h>


#include <asm/io.h>
#include <asm/irq.h>
#include <asm/uaccess.h>
#include <mach/regs-clock.h>
#include <plat/regs-timer.h>
 
#include <mach/regs-gpio.h>
#include <linux/cdev.h>


#undef DEBUG
//#define DEBUG
#ifdef DEBUG
#define DPRINTK(x...) {printk(__FUNCTION__"(%d): ",__LINE__);printk(##x);}
#else
#define DPRINTK(x...) (void)(0)
#endif
//定义设备名为backlight
#define DEVICE_NAME "backlight"


//定义背光变量bl_state,以记录背光的开关状态
static unsigned int bl_state;
//设置背光开关的函数主要是翻转背光变量bl_state
static inline void set_bl(int state)
{
bl_state = !!state;//翻转背光变量bl_state
//把结果写入背光所用的寄存器GPG(4),
s3c2410_gpio_setpin(S3C2410_GPG(4), bl_state);
}
//获取背光开关的函数
static inline unsigned int get_bl(void)
{
return bl_state;
}
//从应用程序读取参数并传递到内核中
static ssize_t dev_write(struct file *file, const char *buffer, size_t count, loff_t * ppos)
{
unsigned char ch;
int ret;
if (count == 0) {
return count;//字节数
}
//从用户层读取参数
ret = copy_from_user(&ch, buffer, sizeof ch) ? -EFAULT : 0;
if (ret) {
return ret;
}
    //判断是奇数还是偶数
ch &= 0x01;
//设置背光状态
set_bl(ch);

return count;
}
//把内核参数传递到用户层
static ssize_t dev_read(struct file *filp, char *buffer, size_t count, loff_t *ppos)
{
int ret;
unsigned char str[] = {'0', '1' };


if (count == 0) {
return 0;
}
//从内核中读数据
ret = copy_to_user(buffer, str + get_bl(), sizeof(unsigned char) ) ? -EFAULT : 0;
if (ret) {
return ret;
}


return sizeof(unsigned char);
}


static struct file_operations dev_fops = {
owner: THIS_MODULE,
read: dev_read,
write: dev_write,
};


static struct miscdevice misc = {
.minor = MISC_DYNAMIC_MINOR,
.name = DEVICE_NAME,
.fops = &dev_fops,
};


static int __init dev_init(void)
{
int ret;


ret = misc_register(&misc);


printk (DEVICE_NAME"\tinitialized\n");
//配置LCD背光引脚状态
s3c2410_gpio_cfgpin(S3C2410_GPG(4), S3C2410_GPIO_OUTPUT);
set_bl(1);
return ret;
}




static void __exit dev_exit(void)
{
misc_deregister(&misc);
}


module_init(dev_init);
module_exit(dev_exit);
MODULE_LICENSE("GPL");
MODULE_AUTHOR("FriendlyARM Inc.");




猜你喜欢

转载自blog.csdn.net/aiku969/article/details/42834467