STM32单片机- LCD1602显示屏输出实验

STM32单片机- LCD1602显示屏输出实验
核心代码由kingsraywii提供,本文作者对其进行整合、更详细地注释和部分代码改进, 添加了芯片ID获取、字符串输出和printf重定向输出功能

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

view plaincopyprint?

  1. /*******************************************************************************
    • 软件功能: LCD1602实验(软件延时方式)
  2. *******************************************************************************/
  3. #include “stm32f10x.h”
  4. #include “delay.h”
  5. #include
  6. /*
  7. 定义宏变量
  8. lcd_RS = data/command(H/L) 数据/命令操作
  9. .lcd_RW = read/write(H/L) 读写操作
  10. lcd_Enable 使能操作
  11. lcd_GPIO lcd1602接在了GPIOX口
  12. */
  13. //接口
  14. #define lcd_GPIO GPIOB
  15. #define lcd_busyflag GPIO_Pin_15 //DB7对应的口
  16. #define lcd_pin_RS GPIO_Pin_1
  17. #define lcd_pin_RW GPIO_Pin_2
  18. #define lcd_pin_Enable GPIO_Pin_0
  19. //状态
  20. #define lcd_DATA 1
  21. #define lcd_COMMAND 0
  22. #define lcd_READ 1
  23. #define lcd_WRITE 0
  24. //功能
  25. #define lcd_RS(n) n ? GPIO_SetBits(lcd_GPIO,lcd_pin_RS) : GPIO_ResetBits(lcd_GPIO,lcd_pin_RS)
  26. #define lcd_RW(n) n ? GPIO_SetBits(lcd_GPIO,lcd_pin_RW) : GPIO_ResetBits(lcd_GPIO,lcd_pin_RW)
  27. #define lcd_Enable(n) n ? GPIO_SetBits(lcd_GPIO,lcd_pin_Enable) : GPIO_ResetBits(lcd_GPIO,lcd_pin_Enable)
  28. void RCC_Configuration(void);
  29. void GPIO_Configuration(void);
  30. void Lcd_Write(u8 type,u8 buf);
  31. void Lcd_WriteString(u8 *buf);
  32. int Lcd_isbusy(void);
  33. void Lcd_Init(void);
  34. int fputc(int ch, FILE *f);
  35. void getSys_ID(void);
  36. /*************************************************
  37. 函数: int main(void)
  38. 功能: main主函数
  39. 参数: 无
  40. 返回: 无
  41. **************************************************/
  42. int main(void)
  43. {
  44. int i=0;
  45. unsigned char* display_1={“chip ID:”};
  46. //unsigned char* display_2={“www.llqqww.com”};
  47. RCC_Configuration();
  48. GPIO_Configuration();
  49. delay_init(72);
  50. Lcd_Init();
  51. //while(1){
  52. while(Lcd_isbusy()); //检测忙信号
  53. Lcd_Write(lcd_COMMAND,0x80); //第一行第一个字开始
  54. for(i=0;display_1[i]!=’\0’;i++){
  55. while(Lcd_isbusy()); //检测忙信号
  56. Lcd_Write(lcd_DATA,display_1[i]);
  57. //delay_ms(500);
  58. }
  59. //while(Lcd_isbusy()); //检测忙信号
  60. //Lcd_Write(lcd_COMMAND,0xC0); //第二行第一个字开始
  61. //Lcd_WriteString(“www.llqqww.com”);
  62. getSys_ID();
  63. /*for(i=0;display_2[i]!=’\0’;i++){
  64. Lcd_Write(lcd_DATA,display_2[i]);
  65. delay_ms(500);
  66. } */
  67. // Lcd_Write(lcd_COMMAND,0x01); //清屏
  68. // }
  69. }
  70. /*
  71. 函数: void RCC_Configuration(void)
  72. 功能: 复位和时钟控制 配置
  73. */
  74. void RCC_Configuration(void)
  75. {
  76. ErrorStatus HSEStartUpStatus; //定义外部高速晶体启动状态枚举变量
  77. RCC_DeInit(); //复位RCC外部设备寄存器到默认值
  78. RCC_HSEConfig(RCC_HSE_ON); //打开外部高速晶振
  79. HSEStartUpStatus = RCC_WaitForHSEStartUp(); //等待外部高速时钟准备好
  80. if(SUCCESS == HSEStartUpStatus) //外部高速时钟已经准别好
  81. {
  82. FLASH_PrefetchBufferCmd(FLASH_PrefetchBuffer_Enable); //开启FLASH预读缓冲功能,加速FLASH的读取。所有程序中必须的用法.位置:RCC初始化子函数里面,时钟起振之后
  83. FLASH_SetLatency(FLASH_Latency_2); //flash操作的延时
  84. RCC_PLLConfig(RCC_PLLSource_HSE_Div1, RCC_PLLMul_9); //配置PLL时钟 == 外部高速晶体时钟(16MHz) * 9 = 72MHz
  85. RCC_PLLCmd(ENABLE); //使能PLL时钟
  86. RCC_HCLKConfig(RCC_SYSCLK_Div1); //配置AHB(HCLK)时钟等于==SYSCLK(系统时钟)
  87. RCC_PCLK2Config(RCC_HCLK_Div1); //配置APB2(PCLK2)钟==AHB时钟
  88. RCC_PCLK1Config(RCC_HCLK_Div2); //配置APB1(PCLK1)钟==AHB1/2时钟
  89. while(RCC_GetFlagStatus(RCC_FLAG_PLLRDY) == RESET){} //等待PLL时钟就绪
  90. RCC_SYSCLKConfig(RCC_SYSCLKSource_PLLCLK); //配置系统时钟 = PLL时钟
  91. while(RCC_GetSYSCLKSource() != 0x08){} //检查PLL时钟是否作为系统时钟
  92. }

109 RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOB | RCC_APB2Periph_AFIO, ENABLE); //允许GPIOB、AFIO时钟
110. }
111.
112. /*
113. 函数: void GPIO_Configuration(void)
114. 功能: GPIO初始化
115. /
116. void GPIO_Configuration(void)
117. {
118.
119. GPIO_InitTypeDef GPIO_InitStructure; //定义GPIO初始化结构体
120.
121. GPIO_InitStructure.GPIO_Pin = GPIO_Pin_All ; //设置初始化引脚
122. GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz; //设置响应速率
123. GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP; //设置I/O模式
124. GPIO_Init(GPIOB, &GPIO_InitStructure); //调用MDK初始化GPIOX口
125.
126. 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为同一类型端口
127. GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_OD; //open drain
128. GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz; //输出的最大频率为50HZ
129. GPIO_Init(GPIOB, &GPIO_InitStructure); //初始化GPIOB端口
130.
131. }
132.
133.
134.
135. /

136. 函数名:Lcd_Write
137. 功能:向lcd1602模块写入命令/数据
138. 参数1: type:
139. lcd_COMMAND 写入命令
140. lcd_DATA 写入数据
141. 参数2: buf 8个二进制位的命令/数据
142. /
143. void Lcd_Write(u8 type,u8 buf)
144. {
145. lcd_RS(type);
146. lcd_RW(lcd_WRITE); //写入 命令/数据操作
147. lcd_Enable(0);
148. delay_us(2); //根据使用手册地址建立时间30ns 这里延长延时至1 us
149.
150. lcd_GPIO->BSRR |= (buf<<8) & 0xff00; //利用BSRR寄存器,将需要置1 的口置1
151. lcd_GPIO->BRR |= ((~buf)<<8) & 0xff00; //利用BRR寄存器,将不要置1,即为0的口置为0
152. delay_us(2); //根据使用手册数据建立时间20ns 这里延长延时至1 us
153.
154. lcd_Enable(1); //使能写入
155. delay_us(2); //根据使用手册脉冲宽度为150ns 这里用了1 us
156. lcd_Enable(0); //结束写入
157. delay_ms(15);
158. }
159. void Lcd_WriteString(u8
buf)
160. {
161. int i=0;
162. for(i=0;buf[i]!=’\0’;i++){
163. while(Lcd_isbusy()); //检测忙信号
164. Lcd_Write(lcd_DATA,buf[i]);
165. //delay_ms(500);
166. }
167. }
168. int Lcd_isbusy()
169. {
170. int result;
171. lcd_RS(lcd_COMMAND);
172. lcd_RW(lcd_READ);
173. lcd_Enable(1);
174. delay_us(2);
175. //result = GPIO_ReadInputDataBit(lcd_GPIO,lcd_busyflag);
176. result = (GPIO_ReadInputData(lcd_GPIO) & 0x8000)>>8;
177.
178. lcd_Enable(0);
179. return result;
180. }
181. /*
182. 初始化LCD1602
183. /
184. void Lcd_Init()
185. {
186. delay_ms(15); //1、延时15ms
187. Lcd_Write(lcd_COMMAND,0x38); //2、写指令38H(不检测忙信号)
188. delay_ms(5); //3、延时5ms
189. Lcd_Write(lcd_COMMAND,0x38); //4、写指令38H(不检测忙信号)
190. delay_ms(5); //5、延时5ms
191. Lcd_Write(lcd_COMMAND,0x38); //6、写指令38H(不检测忙信号)
192. //7、以后每次写指令、读/写数据之前均需检测忙信号
193. //delay_ms(15); //延时15ms
194. while(Lcd_isbusy()); //检测忙信号
195. Lcd_Write(lcd_COMMAND,0x38); //8、写指令38H 显示模式设置 设置16X2显示 5X7点阵 8位数据接口
196.
197. //delay_ms(15); //延时15ms
198. while(Lcd_isbusy()); //检测忙信号
199. Lcd_Write(lcd_COMMAND,0x0c); //9.显示开/关,及光标设置 此处为0000 1100 即开显示、不显示光标、光标不闪烁
200. /
指令码 0000 1DCB
201. D=1 开显示 D=0 关显示 C=1 显示光标 C=0 不显示光标 B=1光标闪烁 B=0 光标不闪烁
202. /
203. //delay_ms(15); //延时15ms
204. while(Lcd_isbusy()); //检测忙信号
205. Lcd_Write(lcd_COMMAND,0x06); //10.写一个指针加1,即控制从左至右显示还是从右至左 此处为0000 0110
206. /
指令码 0000 01NS
207. N=1 读/写一个数据后,指针和光标加1 N=0 读/写一个数据后,指针和光标减1
208. S=1 写入新数据后显示屏整体移1个字符 S=0 写入新数据后显示屏不移动
209. /
210. //delay_ms(15); //延时15ms
211. while(Lcd_isbusy()); //检测忙信号
212. Lcd_Write(lcd_COMMAND,0x01); //11.清屏
213. }
214.
215. int fputc(int ch, FILE f)
216. {
217. while(Lcd_isbusy()); //检测忙信号
218. Lcd_Write(lcd_DATA,((u8)ch));
219. return (ch);
220. }
221.
222. void getSys_ID(void)
223. {
224. u8 Sys_ID[12],i;
225. for(i=0;i<12;i++)
226. {
227. Sys_ID[i]=
(u8
)(0x1FFFF7E8+i); //产品唯一身份标识寄存器(96位)
228. printf("%0.2X",Sys_ID[i]);
229. if(i==3)
230. {
231. while(Lcd_isbusy()); //检测忙信号
232. Lcd_Write(lcd_COMMAND,0xC0); //第二行第一个字开始
233. }
234. }
235.
236. }

你可能感兴趣的:(STM32单片机)