驱动代码:

/*************************************************************************
> File Name: lcd.c
> Author:
> Mail:
> Created Time: 2016年11月02日 星期三 15时21分59秒
************************************************************************/
#include <linux/module.h>
#include <linux/kernel.h>
#include <linux/errno.h>
#include <linux/string.h>
#include <linux/mm.h>
#include <linux/slab.h>
#include <linux/delay.h>
#include <linux/fb.h>
#include <linux/init.h>
#include <linux/dma-mapping.h>
#include <linux/interrupt.h>
#include <linux/workqueue.h>
#include <linux/wait.h>
#include <linux/platform_device.h>
#include <linux/clk.h> #include <asm/io.h>
#include <asm/uaccess.h>
#include <asm/div64.h> #include <asm/mach/map.h>
#include <asm/arch/regs-lcd.h>
#include <asm/arch/regs-gpio.h>
#include <asm/arch/fb.h> static volatile unsigned long *gpbcon;
static volatile unsigned long *gpbdat; static volatile unsigned long *gpccon;
static volatile unsigned long *gpdcon;
static volatile unsigned long *gpgcon; struct lcd_regs {
unsigned long lcdcon1;
unsigned long lcdcon2;
unsigned long lcdcon3;
unsigned long lcdcon4;
unsigned long lcdcon5;
unsigned long lcdsaddr1;
unsigned long lcdsaddr2;
unsigned long lcdsaddr3;
unsigned long redlut;
unsigned long greenlut;
unsigned long bluelut;
unsigned long reserved[];
unsigned long dithmode;
unsigned long tpal;
unsigned long lcdintpnd;
unsigned long lcdsrcpnd;
unsigned long lcdintmsk;
unsigned long lpcsel;
}; static int s3c_lcdfb_setcolreg( unsigned int regno, unsigned int red, unsigned int green, unsigned int blue, unsigned int transp, struct fb_info *info );
static u32 pseudo_palette[];
static volatile struct lcd_regs *lcd_reg; static struct fb_info *s3c_fb; static struct fb_ops s3c_fb_ops = {
.owner= THIS_MODULE,
.fb_setcolreg= s3c_lcdfb_setcolreg,
.fb_fillrect= cfb_fillrect, /* Needed !!! */
.fb_copyarea= cfb_copyarea,/* Needed !!! */
.fb_imageblit= cfb_imageblit,/* Needed !!! */
}; static unsigned int chan_to_field(unsigned int chan, const struct fb_bitfield *bf)
{
chan &= 0xffff;
chan >>= - bf->length;
return chan << bf->offset;
} /*调色板*/
static int s3c_lcdfb_setcolreg( unsigned int regno, unsigned int red, unsigned int green, unsigned int blue, unsigned int transp, struct fb_info *info )
{
unsigned int val;
if(regno > )
return ;
/* 用红绿蓝构造出调色板 */
val = chan_to_field( red, &info->var.red );
val |= chan_to_field( green, &info->var.green );
val |= chan_to_field( blue, &info->var.blue );
((u32 *)(info->pseudo_palette))[regno] = val;
return ;
} static int lcd_init( void )
{
/* 分配一个fb_info结构体 */
s3c_fb = framebuffer_alloc(,NULL); //分配的空间为fb_info + 0 (0为私有空间) /* 设置 */
/* 设置固定的参数 */
strcpy( s3c_fb->fix.id, "mylcd" );
s3c_fb->fix.smem_len = **/;
s3c_fb->fix.type = FB_TYPE_PACKED_PIXELS;
s3c_fb->fix.visual = FB_VISUAL_TRUECOLOR; /* TFT*/
s3c_fb->fix.line_length = *; /* 设置可变的参数 */
s3c_fb->var.xres = ;
s3c_fb->var.yres = ;
s3c_fb->var.xres_virtual = ;
s3c_fb->var.yres_virtual = ;
s3c_fb->var.xres_virtual = ;
s3c_fb->var.yres_virtual = ;
s3c_fb->var.bits_per_pixel = ; s3c_fb->var.red.offset = ;
s3c_fb->var.red.length = ; s3c_fb->var.green.offset = ;
s3c_fb->var.green.length = ; s3c_fb->var.blue.offset = ;
s3c_fb->var.blue.length = ; s3c_fb->var.activate = FB_ACTIVATE_NOW; /* 设置操作函数 */
s3c_fb->fbops = &s3c_fb_ops; /* 其他设置 */
s3c_fb->pseudo_palette = pseudo_palette;
//s3c_fb->screen_base = ; //显存的虚拟地址
s3c_fb->screen_size = **; //显存的大小 /* 配置GPIO */
gpbcon = ioremap(0x56000010,);
gpbdat = gpbcon + ;
gpccon = ioremap(0x56000020,);
gpdcon = ioremap(0x56000030,);
gpgcon = ioremap(0x56000060,); *gpccon = 0xaaaaaaaa;
*gpdcon = 0xaaaaaaaa;
*gpbcon &= ~();
*gpbcon |= ;
*gpbdat &= ~; //关闭背光
*gpgcon |= (<<); /* 根据lcd手册 配置lcd控制器 */
lcd_reg = ioremap(0x4d000000, sizeof(struct lcd_regs));
/*
* VCLK = HCLK / ((CLKVAL+1)*2)
*
* */
lcd_reg->lcdcon1 = (<<)|(<<)|(0x0c<<);
/*
* 垂直方向的时间参数
* VBPD 是VSYNC之后再过多长时间才能发出第一行数据
* LINEVAL 多少行数据 320
* VFPD 发出最后一行数据之后过多久才发出VSYNC信号
* VSPW VSYNC信号的脉冲宽度
* */
lcd_reg->lcdcon2 = (<<)|(<<)|(<<)|(); /*
* 水平方向的时间参数
* HBPD 是HSYNC之后再过多长时间才能发出第一个像素数据
* HOZVAL 有多少列
* HFPD 发出一行的最后一个数据后再过多长时间发出HSYNC信号
* HSPW HSYNC信号的脉冲宽度
* */
lcd_reg->lcdcon3 = (<<) | (<<) | (<<);
lcd_reg->lcdcon4 = ; /*
* 信号的极性
* bits[11] format for 565
* bits[10] 上升沿有效
* bits[9] HSYNC信号反转
* bits[8] VSYNC信号反转
* bits[7] 数据
* bits[6] VDEN
* bits[3] PWREN
* */
lcd_reg->lcdcon5 = (<<) | (<<) | (<<) | (<<); /* 分配显存 */
s3c_fb->screen_base = dma_alloc_writecombine(NULL, s3c_fb->fix.smem_len, &s3c_fb->fix.smem_start,GFP_KERNEL );
lcd_reg->lcdsaddr1 = (s3c_fb->fix.smem_start >> ) & ~(<<);
lcd_reg->lcdsaddr2 = ((s3c_fb->fix.smem_start + s3c_fb->fix.smem_len)>>) & 0x1fffff;
lcd_reg->lcdsaddr3 = (*/); /* 1行的长度 单位是2字节*/ /**/
lcd_reg->lcdcon1 |= ; /*使能LCD控制器*/
lcd_reg->lcdcon5 |= (<<); /* 使能LCD */
*gpbdat |= ; /*使能背光*/
/* 注册 */
register_framebuffer(s3c_fb); return ;
} static void lcd_exit( void )
{
unregister_framebuffer(s3c_fb);
lcd_reg->lcdcon1 &= ~;
*gpbdat &= ~;
dma_free_writecombine(NULL,s3c_fb->fix.smem_len,s3c_fb->screen_base,s3c_fb->fix.smem_start);
iounmap(lcd_reg);
iounmap(gpbcon);
iounmap(gpccon);
iounmap(gpdcon);
iounmap(gpgcon);
framebuffer_release(s3c_fb);
} module_init(lcd_init);
module_exit(lcd_exit);
MODULE_LICENSE("GPL");

注:这是240*320寸的屏幕。

需要在配置内核的时候不要把lcd驱动编译到内核中。

在安装驱动时会提示某些函数找不到。需要安装cfg*.ko等驱动模块。

也可以将触摸屏和按键做为控制终端。需要在 /etc/inittab中添加

tty1::askfirst:-/bin/sh
即可!

sd

嵌入式Linux驱动学习之路(十八)LCD驱动的更多相关文章

  1. 嵌入式Linux驱动学习之路(十五)按键驱动-定时器防抖

    在之前的定时器驱动程序中,我们发现在连续按下按键的时候,正常情况下应该是一次按下对应一次松开.而程序有时候会显示是两次按下,一次松开.这个问题是因为在按下的时候,因为是机械按键,所以电压信号会产生一定 ...

  2. 嵌入式Linux驱动学习之路(十四)按键驱动-同步、互斥、阻塞

    目的:同一个时刻,只能有一个应用程序打开我们的驱动程序. ①原子操作: v = ATOMIC_INIT( i )  定义原子变量v并初始化为i atomic_read(v)        返回原子变量 ...

  3. 嵌入式Linux驱动学习之路(十二)按键驱动-poll机制

    实现的功能是在读取按键信息的时候,如果没有产生按键,则程序休眠在read函数中,利用poll机制,可以在没有退出的情况下让程序自动退出. 下面的程序就是在读取按键信息的时候,如果5000ms内没有按键 ...

  4. 嵌入式Linux驱动学习之路(十)字符设备驱动-my_led

    首先贴上代码: 字符设备驱动代码: /** *file name: led.c */#include <linux/sched.h> #include <linux/signal.h ...

  5. 嵌入式Linux驱动学习之路(十九)触摸屏驱动、tslib测试

    触摸屏使用流程: 1. 按下产生中断. 2.在中断处理程序中启动AD转换XY坐标. 3.AD转换结束并产生AD中断. 4. 在AD的中断处理函数中上报信息,启动定时器. 5. 定时器时间到后进入中断, ...

  6. 嵌入式Linux驱动学习之路(十六)输入子系统

    以前写的一些输入设备的驱动都是采用字符设备处理的.问题由此而来,Linux开源社区的大神们看到了这大量输入设备如此分散不堪,有木有可以实现一种机制,可以对分散的.不同类别的输入设备进行统一的驱动,所以 ...

  7. 嵌入式linux的学习之路[转]

    我认为的一条学习嵌入式Linux的路: 1)学习 Linux系统安装. 常用命令.应用程序安装. 2) 学习 Linux 下的 C 编程.这本书必学<UNIX 环境高级编程>.<UN ...

  8. IOS学习之路十八(通过 NSURLConnection 发送 HTTP 各种请求)

    你想通过 Http 协议向服务器发送一个 Get 的包装请求,并在这个请求中添加了一些请 求参数. 向远程服务器发送一个 GET 请求,然后解析返回的数据.通常一个 GET 请求是添加了 一些参数的, ...

  9. 嵌入式Linux设备驱动程序:在运行时读取驱动程序状态

    嵌入式Linux设备驱动程序:在运行时读取驱动程序状态 Embedded Linux device drivers: Reading driver state at runtime 在运行时了解驱动程 ...

随机推荐

  1. Java并发编程:线程池的使用

    Java并发编程:线程池的使用 在前面的文章中,我们使用线程的时候就去创建一个线程,这样实现起来非常简便,但是就会有一个问题: 如果并发的线程数量很多,并且每个线程都是执行一个时间很短的任务就结束了, ...

  2. Cats(2)- Free语法组合,Coproduct-ADT composition

    上篇我们介绍了Free类型可以作为一种嵌入式编程语言DSL在函数式编程中对某种特定功能需求进行描述.一个完整的应用可能会涉及多样的关联功能,但如果我们为每个应用都设计一套DSL的话,那么在我们的函数式 ...

  3. PHP多维数组根据其中一个字段的值排序

    平时简单的一维数组或者简单的数组排序这里就不多作介绍,这里主要是针对平时做项目中的可能遇到的情况,根据多维数组中的其中一个排序.用到的php函数是:array_multisort. 思路:获取其中你需 ...

  4. 序列化类型 System.Data.Entity.DynamicProxies 的对象时检测到循环引用

    学习 EF Code First+MVC 时遇到了在请求JsonResult时出现 序列化类型 System.Data.Entity.DynamicProxies 的对象时检测到循环引用 的异常,原因 ...

  5. js正则表达式图形化工具-rline

    github地址:https://github.com/finance-sh/rline 在线demo: http://lihuazhai.com/demo/test.html 这是一个js正则表达式 ...

  6. xml与datatable类型互换

    //已测 private DataTable ConvertXMLToDataSet(string xmlData) { StringReader stream = null; XmlTextRead ...

  7. Windows8.1下安装NoSQL-- mongodb安装使用

    1. 官方下载monodb:http://www.mongodb.org/downloads 现在最新版本3.0 2. 以下载Windows 64-bit为例官方最新版的没有分开, 32位和64位是应 ...

  8. Xcode8开发iOS10推送通知过程

    iOS10发布后,简书优先开发增加了iOS10的新通知.本文分享整个feature的开发过程遇到的问题. 1.工程配置 Xcode8发生了很大的变化,直接打开原来的工程编译运行,这个时候是获取不到Pu ...

  9. iOS 10 都有什么改变?

    iOS 10 都有什么改变? 看这一个贴就够了 最全面的试用 苹果在 WWDC 2016 发布会上正式发布了 iOS 10 操作系统,iOS 与 macOS.tvOS 和 watchOS 构建了苹果四 ...

  10. 2016年4月21百度iOS实习生在线笔试题&编程题

    1.一个人上台阶可以一次上1个,2个,或者3个,问这个人上32层的台阶,总共有几种走法? 思路:先建立数学模型,设3步的走 i 次,2步的走 j 次, 1步的走 k 次,上了3*i + 2*j + 1 ...