新闻中心

EEPW首页 > 嵌入式系统 > 设计应用 > STM32单片机学习(9) LCD1602显示屏输出实验

STM32单片机学习(9) LCD1602显示屏输出实验

作者: 时间:2016-11-19 来源:网络 收藏
核心代码由kingsraywii提供,本文作者对其进行整合、更详细地注释和部分代码改进, 添加了芯片ID获取、字符串输出和printf重定向输出功能

注:使用普中科技开发板测试时,需要拔掉Boot1插口 接5V电压,重启

本文引用地址:https://www.eepw.com.cn/article/201611/318258.htm

/**	* 软件功能:	LCD1602实验(软件延时方式)* */#include "stm32f10x.h"#include "delay.h"#include /*定义宏变量lcd_RS = data/command(H/L) 数据/命令操作lcd_RW = read/write(H/L)   读写操作lcd_Enable 使能操作lcd_GPIO lcd1602接在了GPIOX口*///接口#define lcd_GPIO 		GPIOB#define lcd_busyflag    GPIO_Pin_15 //DB7对应的口#define lcd_pin_RS 		GPIO_Pin_1			 #define lcd_pin_RW 		GPIO_Pin_2#define lcd_pin_Enable 	GPIO_Pin_0//状态#define lcd_DATA 	1#define lcd_COMMAND 0#define lcd_READ 	1#define lcd_WRITE 	0//功能#define lcd_RS(n)	  	n ? GPIO_SetBits(lcd_GPIO,lcd_pin_RS) : GPIO_ResetBits(lcd_GPIO,lcd_pin_RS)#define lcd_RW(n)		n ? GPIO_SetBits(lcd_GPIO,lcd_pin_RW) : GPIO_ResetBits(lcd_GPIO,lcd_pin_RW)#define lcd_Enable(n)   n ? GPIO_SetBits(lcd_GPIO,lcd_pin_Enable) : GPIO_ResetBits(lcd_GPIO,lcd_pin_Enable)void RCC_Configuration(void);void GPIO_Configuration(void);void Lcd_Write(u8 type,u8 buf); void Lcd_WriteString(u8 *buf);int Lcd_isbusy(void);void Lcd_Init(void);int fputc(int ch, FILE *f);void getSys_ID(void);/*函数: int main(void)功能: main主函数参数: 无返回: 无/int main(void){int i=0;unsigned char* display_1={"chip ID:"};//unsigned char* display_2={"www.llqqww.com"};RCC_Configuration();GPIO_Configuration();delay_init(72);Lcd_Init();//while(1){while(Lcd_isbusy());       //检测忙信号Lcd_Write(lcd_COMMAND,0x80);    //第一行第一个字开始for(i=0;display_1[i]!=;i++){while(Lcd_isbusy());       //检测忙信号Lcd_Write(lcd_DATA,display_1[i]);//delay_ms(500);}//while(Lcd_isbusy());       //检测忙信号//Lcd_Write(lcd_COMMAND,0xC0); //第二行第一个字开始//Lcd_WriteString("www.llqqww.com");getSys_ID();/*for(i=0;display_2[i]!=;i++){Lcd_Write(lcd_DATA,display_2[i]);delay_ms(500);} *///	Lcd_Write(lcd_COMMAND,0x01);	 //清屏//	} }/*函数: void RCC_Configuration(void)功能: 复位和时钟控制 配置*/void RCC_Configuration(void){ErrorStatus HSEStartUpStatus;                    //定义外部高速晶体启动状态枚举变量RCC_DeInit();                                    //复位RCC外部设备寄存器到默认值RCC_HSEConfig(RCC_HSE_ON);                       //打开外部高速晶振HSEStartUpStatus = RCC_WaitForHSEStartUp();      //等待外部高速时钟准备好if(SUCCESS == HSEStartUpStatus)                  //外部高速时钟已经准别好{FLASH_PrefetchBufferCmd(FLASH_PrefetchBuffer_Enable); //开启FLASH预读缓冲功能,加速FLASH的读取。所有程序中必须的用法.位置:RCC初始化子函数里面,时钟起振之后FLASH_SetLatency(FLASH_Latency_2);                    //flash操作的延时RCC_PLLConfig(RCC_PLLSource_HSE_Div1, RCC_PLLMul_9);  //配置PLL时钟 == 外部高速晶体时钟(16MHz)  * 9 = 72MHz	 RCC_PLLCmd(ENABLE);                                   //使能PLL时钟RCC_HCLKConfig(RCC_SYSCLK_Div1);               //配置AHB(HCLK)时钟等于==SYSCLK(系统时钟)RCC_PCLK2Config(RCC_HCLK_Div1);                //配置APB2(PCLK2)钟==AHB时钟RCC_PCLK1Config(RCC_HCLK_Div2);                //配置APB1(PCLK1)钟==AHB1/2时钟while(RCC_GetFlagStatus(RCC_FLAG_PLLRDY) == RESET){}    //等待PLL时钟就绪RCC_SYSCLKConfig(RCC_SYSCLKSource_PLLCLK);            //配置系统时钟 = PLL时钟while(RCC_GetSYSCLKSource() != 0x08){}                  //检查PLL时钟是否作为系统时钟}RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOB | RCC_APB2Periph_AFIO, ENABLE);  //允许GPIOB、AFIO时钟}/*函数: void GPIO_Configuration(void)功能: GPIO初始化*/void GPIO_Configuration(void){GPIO_InitTypeDef GPIO_InitStructure;        //定义GPIO初始化结构体GPIO_InitStructure.GPIO_Pin = GPIO_Pin_All  ;  //设置初始化引脚GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz; //设置响应速率GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP;  //设置I/O模式GPIO_Init(GPIOB, &GPIO_InitStructure); //调用MDK初始化GPIOX口GPIO_InitStructure.GPIO_Pin = GPIO_Pin_8 | GPIO_Pin_9 | GPIO_Pin_10 | GPIO_Pin_11 |GPIO_Pin_12 | GPIO_Pin_13 | GPIO_Pin_14 | GPIO_Pin_15; //所有GPIO为同一类型端口GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_OD;	 //open drainGPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;	 //输出的最大频率为50HZGPIO_Init(GPIOB, &GPIO_InitStructure);   //初始化GPIOB端口 }/*函数名:Lcd_Write功能:向lcd1602模块写入命令/数据参数1: type:lcd_COMMAND	写入命令lcd_DATA	写入数据参数2: buf 8个二进制位的命令/数据*/void Lcd_Write(u8 type,u8 buf) {lcd_RS(type);lcd_RW(lcd_WRITE);		  //写入 命令/数据操作lcd_Enable(0);delay_us(2); //根据使用手册地址建立时间30ns 这里延长延时至1 uslcd_GPIO->BSRR |= (buf<<8) & 0xff00;     	//利用BSRR寄存器,将需要置1 的口置1lcd_GPIO->BRR  |= ((~buf)<<8) & 0xff00;	   	//利用BRR寄存器,将不要置1,即为0的口置为0delay_us(2);  //根据使用手册数据建立时间20ns 这里延长延时至1 uslcd_Enable(1); //使能写入delay_us(2);  //根据使用手册脉冲宽度为150ns 这里用了1 uslcd_Enable(0); //结束写入delay_ms(15);}void Lcd_WriteString(u8* buf){int i=0;for(i=0;buf[i]!=;i++){while(Lcd_isbusy());       //检测忙信号Lcd_Write(lcd_DATA,buf[i]);//delay_ms(500);}	}int Lcd_isbusy(){int result;lcd_RS(lcd_COMMAND);lcd_RW(lcd_READ);lcd_Enable(1);delay_us(2);//result = GPIO_ReadInputDataBit(lcd_GPIO,lcd_busyflag);result = (GPIO_ReadInputData(lcd_GPIO) & 0x8000)>>8;  lcd_Enable(0);return result;}/*初始化LCD1602*/void Lcd_Init(){delay_ms(15);					//1、延时15msLcd_Write(lcd_COMMAND,0x38);	//2、写指令38H(不检测忙信号)delay_ms(5);		 			//3、延时5msLcd_Write(lcd_COMMAND,0x38);	//4、写指令38H(不检测忙信号)delay_ms(5);					//5、延时5msLcd_Write(lcd_COMMAND,0x38);    //6、写指令38H(不检测忙信号) //7、以后每次写指令、读/写数据之前均需检测忙信号							  //delay_ms(15);					//延时15mswhile(Lcd_isbusy());		    //检测忙信号Lcd_Write(lcd_COMMAND,0x38);	//8、写指令38H  显示模式设置 设置16X2显示 5X7点阵 8位数据接口//delay_ms(15);					//延时15mswhile(Lcd_isbusy());		    //检测忙信号Lcd_Write(lcd_COMMAND,0x0c);    //9.显示开/关,及光标设置 此处为0000 1100 即开显示、不显示光标、光标不闪烁/* 指令码 0000 1DCBD=1 开显示 D=0 关显示  C=1 显示光标 C=0 不显示光标 B=1光标闪烁 B=0 光标不闪烁*///delay_ms(15);					//延时15mswhile(Lcd_isbusy());		    //检测忙信号	 Lcd_Write(lcd_COMMAND,0x06);    //10.写一个指针加1,即控制从左至右显示还是从右至左 此处为0000 0110/* 指令码 0000 01NSN=1 读/写一个数据后,指针和光标加1 N=0 读/写一个数据后,指针和光标减1  S=1 写入新数据后显示屏整体移1个字符  S=0 写入新数据后显示屏不移动*///delay_ms(15);				    //延时15mswhile(Lcd_isbusy());		    //检测忙信号Lcd_Write(lcd_COMMAND,0x01);    //11.清屏}int fputc(int ch, FILE *f){while(Lcd_isbusy());       //检测忙信号Lcd_Write(lcd_DATA,((u8)ch));return (ch);}void getSys_ID(void){u8  Sys_ID[12],i;for(i=0;i<12;i++){Sys_ID[i]=*(u8*)(0x1FFFF7E8+i);  //产品唯一身份标识寄存器(96位)printf("%0.2X",Sys_ID[i]); if(i==3){while(Lcd_isbusy());       //检测忙信号Lcd_Write(lcd_COMMAND,0xC0); //第二行第一个字开始}}}



评论


技术专区

关闭