高通字库芯片GT20L16S1Y驱动 0.96寸 OLED 任意显示中文
程序员文章站
2022-03-05 08:40:17
连续两个月的加班,给ODM客户生产温控器订单,今天终于顺利发货,对于工程师出身的我,终于可以对着电脑,消停几天,研究技术,分享技术了,闲话少说,直接进入正题:半年前有个老客户介绍个*某院的项目,我主业是做各种温控器的,老客户介绍的新客户,没法推脱,就干吧,项目需求具体细节不方便公开,说说本次要分享的部分,要实现的功能是一个OLED任意显示一段中文句子。项目的硬件组成比较简单:STM8L051+ KEY + OLED + GT20L16S1Y软件平台:IAR整个项目没什么难点,但没做过这个的...
连续两个月的加班,给ODM客户生产温控器订单,今天终于顺利发货,对于工程师出身的我,终于可以对着电脑,消停几天,研究技术,分享技术了,闲话少说,直接进入正题:
半年前有个老客户介绍个*某院的项目,我主业是做各种温控器的,老客户介绍的新客户,没法推脱,就干吧,项目需求具体细节不方便公开,说说本次要分享的部分,要实现的功能是一个OLED任意显示一段中文句子。
项目的硬件组成比较简单:STM8L051+ KEY + OLED + GT20L16S1Y
软件平台:IAR
整个项目没什么难点,但没做过这个的,也需要一番折腾,在这里,我把项目的主要代码贴出来供电友们参考,也感谢之前版本的电友。做个靠谱的电友,除了分享单独的.c.h驱动文件,最后还会有调用伪代码,也就是整个代码的调用过程。
一:高通字库芯片驱动程序:
// .c文件
#include "stm8l15x.h"//STM8L051/151等系列共用库函数
#include "GT20L16S1Y.h"
#define S1Y_CLK_L (GPIO_ResetBits(GPIOB,GPIO_Pin_4))
#define S1Y_CLK_H (GPIO_SetBits(GPIOB,GPIO_Pin_4))
#define S1Y_CS_L (GPIO_ResetBits(GPIOB,GPIO_Pin_5))
#define S1Y_CS_H (GPIO_SetBits(GPIOB,GPIO_Pin_5))
#define S1Y_SI_L (GPIO_ResetBits(GPIOB,GPIO_Pin_6))
#define S1Y_SI_H (GPIO_SetBits(GPIOB,GPIO_Pin_6))
#define S1Y_SO GPIO_ReadInputDataBit(GPIOB, GPIO_Pin_7)
void GT20L16_init(void)
{
/* GT20L16S1Y 引脚配置 */
GPIO_Init(GPIOB, GPIO_Pin_4, GPIO_Mode_Out_PP_Low_Fast); //高速推挽输出低电平[SCLK]
GPIO_Init(GPIOB, GPIO_Pin_5, GPIO_Mode_Out_PP_Low_Fast); //高速推挽输出低电平[CS]
GPIO_Init(GPIOB, GPIO_Pin_6, GPIO_Mode_Out_PP_Low_Fast); //高速推挽输出低电平[SI]
GPIO_Init(GPIOB, GPIO_Pin_7, GPIO_Mode_In_PU_No_IT); //上拉输入[SO]
}
/***************************************
ASCII 调用
ASCIICode:表示ASCII 码(8bits)
BaseAdd:说明该套字库在芯片中的起始地址。
r_dat_bat: 是读点阵数据函数。
DZ_Data:是保存读出的点阵数据的数组。
****************************************/
u8 S1Y_ASCII_GetData( u8 ASCIICode, u32 BaseAdd, u8* S1YDZ_Data )
{
if( ( ASCIICode >= 0x20 ) && ( ASCIICode <= 0x7e ) )
{
switch( BaseAdd )
{
case 0x3bfc0:
S1Y_rdat_bat( ( ASCIICode - 0x20 ) * 8 + BaseAdd, 8, S1YDZ_Data ); //5X7
break ;
case 0x66c0:
S1Y_rdat_bat( ( ASCIICode - 0x20 ) * 8 + BaseAdd, 8, S1YDZ_Data ); //7X8
break ;
case 0x3b7c0:
S1Y_rdat_bat( ( ASCIICode - 0x20 ) * 16 + BaseAdd, 16, S1YDZ_Data ); //8X16 A
break ;
case 0x3cf80:
S1Y_rdat_bat( ( ASCIICode - 0x20 ) * 26 + BaseAdd, 16, S1YDZ_Data ); //8X16 F
break ;
case 0x3c2c0:
S1Y_rdat_bat( ( ASCIICode - 0x20 ) * 34 + BaseAdd + 2, 32, S1YDZ_Data ); //16X16 Arial
break ;
case 0x3d580:
S1Y_rdat_bat( ( ASCIICode - 0x20 ) * 34 + BaseAdd + 2, 32, S1YDZ_Data ); //16X16 T
break ;
default:
break;
}
return 1;
}
else
{
return 0;
}
}
/***************************************************
16 点GB2312 标准点阵字库
参数说明:
GBCode表示汉字内码。
MSB 表示汉字内码GBCode 的高8bits。
LSB 表示汉字内码GBCode 的低8bits。
Address 表示汉字或ASCII字符点阵在芯片中的字节地址。
BaseAdd:说明点阵数据在字库芯片中的起始地址。
r_dat_bat 是读点阵数据函数。
DZ_Data是保存读出的点阵数据的数组。
*****************************************************/
void S1Y_gt16_GetData( u8 MSB, u8 LSB, u8* S1YDZ_Data )
{
u32 temp = ( MSB - 0xB0 ) * 94 + LSB - 0xA1;
u32 BaseAdd = 0, Address;
if( MSB == 0xA9 && LSB >= 0xA1 )
{
Address = ( 282 + ( LSB - 0xA1 ) ) * 32 + BaseAdd;
}
else if( MSB >= 0xA1 && MSB <= 0xA3 && LSB >= 0xA1 )
{
Address = ( ( MSB - 0xA1 ) * 94 + ( LSB - 0xA1 ) ) * 32 + BaseAdd;
}
else if( MSB >= 0xB0 && MSB <= 0xF7 && LSB >= 0xA1 )
{
Address = ( 846 + temp ) * 32 + BaseAdd;
}
S1Y_rdat_bat( Address, 32, S1YDZ_Data );
}
//Address=((MSB-0xB0)*94+(LSB-0xA1)+846)*32+BaseAdd;
/****************************************************
8X16 点国标扩展字符
说明:
BaseAdd:说明本套字库在字库芯片中的起始字节地址。
FontCode:表示字符内码(16bits).
Address:表示字符点阵在芯片中的字节地址。
r_dat_bat 是读点阵数据函数。
DZ_Data是保存读出的点阵数据的数组。
*****************************************************/
void S1Y_GB_EXT_816( u16 FontCode, u8* S1YDZ_Data )
{
u32 BaseAdd = 0x3b7d0, Address;
u32 temp1 = ( FontCode - 0xAAA1 );
u32 temp2 = ( FontCode - 0xABA1 + 95 );
if( FontCode >= 0xAAA1 && FontCode <= 0xAAFE )
{
Address = temp1 * 16 + BaseAdd;
}
else if( FontCode >= 0xABA1 && FontCode <= 0xABC0 )
{
Address = temp2 * 16 + BaseAdd;
}
S1Y_rdat_bat( Address, 16, S1YDZ_Data );
}
/****************************************************
从字库中读数据函数
说明:
Address : 表示字符点阵在芯片中的字节地址。
byte_long: 是读点阵数据字节数。
*p_arr : 是保存读出的点阵数据的数组。
*****************************************************/
u8 S1Y_rdat_bat( u32 address, u8 byte_long, u8* p_arr )
{
unsigned int j = 0;
S1Y_CS_L;
S1Y_SendByte( address );
for( j = 0; j < byte_long; j++ )
{
p_arr[j] = S1Y_ReadByte();
}
S1Y_CS_H;
return p_arr[0];
}
void S1Y_SendByte( u32 cmd )
{
u8 i;
cmd = cmd | 0x03000000;
for( i = 0; i < 32; i++ )
{
S1Y_CLK_L;
if( cmd & 0x80000000 )
{
S1Y_SI_H;
}
else
{
S1Y_SI_L;
}
S1Y_CLK_H;
cmd = cmd << 1;
}
}
u8 S1Y_ReadByte( void )
{
u8 i;
u8 dat = 0;
S1Y_CLK_H;
for( i = 0; i < 8; i++ )
{
S1Y_CLK_L;
dat = dat << 1;
if( S1Y_SO )
{
dat = dat | 0x01;
}
else
{
dat &= 0xfe;
}
S1Y_CLK_H ;
}
return dat;
}
//-------------------------------------分割线---------------------------------
//.h 文件
#ifndef _GT20L16S1Y_H_
#define _GT20L16S1Y_H_
void S1Y_SendByte(u32 cmd);
u8 S1Y_ReadByte(void);
u8 S1Y_rdat_bat(u32 address,u8 byte_long,u8 *p_arr);
u8 S1Y_ASCII_GetData(u8 ASCIICode,u32 BaseAdd,u8 *S1YDZ_Data);
void S1Y_gt16_GetData (u8 MSB,u8 LSB,u8 *S1YDZ_Data);
void S1Y_GB_EXT_816(u16 FontCode,u8 *S1YDZ_Data);
void GT20L16_init(void);
#endif
OLED显示就比较简单了,显示任意汉字字符串,也折腾了我几根头发,下面也贴出来分享一下:
// .c文件
#include "oled.h"
#include "oledfont.h"
#include "GT20L16S1Y.h"
//OLED的显存
//存放格式如下.
//[0]0 1 2 3 ... 127
//[1]0 1 2 3 ... 127
//[2]0 1 2 3 ... 127
//[3]0 1 2 3 ... 127
//[4]0 1 2 3 ... 127
//[5]0 1 2 3 ... 127
//[6]0 1 2 3 ... 127
//[7]0 1 2 3 ... 127
void delay_ms( unsigned int ms )
{
unsigned int a;
while( ms )
{
a = 1800;
while( a-- );
ms--;
}
return;
}
//反显函数
void OLED_ColorTurn( u8 i )
{
if( i == 0 )
{
OLED_WR_Byte( 0xA6, OLED_CMD ); //正常显示
}
if( i == 1 )
{
OLED_WR_Byte( 0xA7, OLED_CMD ); //反色显示
}
}
//屏幕旋转180度
void OLED_DisplayTurn( u8 i )
{
if( i == 0 )
{
OLED_WR_Byte( 0xC8, OLED_CMD ); //正常显示
OLED_WR_Byte( 0xA1, OLED_CMD );
}
if( i == 1 )
{
OLED_WR_Byte( 0xC0, OLED_CMD ); //反转显示
OLED_WR_Byte( 0xA0, OLED_CMD );
}
}
//延时
void IIC_delay( void )
{
u8 t = 1;
while( t-- );
}
//起始信号
void I2C_Start( void )
{
OLED_SDA_Set();
OLED_SCL_Set();
IIC_delay();
OLED_SDA_Clr();
IIC_delay();
OLED_SCL_Clr();
}
//结束信号
void I2C_Stop( void )
{
OLED_SDA_Clr();
OLED_SCL_Set();
IIC_delay();
OLED_SDA_Set();
}
//等待信号响应
void I2C_WaitAck( void ) //测数据信号的电平
{
OLED_SDA_Set();
IIC_delay();
OLED_SCL_Set();
IIC_delay();
OLED_SCL_Clr();
IIC_delay();
}
//写入一个字节
void Send_Byte( u8 dat )
{
u8 i;
for( i = 0; i < 8; i++ )
{
OLED_SCL_Clr();//将时钟信号设置为低电平
if( dat & 0x80 ) //将dat的8位从最高位依次写入
{
OLED_SDA_Set();
}
else
{
OLED_SDA_Clr();
}
IIC_delay();
OLED_SCL_Set();
IIC_delay();
OLED_SCL_Clr();
dat <<= 1;
}
}
//发送一个字节
//向SSD1306写入一个字节。
//mode:数据/命令标志 0,表示命令;1,表示数据;
void OLED_WR_Byte( u8 dat, u8 mode )
{
I2C_Start();
Send_Byte( 0x78 );
I2C_WaitAck();
if( mode )
{
Send_Byte( 0x40 );
}
else
{
Send_Byte( 0x00 );
}
I2C_WaitAck();
Send_Byte( dat );
I2C_WaitAck();
I2C_Stop();
}
//坐标设置
void OLED_Set_Pos( u8 x, u8 y )
{
OLED_WR_Byte( 0xb0 + y, OLED_CMD );
OLED_WR_Byte( ( ( x & 0xf0 ) >> 4 ) | 0x10, OLED_CMD );
OLED_WR_Byte( ( x & 0x0f ), OLED_CMD );
}
//开启OLED显示
void OLED_Display_On( void )
{
OLED_WR_Byte( 0X8D, OLED_CMD ); //SET DCDC命令
OLED_WR_Byte( 0X14, OLED_CMD ); //DCDC ON
OLED_WR_Byte( 0XAF, OLED_CMD ); //DISPLAY ON
}
//关闭OLED显示
void OLED_Display_Off( void )
{
OLED_WR_Byte( 0X8D, OLED_CMD ); //SET DCDC命令
OLED_WR_Byte( 0X10, OLED_CMD ); //DCDC OFF
OLED_WR_Byte( 0XAE, OLED_CMD ); //DISPLAY OFF
}
//清屏函数,清完屏,整个屏幕是黑色的!和没点亮一样!!!
void OLED_Clear( void )
{
u8 i, n;
for( i = 0; i < 4; i++ )
{
OLED_WR_Byte( 0xb0 + i, OLED_CMD ); //设置页地址(0~7)
OLED_WR_Byte( 0x00, OLED_CMD ); //设置显示位置—列低地址
OLED_WR_Byte( 0x10, OLED_CMD ); //设置显示位置—列高地址
for( n = 0; n < 128; n++ )
{
OLED_WR_Byte( 0, OLED_DATA );
}
} //更新显示
}
//在指定位置显示一个字符,包括部分字符
//x:0~127
//y:0~63
//sizey:选择字体 6x8 8x16
void OLED_ShowChar( u8 x, u8 y, u8 chr, u8 sizey )
{
u8 c = 0, sizex = sizey / 2;
u16 i = 0, size1;
if( sizey == 8 )
{
size1 = 6;
}
else
{
size1 = ( sizey / 8 + ( ( sizey % 8 ) ? 1 : 0 ) ) * ( sizey / 2 );
}
c = chr - ' '; //得到偏移后的值
OLED_Set_Pos( x, y );
for( i = 0; i < size1; i++ )
{
if( i % sizex == 0 && sizey != 8 )
{
OLED_Set_Pos( x, y++ );
}
if( sizey == 8 )
{
OLED_WR_Byte( asc2_0806[c][i], OLED_DATA ); //6X8字号
}
else if( sizey == 16 )
{
OLED_WR_Byte( asc2_1608[c][i], OLED_DATA ); //8x16字号
}
// else if(sizey==xx) OLED_WR_Byte(asc2_xxxx[c][i],OLED_DATA);//用户添加字号
else
{
return;
}
}
}
//m^n函数
u32 oled_pow( u8 m, u8 n )
{
u32 result = 1;
while( n-- )
{
result *= m;
}
return result;
}
//显示数字
//x,y :起点坐标
//num:要显示的数字
//len :数字的位数
//sizey:字体大小
void OLED_ShowNum( u8 x, u8 y, u32 num, u8 len, u8 sizey )
{
u8 t, temp, m = 0;
u8 enshow = 0;
if( sizey == 8 )
{
m = 2;
}
for( t = 0; t < len; t++ )
{
temp = ( num / oled_pow( 10, len - t - 1 ) ) % 10;
if( enshow == 0 && t < ( len - 1 ) )
{
if( temp == 0 )
{
OLED_ShowChar( x + ( sizey / 2 + m )*t, y, ' ', sizey );
continue;
}
else
{
enshow = 1;
}
}
OLED_ShowChar( x + ( sizey / 2 + m )*t, y, temp + '0', sizey );
}
}
//显示一个字符号串
void OLED_ShowString( u8 x, u8 y, u8* chr, u8 sizey )
{
u8 j = 0;
while( chr[j] != '\0' )
{
OLED_ShowChar( x, y, chr[j++], sizey );
if( sizey == 8 )
{
x += 6;
}
else
{
x += sizey / 2;
}
}
}
//显示汉字
void OLED_ShowChinese( u8 x, u8 y, u8 *temp, u8 sizey )
{
u16 i, size1 = ( sizey / 8 + ( ( sizey % 8 ) ? 1 : 0 ) ) * sizey;
for( i = 0; i < size1; i++ )
{
if( i % sizey == 0 )
{
OLED_Set_Pos( x, y++ );
}
if( sizey == 16 )
{
OLED_WR_Byte( temp[i], OLED_DATA ); //16x16字号
}
// else if(sizey==xx) OLED_WR_Byte(xxx[c][i],OLED_DATA);//用户添加字号
else
{
return;
}
}
}
//----------------------------------------------------------
//显示一个gb2312汉字(从字库读数据)
void OLED_ShowGB2312(unsigned char x,unsigned char y,unsigned char textH,unsigned char textL)
{
unsigned char fontbuf[32];
S1Y_gt16_GetData(textH,textL,fontbuf);
OLED_ShowChinese( x, y, fontbuf, 16 );
// OLED_Set_Pos(x,y);
// for(t=0;t<16;t++)
// OLED_WR_Byte(fontbuf[t],OLED_DATA);
// OLED_Set_Pos(x,y+1);
// for(t=0;t<16;t++)
// OLED_WR_Byte(fontbuf[t+16],OLED_DATA);
}
void OLED_ShowStr(unsigned char x,unsigned char y,unsigned char* text)
{
unsigned char i=0;
while((text[i]>0x00))
{
if(((text[i]>=0xb0) &&(text[i]<=0xf7))&&(text[i+1]>=0xa1))
{
OLED_ShowGB2312(x,y,text[i],text[i+1]);
i+=2;
x+=16;
}
else if((text[i]>=0x20) && (text[i]<=0x7e))
{
//OLED_ShowASCII(x,y,text[i]);
i+=1;
x+=8;
}
else
i++;
}
}
//--------------------------------------------------------------------
//显示图片
//x,y显示坐标
//sizex,sizey,图片长宽
//BMP:要显示的图片
void OLED_DrawBMP( u8 x, u8 y, u8 sizex, u8 sizey, u8 BMP[] )
{
u16 j = 0;
u8 i, m;
sizey = sizey / 8 + ( ( sizey % 8 ) ? 1 : 0 );
for( i = 0; i < sizey; i++ )
{
OLED_Set_Pos( x, i + y );
for( m = 0; m < sizex; m++ )
{
OLED_WR_Byte( BMP[j++], OLED_DATA );
}
}
}
//初始化
void OLED_Init( void )
{
GPIO_Init( OLED_PORT,OLED_SCL,GPIO_Mode_Out_PP_High_Fast );
GPIO_Init( OLED_PORT,OLED_SDA,GPIO_Mode_Out_PP_High_Fast );
OLED_WR_Byte( 0xAE, OLED_CMD ); /*display off*/
OLED_WR_Byte( 0x00, OLED_CMD ); /*set lower column address*/
OLED_WR_Byte( 0x10, OLED_CMD ); /*set higher column address*/
OLED_WR_Byte( 0x00, OLED_CMD ); /*set display start line*/
OLED_WR_Byte( 0xB0, OLED_CMD ); /*set page address*/
OLED_WR_Byte( 0x81, OLED_CMD ); /*contract control*/
OLED_WR_Byte( 0xff, OLED_CMD ); /*128*/
OLED_WR_Byte( 0xA1, OLED_CMD ); /*set segment remap*/
OLED_WR_Byte( 0xA6, OLED_CMD ); /*normal / reverse*/
OLED_WR_Byte( 0xA8, OLED_CMD ); /*multiplex ratio*/
OLED_WR_Byte( 0x1F, OLED_CMD ); /*duty = 1/32*/
OLED_WR_Byte( 0xC8, OLED_CMD ); /*Com scan direction*/
OLED_WR_Byte( 0xD3, OLED_CMD ); /*set display offset*/
OLED_WR_Byte( 0x00, OLED_CMD );
OLED_WR_Byte( 0xD5, OLED_CMD ); /*set osc division*/
OLED_WR_Byte( 0x80, OLED_CMD );
OLED_WR_Byte( 0xD9, OLED_CMD ); /*set pre-charge period*/
OLED_WR_Byte( 0x1f, OLED_CMD );
OLED_WR_Byte( 0xDA, OLED_CMD ); /*set COM pins*/
OLED_WR_Byte( 0x00, OLED_CMD );
OLED_WR_Byte( 0xdb, OLED_CMD ); /*set vcomh*/
OLED_WR_Byte( 0x40, OLED_CMD );
OLED_WR_Byte( 0x8d, OLED_CMD ); /*set charge pump enable*/
OLED_WR_Byte( 0x14, OLED_CMD );
OLED_Clear();
OLED_WR_Byte( 0xAF, OLED_CMD ); /*display ON*/
}
//-----------------------------------分割线------------------------------------------
// .h文件
#ifndef __OLED_H
#define __OLED_H
#include "stm8l15x.h"//STM8L051/151等系列共用库函数
#define u8 unsigned char
#define u16 unsigned int
#define u32 unsigned int
#define OLED_CMD 0 //写命令
#define OLED_DATA 1 //写数据
#define OLED_PORT GPIOC//SCL
老板子线:
//#define OLED_SCL GPIO_Pin_1//SCL
//#define OLED_SDA GPIO_Pin_0//SDA
// 新板子线:
#define OLED_SCL GPIO_Pin_0//SCL
#define OLED_SDA GPIO_Pin_1//SDA
//-----------------OLED端口定义----------------
#define OLED_SCL_Clr() GPIO_ResetBits(OLED_PORT, OLED_SCL)//SCL IIC接口的数据信号
#define OLED_SCL_Set() GPIO_SetBits(OLED_PORT, OLED_SCL)
#define OLED_SDA_Clr() GPIO_ResetBits(OLED_PORT, OLED_SDA )//SDA IIC接口的时钟信号
#define OLED_SDA_Set() GPIO_SetBits(OLED_PORT, OLED_SDA )
//--------------------------------------------------
//OLED控制用函数
void delay_ms(unsigned int ms);
void OLED_ColorTurn(u8 i);
void OLED_DisplayTurn(u8 i);
void OLED_WR_Byte(u8 dat,u8 cmd);
void OLED_Set_Pos(u8 x, u8 y);
void OLED_Display_On(void);
void OLED_Display_Off(void);
void OLED_Clear(void);
void OLED_ShowChar(u8 x,u8 y,u8 chr,u8 sizey);
u32 oled_pow(u8 m,u8 n);
void OLED_ShowNum(u8 x,u8 y,u32 num,u8 len,u8 sizey);
void OLED_ShowString(u8 x,u8 y,u8 *chr,u8 sizey);
//void OLED_ShowChinese(u8 x,u8 y,u8 no,u8 sizey);
void OLED_ShowChinese( u8 x, u8 y, u8 *temp, u8 sizey );
void OLED_DrawBMP(u8 x,u8 y,u8 sizex, u8 sizey,u8 BMP[]);
void OLED_Init(void);
void OLED_ShowStr(unsigned char x,unsigned char y,unsigned char* text);
#endif
最后是调用过程伪代码:
// 调用显示任意汉字字符串伪代码段
void main(void)
{
if(key.code == true)
{
OLED_Clear();
OLED_ShowStr( 0, 0, tab[HZ] );// tab[]里面存放的是汉字字符串,例如:"华温冷链巴拉巴拉"
Delay( 100 );
}
}
最后的最后,刚忙完,写博客的时候有点累,有任何不明白的地方,欢迎一起探讨 QQ741684134。PS:遇到很多网友找我聊很基础的单片机应用问题,请自行百度,因为实在很忙。
本文地址:https://blog.csdn.net/xuechengchang/article/details/110181168
上一篇: 网易云创建添加推荐歌谱和删除操作
下一篇: 春节初一的风俗有哪些