这个是三星的主控S6D1121彩屏(240*320*18bit,262Kcr)的彩屏的驱动程序.
采用的是80System+16bit接口
不知道是什么原因, 这个主控网上很少有资料,我也是看了好久的DS才明白了一些(全英文, 基本没中文资料, 反正是不好找到)
这是我接触的第一块彩屏哦~~
如果有用这个芯片的, 可能会有些方便, 也方便交流下使用经验.
这里总共分4个文件:
common.c,common.h:一些公共的包含与定义,函数
s6d1121.c,s6d1121.h:主控芯片驱动程序
//common.h - 一些公用的定义,函数等
#ifndef __COMMON_H__
#define __COMMON_H__
#include <reg52.h>
#include <intrins.h>
typedef unsigned int uint;
typedef unsigned char uchar;
void delay_us(uint us);
void delay_ms(uint ms);
#endif// !__COMMON_H__//common.c
#include "common.h"
void delay_ms(uint ms)
{
uchar k;
while(ms--){
for(k=125; k>0; k--){
_nop_();
}
}
}
void delay_us(uint us)
{
while(us--){
_nop_();
}
}//s6d1121.h
//S6D1121.h
//S6D1121驱动程序
//应用层请不要调用
//应调用GDI接口
//
//女孩不哭 QQ:191035066
//2012-11-05
#ifndef __S6D1121_H__
#define __S6D1121_H__
#include "common.h"
void LCD_SetAddress(uint x, uint y, uint width, uint height);
void LCD_WriteBus(uint dat);
void LCD_WriteCmdWithData(uchar cmd, uint value);
void LCD_WriteData(uint dat);
void LCD_WriteCmd(uchar cmd);
uint LCD_ReadBus(void);
uint LCD_ReadData(void);
void LCD_Init(void);
#endif// !__S6D1121_H__//s6d1121.c
/* 文件:S6D1121.c
* 功能:三星S6D1121主控彩屏驱动程序
* 版本:1.0
* 作者:女孩不哭
* Q--Q:191035066
* 时间:2012-11-05
* 说明:这里是底层驱动程序,用户不应该随便调用
* 请调用编写的图形接口程序
* 参考:http://www.elecfreaks.com/wiki/index.php?title=2.4%22_TFT_LCD:_TFT01-2.4
*/
#include "common.h"
#include "s6d1121.h"
//I/O口定义,除*_RST外,均Active Low
sbit LCD_RST= P2^0; //复位
sbit LCD_CS = P2^1; //片选
sbit LCD_RD = P2^2; //读取
sbit LCD_WR = P2^3; //写入
sbit LCD_RS = P2^4; //寄存器/数据选择
//数据总线端口定义
//如果使用16位总线,需要宏定义:BUS_16
//当使用8位总线时,仅*_HI有效
#define LCD_DATA_PORT_HI P0
#define LCD_DATA_PORT_LOW P1
//初始化,初始化之前
//MCU应该延时一段时间
//以使电源稳定
void LCD_Init(void)
{
LCD_RST = 1;
delay_ms(5);
LCD_RST = 0;
delay_ms(5);
LCD_RST = 1;
LCD_CS = 1;
LCD_RD = 1;
LCD_WR = 1;
delay_ms(5);
LCD_CS = 0;//片选使能
//详细的说明见一开始的网址
LCD_WriteCmdWithData(0x11,0x2004);
LCD_WriteCmdWithData(0x13,0xcc00);
LCD_WriteCmdWithData(0x15,0x2600);
LCD_WriteCmdWithData(0x14,0x252a);
LCD_WriteCmdWithData(0x12,0x0033);
LCD_WriteCmdWithData(0x13,0xcc04);
delay_ms(1);
LCD_WriteCmdWithData(0x13,0xcc06);
delay_ms(1);
LCD_WriteCmdWithData(0x13,0xcc4f);
delay_ms(1);
LCD_WriteCmdWithData(0x13,0x674f);
LCD_WriteCmdWithData(0x11,0x2003);
delay_ms(1);
LCD_WriteCmdWithData(0x30,0x2609);
LCD_WriteCmdWithData(0x31,0x242c);
LCD_WriteCmdWithData(0x32,0x1F23);
LCD_WriteCmdWithData(0x33,0x2425);
LCD_WriteCmdWithData(0x34,0x2226);
LCD_WriteCmdWithData(0x35,0x2523);
LCD_WriteCmdWithData(0x36,0x1C1A);
LCD_WriteCmdWithData(0x37,0x131D);
LCD_WriteCmdWithData(0x38,0x0B11);
LCD_WriteCmdWithData(0x39,0x1210);
LCD_WriteCmdWithData(0x3A,0x1315);
LCD_WriteCmdWithData(0x3B,0x3619);
LCD_WriteCmdWithData(0x3C,0x0D00);
LCD_WriteCmdWithData(0x3D,0x000D);
LCD_WriteCmdWithData(0x16,0x0007);
LCD_WriteCmdWithData(0x02,0x0013);
LCD_WriteCmdWithData(0x03,0x0003);
LCD_WriteCmdWithData(0x01,0x0127);
delay_ms(1);
LCD_WriteCmdWithData(0x08,0x0303);
LCD_WriteCmdWithData(0x0A,0x000B);
LCD_WriteCmdWithData(0x0B,0x0003);
LCD_WriteCmdWithData(0x0C,0x0000);
LCD_WriteCmdWithData(0x41,0x0000);
LCD_WriteCmdWithData(0x50,0x0000);
LCD_WriteCmdWithData(0x60,0x0005);
LCD_WriteCmdWithData(0x70,0x000B);
LCD_WriteCmdWithData(0x71,0x0000);
LCD_WriteCmdWithData(0x78,0x0000);
LCD_WriteCmdWithData(0x7A,0x0000);
LCD_WriteCmdWithData(0x79,0x0007);
LCD_WriteCmdWithData(0x07,0x0053);
LCD_WriteCmdWithData(0x79,0x0000);
LCD_WriteCmd(0x22);
LCD_CS = 1;//关闭片选
delay_ms(50);
LCD_CS = 0;
}
/* 函数:写命令
* 参数:uchar cmd:寄存器地址
* 返回:(无)
* 说明:RS为低为写寄存器
*/
void LCD_WriteCmd(uchar cmd)
{
LCD_RS = 0;//写命令模
LCD_WriteBus((uint)cmd);
}
/* 函数:写数据值
* 参数:uint dat:16位的数据
* 返回:(无)
* 说明:RS应为高
*/
void LCD_WriteData(uint dat)
{
LCD_RS = 1;//写数据
LCD_WriteBus(dat);
}
/*uint LCD_ReadData(void)
{
LCD_RS = 1;
return LCD_ReadBus();
}
uint LCD_ReadBus(void)
{
uint ret=0;
#ifdef BUS_16
LCD_RD = 0;
LCD_RD = 1;
ret = LCD_DATA_PORT_HI;
ret <<= 8;
ret |= LCD_DATA_PORT_LOW;
#else
LCD_RD = 0;
LCD_RD = 1;
ret = LCD_DATA_PORT_HI;
ret <<= 8;
LCD_RD = 0;
LCD_RD = 1;
ret |= LCD_DATA_PORT_HI;
#endif
return ret;
}*/
/* 函数:写寄存器
* 参数:
* uchar cmd:寄存器地址(8位)
* uint value:写给寄存器的值(16位)
* 返回:(无)
*/
void LCD_WriteCmdWithData(uchar cmd, uint value)
{
LCD_WriteCmd((uchar)cmd); //写寄存器地址
LCD_WriteData(value); //写寄存器值
}
//把命令/数据向总线写入
void LCD_WriteBus(uint dat)
{
#ifdef BUS_16 //16位总线模式
LCD_DATA_PORT_HI = (uchar)(dat>>8);
LCD_DATA_PORT_LOW = (uchar)dat;
LCD_WR = 0;
LCD_WR = 1; //S6D1121在每次的WR的上升沿读取数据
#else
//此时仅高字节有效
//先写高字节,再写低字节
LCD_DATA_PORT_HI = (uchar)(dat>>8);
LCD_WR = 0;
LCD_WR = 1;
LCD_DATA_PORT_HI = (uchar)dat;
LCD_WR = 0;
LCD_WR = 1;
#endif
}
/*
* 函数:设定写入数据的区域
* 参数:
* int x:X坐标(范围:0~239)
* int y:Y坐标(范围:0~319)
* uint width:宽
* uint height:高
* 返回:(无)
* 说明:(x1,y1; x2,y2)两点构成一个矩形
* 设定好区域后,在下一个指令前应该写缓冲区
*/
void LCD_SetAddress(uint x, uint y, uint width, uint height)
{
LCD_WriteCmdWithData(0x46, (x+width-1)<<8|x); //一次性设定x1,x2(高)
LCD_WriteCmdWithData(0x47, y+height-1); //y2
LCD_WriteCmdWithData(0x48, y); //y1
LCD_WriteCmdWithData(0x20, x); //设定RAM开始写的地址
LCD_WriteCmdWithData(0x21, y);
LCD_WriteCmd(0x22); //设定好写寄存器,开始写
}XXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXX
上一篇随笔贴出了驱动程序, 这篇文章的代码主要就是针对上层的调用了, 但是, 其实, 只要是底层的驱动程序能能够工作了, 图形设备接口应该就可以随便移植的咯~
一般彩屏的操作方式应该都是差不多的吧?
说明下, 本人不是很擅长图形方面的算法设计, 有些程序采用了大师们写的程序, 有些则是我自己写的(写得很差), 不过还是能够实现功能的哈~
我测试程序时是用的死太惨的51单片机测试的, ROM比较小, 只有60KB(整个字库有大约225KB), 完全没法装进去的, 所以就把要用的保存进去了, 所以会有ascii.c/.h文件的存在, 当然, 如果后来会用SD/MMC/U盘, 那就太好了,
不过, 很遗憾的是, SD卡我一直没有能够成功读取~ 慢慢来吧.
代码文件:
ascii.c/ascii.h:字库文件
common.c/common.h:一些公共的头文件/类型声明等等(和前面那篇随笔是一样的,没贴出)
gdi.c/gdi.h:图形设备接口的实现
有了这四个文件, 再加上前面的驱动程序, 把她们放到一个项目下, 就可以写程序调用了哈, 很方便的说~
//ascii.h
#ifndef __ASCIII_H__
#define __ASCIII_H__
uchar* asc_get_char(uchar uch);
uchar* asc_get_word(uchar* word);
uchar* asc_tpow(void);
#endif// !__ASCIII_H__//ascii.c
#include "common.h"
//这个是two's pow(2的次方), 其实没什么必要的, 可以忽略
uchar code t_p[8] = {0x80,0x40,0x20,0x10,0x08,0x04,0x02,0x01};
//这个是标准ASCII字符对应的像素值了
//应该有注意到, 我这里是宽8*高12的, 而我却在上下各加
//两个空行, 形成了宽8*高16的字符, 这样做的目的主要是为了方便
//因为我的中文字库是用的16*16(不是很小)
//是我从别人那里直接拷过来的, 看起来字体不是很好看
//这个是不需要程序获取的, 程序只需做的是调用asc_get_char(your_char)
//来得到一个字符数组的偏移就行了
uchar code ascii[][12] = {
/* */{0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00},
/* ! */{0x00,0x00,0x20,0x20,0x20,0x20,0x20,0x20,0x00,0x20,0x00,0x00},
/* " */{0x00,0x50,0x50,0x50,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00},
/* # */{0x00,0x00,0x28,0x28,0xFC,0x28,0x50,0xFC,0x50,0x50,0x00,0x00},
/* $ */{0x00,0x20,0x78,0xA8,0xA0,0x60,0x30,0x28,0xA8,0xF0,0x20,0x00},
/* % */{0x00,0x00,0x48,0xA8,0xB0,0x50,0x28,0x34,0x54,0x48,0x00,0x00},
/* & */{0x00,0x00,0x20,0x50,0x50,0x78,0xA8,0xA8,0x90,0x6C,0x00,0x00},
/* ' */{0x00,0x40,0x40,0x80,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00},
/* ( */{0x00,0x04,0x08,0x10,0x10,0x10,0x10,0x10,0x10,0x08,0x04,0x00},
/* ) */{0x00,0x40,0x20,0x10,0x10,0x10,0x10,0x10,0x10,0x20,0x40,0x00},
/* * */{0x00,0x00,0x00,0x20,0xA8,0x70,0x70,0xA8,0x20,0x00,0x00,0x00},
/* + */{0x00,0x00,0x20,0x20,0x20,0xF8,0x20,0x20,0x20,0x00,0x00,0x00},
/* , */{0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x40,0x40,0x80},
/* - */{0x00,0x00,0x00,0x00,0x00,0xF8,0x00,0x00,0x00,0x00,0x00,0x00},
/* . */{0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x40,0x00,0x00},
/* / */{0x00,0x08,0x10,0x10,0x10,0x20,0x20,0x40,0x40,0x40,0x80,0x00},
/* 0 */{0x00,0x00,0x70,0x88,0x88,0x88,0x88,0x88,0x88,0x70,0x00,0x00},
/* 1 */{0x00,0x00,0x20,0x60,0x20,0x20,0x20,0x20,0x20,0x70,0x00,0x00},
/* 2 */{0x00,0x00,0x70,0x88,0x88,0x10,0x20,0x40,0x80,0xF8,0x00,0x00},
/* 3 */{0x00,0x00,0x70,0x88,0x08,0x30,0x08,0x08,0x88,0x70,0x00,0x00},
/* 4 */{0x00,0x00,0x10,0x30,0x50,0x50,0x90,0x78,0x10,0x18,0x00,0x00},
/* 5 */{0x00,0x00,0xF8,0x80,0x80,0xF0,0x08,0x08,0x88,0x70,0x00,0x00},
/* 6 */{0x00,0x00,0x70,0x90,0x80,0xF0,0x88,0x88,0x88,0x70,0x00,0x00},
/* 7 */{0x00,0x00,0xF8,0x90,0x10,0x20,0x20,0x20,0x20,0x20,0x00,0x00},
/* 8 */{0x00,0x00,0x70,0x88,0x88,0x70,0x88,0x88,0x88,0x70,0x00,0x00},
/* 9 */{0x00,0x00,0x70,0x88,0x88,0x88,0x78,0x08,0x48,0x70,0x00,0x00},
/* : */{0x00,0x00,0x00,0x00,0x20,0x00,0x00,0x00,0x00,0x20,0x00,0x00},
/* ; */{0x00,0x00,0x00,0x00,0x00,0x20,0x00,0x00,0x00,0x20,0x20,0x00},
/* < */{0x00,0x04,0x08,0x10,0x20,0x40,0x20,0x10,0x08,0x04,0x00,0x00},
/* = */{0x00,0x00,0x00,0x00,0xF8,0x00,0x00,0xF8,0x00,0x00,0x00,0x00},
/* > */{0x00,0x40,0x20,0x10,0x08,0x04,0x08,0x10,0x20,0x40,0x00,0x00},
/* ? */{0x00,0x00,0x70,0x88,0x88,0x10,0x20,0x20,0x00,0x20,0x00,0x00},
/* @ */{0x00,0x00,0x70,0x88,0x98,0xA8,0xA8,0xB8,0x80,0x78,0x00,0x00},
/* A */{0x00,0x00,0x20,0x20,0x30,0x50,0x50,0x78,0x48,0xCC,0x00,0x00},
/* B */{0x00,0x00,0xF0,0x48,0x48,0x70,0x48,0x48,0x48,0xF0,0x00,0x00},
/* C */{0x00,0x00,0x78,0x88,0x80,0x80,0x80,0x80,0x88,0x70,0x00,0x00},
/* D */{0x00,0x00,0xF0,0x48,0x48,0x48,0x48,0x48,0x48,0xF0,0x00,0x00},
/* E */{0x00,0x00,0xF8,0x48,0x50,0x70,0x50,0x40,0x48,0xF8,0x00,0x00},
/* F */{0x00,0x00,0xF8,0x48,0x50,0x70,0x50,0x40,0x40,0xE0,0x00,0x00},
/* G */{0x00,0x00,0x38,0x48,0x80,0x80,0x9C,0x88,0x48,0x30,0x00,0x00},
/* H */{0x00,0x00,0xCC,0x48,0x48,0x78,0x48,0x48,0x48,0xCC,0x00,0x00},
/* I */{0x00,0x00,0xF8,0x20,0x20,0x20,0x20,0x20,0x20,0xF8,0x00,0x00},
/* J */{0x00,0x00,0x7C,0x10,0x10,0x10,0x10,0x10,0x10,0x90,0xE0,0x00},
/* K */{0x00,0x00,0xEC,0x48,0x50,0x60,0x50,0x50,0x48,0xEC,0x00,0x00},
/* L */{0x00,0x00,0xE0,0x40,0x40,0x40,0x40,0x40,0x44,0xFC,0x00,0x00},
/* M */{0x00,0x00,0xD8,0xD8,0xD8,0xD8,0xA8,0xA8,0xA8,0xA8,0x00,0x00},
/* N */{0x00,0x00,0xDC,0x48,0x68,0x68,0x58,0x58,0x48,0xE8,0x00,0x00},
/* O */{0x00,0x00,0x70,0x88,0x88,0x88,0x88,0x88,0x88,0x70,0x00,0x00},
/* P */{0x00,0x00,0xF0,0x48,0x48,0x70,0x40,0x40,0x40,0xE0,0x00,0x00},
/* Q */{0x00,0x00,0x70,0x88,0x88,0x88,0x88,0xE8,0x98,0x70,0x18,0x00},
/* R */{0x00,0x00,0xF0,0x48,0x48,0x70,0x50,0x48,0x48,0xEC,0x00,0x00},
/* S */{0x00,0x00,0x78,0x88,0x80,0x60,0x10,0x08,0x88,0xF0,0x00,0x00},
/* T */{0x00,0x00,0xF8,0xA8,0x20,0x20,0x20,0x20,0x20,0x70,0x00,0x00},
/* U */{0x00,0x00,0xCC,0x48,0x48,0x48,0x48,0x48,0x48,0x30,0x00,0x00},
/* V */{0x00,0x00,0xCC,0x48,0x48,0x50,0x50,0x30,0x20,0x20,0x00,0x00},
/* W */{0x00,0x00,0xA8,0xA8,0xA8,0x70,0x50,0x50,0x50,0x50,0x00,0x00},
/* X */{0x00,0x00,0xD8,0x50,0x50,0x20,0x20,0x50,0x50,0xD8,0x00,0x00},
/* Y */{0x00,0x00,0xD8,0x50,0x50,0x20,0x20,0x20,0x20,0x70,0x00,0x00},
/* Z */{0x00,0x00,0xF8,0x90,0x10,0x20,0x20,0x40,0x48,0xF8,0x00,0x00},
/* [ */{0x00,0x38,0x20,0x20,0x20,0x20,0x20,0x20,0x20,0x20,0x38,0x00},
/* \ */{0x00,0x40,0x40,0x40,0x20,0x20,0x10,0x10,0x10,0x08,0x00,0x00},
/* ] */{0x00,0x70,0x10,0x10,0x10,0x10,0x10,0x10,0x10,0x10,0x70,0x00},
/* ^ */{0x00,0x20,0x50,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00},
/* _ */{0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0xFC},
/* ` */{0x00,0x20,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00},
/* a */{0x00,0x00,0x00,0x00,0x00,0x30,0x48,0x38,0x48,0x3C,0x00,0x00},
/* b */{0x00,0x00,0xC0,0x40,0x40,0x70,0x48,0x48,0x48,0x70,0x00,0x00},
/* c */{0x00,0x00,0x00,0x00,0x00,0x38,0x48,0x40,0x40,0x38,0x00,0x00},
/* d */{0x00,0x00,0x18,0x08,0x08,0x38,0x48,0x48,0x48,0x3C,0x00,0x00},
/* e */{0x00,0x00,0x00,0x00,0x00,0x30,0x48,0x78,0x40,0x38,0x00,0x00},
/* f */{0x00,0x00,0x1C,0x20,0x20,0x78,0x20,0x20,0x20,0x78,0x00,0x00},
/* g */{0x00,0x00,0x00,0x00,0x00,0x3C,0x48,0x30,0x40,0x78,0x44,0x38},
/* h */{0x00,0x00,0xC0,0x40,0x40,0x70,0x48,0x48,0x48,0xEC,0x00,0x00},
/* i */{0x00,0x00,0x20,0x00,0x00,0x60,0x20,0x20,0x20,0x70,0x00,0x00},
/* j */{0x00,0x00,0x10,0x00,0x00,0x30,0x10,0x10,0x10,0x10,0x10,0xE0},
/* k */{0x00,0x00,0xC0,0x40,0x40,0x5C,0x50,0x70,0x48,0xEC,0x00,0x00},
/* l */{0x00,0x00,0xE0,0x20,0x20,0x20,0x20,0x20,0x20,0xF8,0x00,0x00},
/* m */{0x00,0x00,0x00,0x00,0x00,0xF0,0xA8,0xA8,0xA8,0xA8,0x00,0x00},
/* n */{0x00,0x00,0x00,0x00,0x00,0xF0,0x48,0x48,0x48,0xEC,0x00,0x00},
/* o */{0x00,0x00,0x00,0x00,0x00,0x30,0x48,0x48,0x48,0x30,0x00,0x00},
/* p */{0x00,0x00,0x00,0x00,0x00,0xF0,0x48,0x48,0x48,0x70,0x40,0xE0},
/* q */{0x00,0x00,0x00,0x00,0x00,0x38,0x48,0x48,0x48,0x38,0x08,0x1C},
/* r */{0x00,0x00,0x00,0x00,0x00,0xD8,0x60,0x40,0x40,0xE0,0x00,0x00},
/* s */{0x00,0x00,0x00,0x00,0x00,0x78,0x40,0x30,0x08,0x78,0x00,0x00},
/* t */{0x00,0x00,0x00,0x20,0x20,0x70,0x20,0x20,0x20,0x18,0x00,0x00},
/* u */{0x00,0x00,0x00,0x00,0x00,0xD8,0x48,0x48,0x48,0x3C,0x00,0x00},
/* v */{0x00,0x00,0x00,0x00,0x00,0xEC,0x48,0x50,0x30,0x20,0x00,0x00},
/* w */{0x00,0x00,0x00,0x00,0x00,0xA8,0xA8,0x70,0x50,0x50,0x00,0x00},
/* x */{0x00,0x00,0x00,0x00,0x00,0xD8,0x50,0x20,0x50,0xD8,0x00,0x00},
/* y */{0x00,0x00,0x00,0x00,0x00,0xEC,0x48,0x50,0x30,0x20,0x20,0xC0},
/* z */{0x00,0x00,0x00,0x00,0x00,0x78,0x10,0x20,0x20,0x78,0x00,0x00},
/* { */{0x00,0x18,0x10,0x10,0x10,0x20,0x10,0x10,0x10,0x10,0x18,0x00},
/* | */{0x10,0x10,0x10,0x10,0x10,0x10,0x10,0x10,0x10,0x10,0x10,0x10},
/* } */{0x00,0x60,0x20,0x20,0x20,0x10,0x20,0x20,0x20,0x20,0x60,0x00},
/* ~ */{0x40,0xA4,0x18,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00}
};
//用了一个结构体来索引
//因为没有足够的空间来保存字库
//所以就不能按照平常的区位码来索引汉字的代码了
//which[3]:保存一个中文汉字,what[32]:保存的代码
//生成方式:http://www.cnblogs.com/nbsofer/archive/2012/11/01/2749026.html
//同样是内部使用, 外部程序只需要调用 asc_get_word(相对于*_char而言)(见下)
struct wmap_s{
uchar which[3];
uchar what[32];
} code wmap[] = {
{"我",{0x04,0x80,0x0E,0xA0,0x78,0x90,0x08,0x90,0x08,0x84,0xFF,0xFE,0x08,0x80,0x08,0x90,0x0A,0x90,0x0C,0x60,0x18,0x40,0x68,0xA0,0x09,0x20,0x0A,0x14,0x28,0x14,0x10,0x0C}},
{"的",{0x10,0x40,0x10,0x40,0x22,0x44,0x7F,0x7E,0x42,0x84,0x43,0x04,0x42,0x04,0x42,0x84,0x7E,0x64,0x42,0x24,0x42,0x04,0x42,0x04,0x42,0x04,0x7E,0x04,0x42,0x28,0x00,0x10}},
{"是",{0x0F,0xE0,0x08,0x20,0x08,0x20,0x0F,0xE0,0x08,0x20,0x08,0x20,0x0F,0xE0,0x00,0x04,0xFF,0xFE,0x01,0x00,0x09,0x20,0x09,0xF0,0x09,0x00,0x15,0x00,0x23,0x06,0x40,0xFC}},
{"现",{0x00,0x08,0x09,0xFC,0xFD,0x08,0x11,0x28,0x11,0x28,0x11,0x28,0x11,0x28,0x7D,0x48,0x11,0x48,0x11,0x48,0x10,0x40,0x1C,0xA0,0xF0,0xA0,0x41,0x22,0x02,0x22,0x0C,0x1E}},
{"在",{0x02,0x00,0x02,0x00,0x02,0x04,0xFF,0xFE,0x04,0x00,0x04,0x40,0x08,0x40,0x08,0x50,0x13,0xF8,0x30,0x40,0x50,0x40,0x90,0x40,0x10,0x40,0x10,0x44,0x17,0xFE,0x10,0x00}},
{"时",{0x00,0x08,0x04,0x08,0x7E,0x08,0x44,0x08,0x47,0xFE,0x44,0x08,0x44,0x08,0x7C,0x88,0x44,0x48,0x44,0x48,0x44,0x08,0x44,0x08,0x7C,0x08,0x44,0x48,0x00,0x28,0x00,0x10}},
{"间",{0x20,0x04,0x1B,0xFE,0x08,0x04,0x40,0x24,0x4F,0xF4,0x48,0x24,0x48,0x24,0x48,0x24,0x4F,0xE4,0x48,0x24,0x48,0x24,0x48,0x24,0x4F,0xE4,0x48,0x24,0x40,0x14,0x40,0x08}},
{"温",{0x00,0x08,0x43,0xFC,0x32,0x08,0x12,0x08,0x83,0xF8,0x62,0x08,0x22,0x08,0x0B,0xF8,0x10,0x00,0x27,0xFC,0xE4,0xA4,0x24,0xA4,0x24,0xA4,0x24,0xA4,0x2F,0xFE,0x20,0x00}},
{"度",{0x01,0x00,0x00,0x84,0x3F,0xFE,0x22,0x20,0x22,0x28,0x3F,0xFC,0x22,0x20,0x23,0xE0,0x20,0x00,0x2F,0xF0,0x22,0x20,0x21,0x40,0x20,0x80,0x43,0x60,0x8C,0x1E,0x30,0x04}},
{"初",{0x20,0x00,0x10,0x00,0x10,0x04,0x05,0xFE,0xFC,0x44,0x08,0x44,0x10,0x44,0x34,0x44,0x58,0x44,0x94,0x44,0x10,0x44,0x10,0x84,0x10,0x84,0x11,0x04,0x12,0x28,0x14,0x10}},
{"始",{0x10,0x40,0x10,0x40,0x10,0x40,0x10,0x80,0xFC,0x88,0x25,0x04,0x27,0xFE,0x24,0x02,0x24,0x04,0x49,0xFE,0x29,0x04,0x11,0x04,0x29,0x04,0x45,0x04,0x85,0xFC,0x01,0x04}},
{"化",{0x08,0x80,0x08,0x80,0x08,0x80,0x10,0x88,0x10,0x98,0x30,0xA0,0x50,0xC0,0x90,0x80,0x11,0x80,0x12,0x80,0x14,0x80,0x10,0x80,0x10,0x82,0x10,0x82,0x10,0x7E,0x10,0x00}},
};
//函数:asc_get_word(uchar* pword)
//用来获取一个中文汉字的代码保存位置偏移
//如果不存在, 返回NULL了(这里直接用0代替了)
uchar* asc_get_word(uchar* pword)
{
uint k;
uint len = sizeof(wmap)/sizeof(wmap[0]);
for(k=0; k<len; k++){
if(wmap[k].which[0] == pword[0] &&
wmap[k].which[1] == pword[1])
{
return &wmap[k].what[0];
}
}
return 0;
}
//同上, 不过这个是用来获取ASCII字符的
//应该注意到的是:我是根据标准ASCII的顺序来索引ASCII字符的
//所以,上面的ASCII字符代码表不能再添加/删除
//当然, 中文字库不存在这个限制
//使用:传入一个ASCII字符,返回编码的偏移(没有的话返回0)
uchar* asc_get_char(char uch)
{
char idx;
idx = (uch-32);//从空格算起, 控制字符就没用了,回车在其它程序中已经单独处理
if(idx<0 || idx>sizeof(ascii)/sizeof(ascii[0]))
return 0;
return &ascii[idx][0];
}
//返回2的次方, 用得着的时候就用吧
//比如: uchar* tp = asc_tpow();
//tp[0]=2^7,tp[1]=2^6, ...
uchar* asc_tpow(void)
{
return &t_p[0];
}//gdi.h
#ifndef __GDI_H__
#define __GDI_H__
#include "common.h"
#include "s6d1121.h"
//颜色描述表(前景色, 背景色)
typedef struct color_table_s{
uint fgc;
uint bgc;
}color_table;
//很有用的rgb(r,g,b) 宏 rgb(红,绿,蓝)
#define rgb(r,g,b) ((uint)(\
((uint)(uchar)(r)<<8 & 0xf800|(uchar)(r)>>7)|\
((uint)(uchar)(g)<<3 & 0x7e00)|\
((uint)(uchar)(b)>>2 | (uchar)(b)>>7)))
//function prototype
//函数的说明直接见各函数
void rectangle(int left, int top, int width, int height, uint color);
void line(int x1, int y1, int x2, int y2, uint color);
void triangle(int x1, int y1, int x2, int y2, int x3, int y3, uint color);
void set_pixel(uint x, uint y, uint color);
void fill_rect(uint left, uint top, uint width, uint height, uint color);
void clear_screen(uint color);
void char_out(int x, int y, char ch, uint fgc, uint bgc);
void word_out(int x, int y, uchar* pword, uint fgc, uint bgc);
void text_out(int x, int y, uchar* str, uint fgc, uint bgc, color_table* pct);
int cirpot(int x0,int y0,int x,int y,uint color);
void circle(int x0,int y0,int r,uint color);
float sin(float rad);
uchar* itoa(int x);
//some useful function macros
//一些有用的宏,主要是用于把应用层的程序和驱动层的程序隔离
//其实还是是调用的该驱动层的函数
/* 设定待写入/读取数据的缓冲区:X坐标,Y坐标,宽,高 */
#define set_area(x,y,width,height) LCD_SetAddress((x),(y),(width),(height))
/* 在 set_area 调用后, 写入数据的操作 */
#define set_data(dat) LCD_WriteData((dat))
//初始化彩屏
#define init_lcd LCD_Init
//清屏
#define clr_scr clear_screen
//设定彩屏主控内部寄存器的值(不应该暴露给应用层的,以后再更新,必然的)
#define set_reg(reg,value) LCD_WriteCmdWithData((reg),(value))
#endif// !__GDI_H__//gdi.c
#include "common.h"
#include "gdi.h"
#include "s6d1121.h"
#include "ascii.h"
//用来把指定的整数转换为字符串
//保存在静态变量buf中并返回给调用者
//显示0还是显示空格在于自己了
uchar* itoa(int x)
{
static uchar buf[6];
buf[0] = x/10000+'0';
x %= 10000;
buf[1] = x/1000+'0';
x %= 1000;
buf[2] = x/100+'0';
x %= 100;
buf[3] = x/10+'0';
x %= 10;
buf[4] = x+'0';
buf[5] = 0;
return &buf[0];
}
//画矩形的函数
//参数:左上角的X坐标,左上角的Y坐标, 宽, 高, 颜色(有一些常用的颜色应该被定义,
//下次再修改, 51的刷屏速度忍无可忍)
void rectangle(int left, int top, int width, int height, uint color)
{
int x2,y2;
x2 = left+width-1;
y2 = top+height-1;
//必须保存在有效范围内
if(left<0 || top<0 || x2>239 || y2>319)
return;
//看到了吧, 实现就是转换成画直线
//所以:最关键的怎样画点,直线
line(left, top, x2, top, color);
line(x2, top, x2, y2, color);
line(x2, y2, left, y2, color);
line(left, y2, left, top, color);
}
//画三角形, 参数为三个坐标点+颜色
//其实还是是转换成画三条直线
void triangle(int x1, int y1, int x2, int y2, int x3, int y3,uint color)
{
line(x1,y1,x2,y2, color);
line(x2,y2,x3,y3, color);
line(x1,y1,x3,y3, color);
}
//画直线, 可怜这是我自己写的算法, 效率太低了
//而且,如果不仔细读读代码, 很难明白的
//以后还是换成大师们已经写好了的吧
//参数:两点成线, 不用多说, 当然, 颜色有要有的
//画线就是转换成画点,所以关键是怎么画点,set_pixel再介绍
void line(int x1, int y1, int x2, int y2, uint color)
{
int w,h;//记录长宽
float rate1,rate2;//高与宽的比率
int xaxis, yaxis, k;
int ffalling = 0;//直线是否为下降沿
if(x1>x2){
k = x1; x1 = x2; x2 = k;
k = y1; y1 = y2; y2 = k;
}
w = x2-x1+1;//get horizontal distance
h = y2-y1;//get vertical distance
if(h<0) h = -h;
h = h+1;
rate1 = (float)h/w;//根据相似三角形计算坐标
rate2 = (float)w/h;
if(w>h){//宽大于高
ffalling = y1-y2;
if(y1 == y2)
ffalling = 0;
for(k=0; k<w; k++){
if(ffalling<0){
//+0.5是作四四舍五入
yaxis = y2-(uint)((w-k)*rate1+0.5)+1;
}else if(ffalling>0){
yaxis = y1-(uint)(k*rate1+0.5)+1;
}else{
yaxis = y1;
}
set_pixel(k+x1,yaxis, color);
}
}else{ //h>w
ffalling = y1-y2;
if(x1==x2)
ffalling = 0;
for(k=0; k<h; k++){
if(ffalling<0){
xaxis = x1+(uint)(k*rate2+0.5)+1;
}else if(ffalling>0){
xaxis = x1+((h-k)*rate2+0.5)+1;
}else{
xaxis = x1;
}
set_pixel(xaxis, (y1<y2?y1:y2)+k, color);
}
}
}
//这个就是画点函数了
//set_area,set_data 完工
void set_pixel(uint x, uint y, uint color)
{
if(x>239 || y>319) return;
set_area(x, y, 1, 1);
set_data(color);
}
//填充一块矩形为指定颜色
void fill_rect(uint left, uint top, uint width, uint height, uint color)
{
uint a,b;
if(left>239 || top>319) return;
if(left+width-1>239) width = 239-left+1;//clip x border
if(top+height-1>319) height = 319-top+1;//clip y border
set_area(left, top, width, height);
for(a=0; a<height; a++){
for(b=0; b<width; b++){
set_data(color);
}
}
}
//用 颜色 color 清屏
void clear_screen(uint color)
{
uint i,j;
set_area(0,0,240,320);
for(i=0;i<320;i++){
for (j=0;j<240;j++){
set_data(color);
}
}
}
//在指定位置输出一个ASCII字符
//属于内部调用函数
//参数:X坐标,Y坐标,字符,前景色,背景色
//如果不知道是怎样写字的, 看去看看我那篇
//介绍字库的使用的文章吧(在前面)
void char_out(int x, int y, char ch, uint fgc, uint bgc)
{
char k,i;
uchar* pch = 0;
uchar* t_p = asc_tpow();
pch = asc_get_char(ch);
if(!pch) return;
//我的字库是8*12的
//但是我把它转换成了8*16
set_area(x, y, 8, 16);
//所以我先写三行空白行
for(k=0; k<24; k++){
set_data(bgc);
}
for(k=0; k<12; k++){
for(i=0; i<8; i++){
set_data(*pch&t_p[i]?fgc:bgc);
}
pch++;
}
//再写一行空白行,用背景色填充
for(k=0; k<8; k++){
set_data(bgc);
}
}
//前面是输出ASCII字符
//这个是在指定位置输出指定的汉字
//属于内部调用函数
//参数:X坐标,Y坐标,一个汉字的数组(uchar数组,3个字符,2个最好,节约嘛),前景色,背景色
void word_out(int x, int y, uchar* pword, uint fgc, uint bgc)
{
uchar xx,k,yy;
uchar* pch = 0;
uchar* t_p = asc_tpow();
//"汉字两个都>=0xa0, 不然就是错的"
if(pword[0]<0xa0 || pword[1]<0xa0)
return;
pch = asc_get_word(pword);
if(!pch) return;
//"需要检测x+15,y+15是否出界,则k应为实际打印的行/列数"
//"并设定一个参数为是否依然打印不能完整打印的char/word"
set_area(x,y,16,16);
//16*2*8:32个字节就这样写进去咯
for(k=0; k<16; k++){//16行
for(yy=0; yy<2; yy++){//一行2个字节(16位)
for(xx=0; xx<8; xx++){//一个字节8个位
set_data(*pch&t_p[xx]?fgc:bgc);
}
pch++;
}
}
}
//这个都是真正的输出文本的函数的
//参数:X坐标,Y坐标,字符串指针,前景色,背景色,颜色索引表(可选)
//说明:如果x,y均等于-1,则程序在上一次的结尾接着输出
//仅可用的控制字符:'\n',换行
void text_out(int x, int y, uchar* str, uint fgc, uint bgc, color_table* pct)
{
uchar* pch = str;
uint k;
uint fgcolor, bgcolor;
//保存最近一次调用时的坐标
static uchar lastx=0;
static uint lasty=0;
//判断是否为接着输出
if(x==-1 && y==-1){
x = lastx;
y = lasty;
}
if(!pch) return;
//遍历每个字符调用
// char_out 进行字符输出
// word_out 进行中文输出
for(;*pch;){
if(pct){
fgcolor = pct->fgc;
bgcolor = pct->bgc;
}else{
fgcolor = fgc;
bgcolor = bgc;
}
if(*pch >= 0xa0){//word
word_out(x, y, pch, fgcolor, bgcolor);
x += 16;//more 8 pixels than chars
pch++;
}else if(*pch == '\n'){//换行处理
x = 0;
y += 16;
}else{//char
char_out(x, y, *pch, fgcolor, bgcolor);
x += 8;
}
pch++;
//check if return
//word to next line
//这里是当输出的汉字/字符不能完全显示是的处理(换行)
if(/*x==240*/(*pch>=0xa0&&x>224) || x>232){//to next line
if(*pch){
set_area(x,y,240,320);
for(k=0; k<16*(239-x+1); k++){
set_data(bgcolor);
}
}
x = 0;
y += 16;
}
if(/*y==320*/y>304){//is bottom
if(*pch){
set_area(x,y,240,320);
for(k=0; k<240*(319-y+1); k++){
set_data(bgcolor);
}
}
y = 0;
}
if(pct) pct++;
}
lastx = x;
lasty = y;
}
/*
float sin(float rad)
{
float x1 = rad;
float x3 = x1*rad*rad;
float x5 = x3*rad*rad;
float x7 = x5*rad*rad;
return x1/1-x3/6+x5/120-x7/5040;
}
*/
/*float cos(float rad)
{
}*/
//画圆函数
//大师写的哦~ 膜拜
//基本没看懂
//不记得名字了, 好像是叫Bruce...画圆算法
void circle(int x0,int y0, int r,uint color)
{
int x,y,d;
x=0;
y=(int)r;
d=(int)(3-2*r);
while(x<y){
cirpot(x0,y0,x,y,color);
if(d<0){
d+=4*x+6;
}else{
d+=4*(x-y)+10;
y--;
}
x++;
}
if(x==y)
cirpot(x0,y0,x,y,color);
}
int cirpot(int x0,int y0,int x,int y,uint color)
{
set_pixel((x0+x),(y0+y), color);
set_pixel((x0+y),(y0+x), color);
set_pixel((x0+y),(y0-x), color);
set_pixel((x0+x),(y0-y), color);
set_pixel((x0-x),(y0-y), color);
set_pixel((x0-y),(y0-x), color);
set_pixel((x0-y),(y0+x), color);
set_pixel((x0-x),(y0+y), color);
return 0;
}
/*int abs(int a, int b)
{
if(a>b) return a-b;
else return b-a;
}*/就这样吧~ 晚安~
5766

被折叠的 条评论
为什么被折叠?



