连续两个月的加班,给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:遇到很多网友找我聊很基础的单片机应用问题,请自行百度,因为实在很忙。
联系客服