核心代码由
kingsraywii
提供,本文作者对其进行整合、更详细地注释和部分代码改进, 添加了
芯片ID获取
、字符串输出和
printf重定向输出
功能
注:使用普中科技开发板测试时,需要拔掉Boot1插口 接5V电压,重启
/*******************************************************************************
*
* 软件功能: 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_i***usy(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_i***usy()); //检测忙信号
Lcd_Write(lcd_COMMAND,0x80); //第一行第一个字开始
for(i=0;display_1
!=' ';i++){ while(Lcd_i***usy()); //检测忙信号 Lcd_Write(lcd_DATA,display_1); //delay_ms(500); } //while(Lcd_i***usy()); //检测忙信号 //Lcd_Write(lcd_COMMAND,0xC0); //第二行第一个字开始 //Lcd_WriteString("www.llqqww.com"); getSys_ID(); /*for(i=0;display_2!=' ';i++){ Lcd_Write(lcd_DATA,display_2); 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 drain GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz; //输出的最大频率为50HZ GPIO_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 us
lcd_GPIO->BSRR |= (buf<<8) & 0xff00; //利用BSRR寄存器,将需要置1 的口置1 lcd_GPIO->BRR |= ((~buf)<<8) & 0xff00; //利用BRR寄存器,将不要置1,即为0的口置为0 delay_us(2); //根据使用手册数据建立时间20ns 这里延长延时至1 us
lcd_Enable(1); //使能写入 delay_us(2); //根据使用手册脉冲宽度为150ns 这里用了1 us lcd_Enable(0); //结束写入 delay_ms(15); } void Lcd_WriteString(u8* buf) { int i=0; for(i=0;buf!=' ';i++){ while(Lcd_i***usy()); //检测忙信号 Lcd_Write(lcd_DATA,buf); //delay_ms(500); } } int Lcd_i***usy() { 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、延时15ms Lcd_Write(lcd_COMMAND,0x38); //2、写指令38H(不检测忙信号) delay_ms(5); //3、延时5ms Lcd_Write(lcd_COMMAND,0x38); //4、写指令38H(不检测忙信号) delay_ms(5); //5、延时5ms Lcd_Write(lcd_COMMAND,0x38); //6、写指令38H(不检测忙信号) //7、以后每次写指令、读/写数据之前均需检测忙信号 //delay_ms(15); //延时15ms while(Lcd_i***usy()); //检测忙信号 Lcd_Write(lcd_COMMAND,0x38); //8、写指令38H 显示模式设置 设置16X2显示 5X7点阵 8位数据接口
//delay_ms(15); //延时15ms while(Lcd_i***usy()); //检测忙信号 Lcd_Write(lcd_COMMAND,0x0c); //9.显示开/关,及光标设置 此处为0000 1100 即开显示、不显示光标、光标不闪烁 /* 指令码 0000 1DCB D=1 开显示 D=0 关显示 C=1 显示光标 C=0 不显示光标 B=1光标闪烁 B=0 光标不闪烁 */ //delay_ms(15); //延时15ms while(Lcd_i***usy()); //检测忙信号 Lcd_Write(lcd_COMMAND,0x06); //10.写一个指针加1,即控制从左至右显示还是从右至左 此处为0000 0110 /* 指令码 0000 01NS N=1 读/写一个数据后,指针和光标加1 N=0 读/写一个数据后,指针和光标减1 S=1 写入新数据后显示屏整体移1个字符 S=0 写入新数据后显示屏不移动 */ //delay_ms(15); //延时15ms while(Lcd_i***usy()); //检测忙信号 Lcd_Write(lcd_COMMAND,0x01); //11.清屏 }
int fputc(int ch, FILE *f) { while(Lcd_i***usy()); //检测忙信号 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=*(u8*)(0x1FFFF7E8+i); //产品唯一身份标识寄存器(96位) printf("%0.2X",Sys_ID); if(i==3) { while(Lcd_i***usy()); //检测忙信号 Lcd_Write(lcd_COMMAND,0xC0); //第二行第一个字开始 } }
}
|