十一、ATMEGA16驱动12864液晶(驱动器为HD61202)
十一—(01)、今天只是个简单的动态的驱动~~(以后还会更新的~~移位什么的~~毕竟这些显示屏还是挺重要的~~而且更重要的是赶紧买板子~~)
先上图片~~
这个是用ATMEGA16仿真的(我真的很疼~~对比一下下面51仿真的,数据被压缩,完全不能理解~~数据完全一样~~)
然后是程序:
IAR11_1.c
//------------------------------------------------------------------------------
//main
#include"ioavr.h"
#include"intrinsics.h"
#include"Lcd12864Init.h"
#include"MacroAndConst.h"
#include"ZiMo.h"
#include"Lcd12864Display.h"
//------------------------------------------------------------------------------
//端口初始化函数
void port_init()
{
DDRC=0XFF;
PORTC=0XFF;
DDRD=0XFF;
PORTD=0XFF;
}
//------------------------------------------------------------------------------
//main
void main()
{
port_init();
lcd_init();
lcd_set_row(0); //这里怎么不设置行也可以~~难道是默认的~~
lcd_display_static(qianming);
while(1);
}
LCD12864Init.c
//------------------------------------------------------------------------------
//LCD12864初始化函数—可调用—ATMEGA16单片机
//这里面的lcd_check_busy竟然还不能用~~
#include"ioavr.h"
#include"intrinsics.h"
#include"MacroAndConst.h"
#include"delayics.h"
#include"bitics.h"
//------------------------------------------------------------------------------
//RS、RW、EN、CS1、CS2引脚输出高低电平的宏定义
#define lcd_rs_1 SET_BIT(PORTD,2) //数据命令选择端
#define lcd_rs_0 CLR_BIT(PORTD,2)
#define lcd_rw_1 SET_BIT(PORTD,1) //读写命令选择端
#define lcd_rw_0 CLR_BIT(PORTD,1)
#define lcd_en_1 SET_BIT(PORTD,0) //使能信号
#define lcd_en_0 CLR_BIT(PORTD,0)
#define lcd_cs1_1 SET_BIT(PORTD,4) //左半屏CS1
#define lcd_cs1_0 CLR_BIT(PORTD,4)
#define lcd_cs2_1 SET_BIT(PORTD,3) //右半屏CS2
#define lcd_cs2_0 CLR_BIT(PORTD,3)
//------------------------------------------------------------------------------
#define data_port PORTC //数据传输端口宏定义
#define busy 0x80
//------------------------------------------------------------------------------
//读“忙”函数;读取数据线的最高位DB7为1则为Busy
void lcd_check_busy()
{
uchar temp;
lcd_rs_0;
lcd_rw_1; //读指令
data_port=0x00;
lcd_en_1;
while(1)
{
lcd_en_1; //使能
DDRC=0x00;
temp=PINC;
if(temp<0x80) //如果数据口读入的数据小于0x80,说明最高位为0,LCD空闲,执行break跳出循环
break;
delay_us(1);
lcd_en_0; //禁能
}
DDRC=0xff; //在E的下降沿数据被锁存(写)入它们的控制驱动器
lcd_en_0;
}
//------------------------------------------------------------------------------
//LCD写指令函数
void lcd_write_command(uchar com)
{
// lcd_check_busy();
lcd_rs_0;
lcd_rw_0;
data_port=com;
lcd_en_1;
delay_us(1);
delay_us(1);
lcd_en_0;
}
//------------------------------------------------------------------------------
//LCD写数据函数
void lcd_write_data(uchar dat)
{
// lcd_check_busy();
lcd_rs_1;
lcd_rw_0;
data_port=dat;
lcd_en_1;
delay_us(1);
delay_us(1);
lcd_en_0;
}
//------------------------------------------------------------------------------
//LCD设置页函数。DB7~DB3为器件固定的位!为10111;然后DB2~DB0是设置8页的(一共就8页~~)
//起始页为0xB8,显示的RAM共64行,分为8页,每页8行~~
void lcd_set_page(uchar page)
{
page=0xb8|page; //用页的首地址或上你要显示的页数,就是当前显示的页
lcd_write_command(page);
}
//------------------------------------------------------------------------------
//LCD设置行函数
//DB6和DB7为器件固定位(感觉应该是为了和以后的器件兼容才这么设定的吧~~),都是1;其中DB5~DB0是用户设定的行,总共64行
//起始行的首地址为0xC0
void lcd_set_row(uchar row)
{
row=0xc0|row;
lcd_write_command(row); //用来写入指令,从哪一行开始~~
}
//------------------------------------------------------------------------------
//LCD设置列函数
//DB6和DB7为器件的固定位!是1、0。DB5~DB0是用户设定的列,总共64列,但是要注意:CS1和CS2选中左右两个屏幕,每个64列,两个就是128列了
//起始列为0x40
void lcd_set_column(uchar column)
{
column=0x40|column;
lcd_write_command(column); //写入列的首地址的指令~~
}
//------------------------------------------------------------------------------
//显示开关函数。0x3e为关显示;0x3f为开显示
void lcd_set_switch(uchar on_off)
{
on_off=0x3e|on_off; //其中只能为0或者1
lcd_write_command(on_off);
}
//------------------------------------------------------------------------------
//LCD选择屏幕的函数
//CS1和CS2都是低电平有效,CS1选择左半屏;CS2选择右半屏
void lcd_select_screen(uchar screen)
{
switch(screen)
{
case 0: lcd_cs1_0; lcd_cs2_0; break; //选中全屏
case 1: lcd_cs1_0; lcd_cs2_1; break; //选中左半屏
case 2: lcd_cs1_1; lcd_cs2_0; break; //选中右半屏
default: break;
}
}
//------------------------------------------------------------------------------
//LCD清屏函数
void lcd_clear_screen(uchar screen)
{
uchar i,j;
lcd_select_screen(screen); //清楚选中的屏幕
for(i=0;i<8;i++) //一共有8页,所以循环8次
{
lcd_set_page(i); //设置页,这里从第0页,第0行开始
lcd_set_column(0); //设置列,从第0列开始
for(j=0;j<64;j++) //控制列数0~63
{
lcd_write_data(0x00); //都清除为0,同时地址指针加1
}
}
}
//------------------------------------------------------------------------------
//LCD初始化函数
void lcd_init()
{
// lcd_check_busy();
lcd_select_screen(0); //选中全屏
lcd_set_switch(0); //关显示
lcd_select_screen(0);
lcd_set_switch(1);
lcd_select_screen(0);
lcd_clear_screen(0);
}
LCD12864Display.c
//------------------------------------------------------------------------------
//LCD12864显示函数—可以调用
//改变LCD显示函数的显示方法可以实现屏幕的左移~~上移~~等等~~
#include"ioavr.h"
#include"Lcd12864Init.h"
#include"MacroAndConst.h"
//------------------------------------------------------------------------------
//LCD静态显示函数
void lcd_display_static(uchar __flash *p)
{
uchar i,j;
uchar page;
for(page=0;page<8;page++) //8页
{
lcd_select_screen(1); //选中左半屏
lcd_set_page(page); //设置页
// lcd_set_row(0); //在这里设置行也可以~~晕了~~datasheet上写着是0~63行,难道可以理解为每页的第一行
//这个有待考虑
lcd_set_column(0); //设置列,共64列,从左半屏的第一列开始
for(i=0;i<64;i++)
{
lcd_write_data(p[page*128+i]); //显示数据,同时列的指针加1
}
lcd_select_screen(2); //选中右半屏
lcd_set_page(page); //设置页
lcd_set_column(0); //设置列,从右半屏的第一列开始
for(j=0;j<64;j++)
{
lcd_write_data(p[page*128+i+j]); //显示数据,同时列的指针加1
}
}
}
上一篇:AVR单片机(学习ing)—(一)、ATMEGA16的I\0端口的使用—01
下一篇:AVR单片机(学习ing)—(十)、ATMEGA16的同步串行接口SPI—03
推荐阅读最新更新时间:2024-03-16 15:46
设计资源 培训 开发板 精华推荐
- 邀请小伙伴一起学AM437x,好礼有你!
- 福禄克首款热成像万用表Fluke-279FC等你来尝鲜!晒心得享好礼喽!
- Silicon Labs BG22-EK4108A 开发套件 用科技连接未来 申请进行时!
- 趣味电子技术史话栏目开播啦!第一集:白炽灯到底是谁发明的?
- 如何用3个关键步骤,来确保下一代设计安全性,深入解读嵌入式设备DeepCover加密控制器,看视频答题赢好礼!
- #Micropython大作战#第二弹:发表mciropython原创教程赢好礼!
- 任选下载有礼|《新概念模拟电路》全五册合集/《ADI 参考电路合集》
- 是德科技下载有好礼:高速数字测试修炼手册
- 下资料赢京东卡|泰克“软硬”兼施 打造超值示波器
- EEWorld 芯积分兑换年度回馈来袭~多种赚积分捷径曝光+礼品兑换剧透