In previous post, I showed about the LCD interfacing in 4-bit mode. The HD44780 allow the micro-processor to send the 8-bit data or command using an 4-bit interfacing. This process requires the microprocessor to send the 8-bit command or data twice over the 4-bit wire. The microprocessor send the higher nibble D4...D7 first, and then the lower nibble D0...D3. This technique saves the number of microprocessor digital output pins and circuit wiring. For more information about 4-bit LCD interfacing you can read this article.
Circuit Wiring On Bread Board |
In this programming example, I use the lower byte of Port A. It's 16-bit wide. Where the LCD connections between the STM32F103C8R6 Blue Pill are,
- LCD Register Select (RS) connects to PA0,
- LCD Read or Write (R/W) connects to GND as the LCD just need to receive command or data,
- LCD Enable (EN) connects to PA1,
- LCD Data D0...D3 left unconnected or connects GND, while the LCD Data D4...D7 connects to the Blue Pill PA4...PA7.
STM32CubeIDE Code Configuration Wizard allows the programmer to select any peripherals and system settings without writing code in C/C++ source file.
GPIO Selection |
Clock Configuration |
Schematic |
JHD16 LCD Module Front |
JHD16 LCD Module Back |
The source source code is quite easy as it's similar to LCD interfacing in 8-bit mode.
/* USER CODE BEGIN Header */ /** ****************************************************************************** * @file : main.c * @brief : Main program body ****************************************************************************** * @attention * * <h2><center>© Copyright (c) 2023 STMicroelectronics. * All rights reserved.</center></h2> * * This software component is licensed by ST under BSD 3-Clause license, * the "License"; You may not use this file except in compliance with the * License. You may obtain a copy of the License at: * opensource.org/licenses/BSD-3-Clause * ****************************************************************************** */ /* USER CODE END Header */ /* Includes ------------------------------------------------------------------*/ #include "main.h" /* Private function prototypes -----------------------------------------------*/ void SystemClock_Config(void); static void MX_GPIO_Init(void); /** * @brief The application entry point. * @retval int */ #define DPORT GPIOA->ODR void delay1(uint16_t dTime){ for(uint16_t i=0;i<dTime;i++); } void delay2(uint8_t dTime){ for(uint8_t i=0;i<dTime;i++) delay1(5000); } void lcdCmd(uint8_t cmd){ uint8_t temp=0x02; DPORT=temp|(cmd&0xF0); delay1(100); temp=0; DPORT=temp|(cmd&0xF0); delay1(1000); temp=0x02; DPORT=temp|(cmd<<4); delay1(100); temp=0; DPORT=temp|(cmd<<4); } void lcdDat(uint8_t dat){ uint8_t temp=0x03; DPORT=temp|(dat&0xF0); delay1(100); temp=0x01; DPORT=temp|(dat&0xF0); delay1(1000); temp=0x03; DPORT=temp|(dat<<4); delay1(100); temp=0x01; DPORT=temp|(dat<<4); } void lcdInit(void){ DPORT=0x00; delay1(2000); lcdCmd(0x33); delay1(100); lcdCmd(0x32); delay1(100); lcdCmd(0x28); delay1(100); lcdCmd(0x0F); delay1(100); lcdCmd(0x01); delay1(2000); lcdCmd(0x06); delay1(100); } void lcdStr(uint8_t *str){ while(*str) lcdDat(*str++); } void lcdXY(uint8_t x,uint8_t y){ // 16x2 LCD uint8_t tbe[2]={0x80,0xC0}; // 20x4 LCD //uint8_t tbe[]={0x80,0xC0,0x94,0xD4}; lcdCmd(tbe[y-1]+x-1); delay1(100); } void lcdClear(void){ lcdCmd(0x01); delay1(100); } int main(void) { /* Reset of all peripherals, Initializes the Flash interface and the Systick. */ HAL_Init(); /* Configure the system clock */ SystemClock_Config(); /* Initialize all configured peripherals */ MX_GPIO_Init(); lcdInit(); lcdClear(); delay2(100); lcdXY(2,1); lcdStr("STM32F103C8T6"); delay2(100); lcdXY(1,2); lcdStr("4-Bit Mode LCD"); /* Infinite loop */ /* USER CODE BEGIN WHILE */ while (1) { /* USER CODE END WHILE */ /* USER CODE BEGIN 3 */ } /* USER CODE END 3 */ } /** * @brief System Clock Configuration * @retval None */ void SystemClock_Config(void) { RCC_OscInitTypeDef RCC_OscInitStruct = {0}; RCC_ClkInitTypeDef RCC_ClkInitStruct = {0}; /** Initializes the RCC Oscillators according to the specified parameters * in the RCC_OscInitTypeDef structure. */ RCC_OscInitStruct.OscillatorType = RCC_OSCILLATORTYPE_HSI; RCC_OscInitStruct.HSIState = RCC_HSI_ON; RCC_OscInitStruct.HSICalibrationValue = RCC_HSICALIBRATION_DEFAULT; RCC_OscInitStruct.PLL.PLLState = RCC_PLL_NONE; if (HAL_RCC_OscConfig(&RCC_OscInitStruct) != HAL_OK) { Error_Handler(); } /** Initializes the CPU, AHB and APB buses clocks */ RCC_ClkInitStruct.ClockType = RCC_CLOCKTYPE_HCLK|RCC_CLOCKTYPE_SYSCLK |RCC_CLOCKTYPE_PCLK1|RCC_CLOCKTYPE_PCLK2; RCC_ClkInitStruct.SYSCLKSource = RCC_SYSCLKSOURCE_HSI; RCC_ClkInitStruct.AHBCLKDivider = RCC_SYSCLK_DIV1; RCC_ClkInitStruct.APB1CLKDivider = RCC_HCLK_DIV1; RCC_ClkInitStruct.APB2CLKDivider = RCC_HCLK_DIV1; if (HAL_RCC_ClockConfig(&RCC_ClkInitStruct, FLASH_LATENCY_0) != HAL_OK) { Error_Handler(); } } /** * @brief GPIO Initialization Function * @param None * @retval None */ static void MX_GPIO_Init(void) { GPIO_InitTypeDef GPIO_InitStruct = {0}; /* GPIO Ports Clock Enable */ __HAL_RCC_GPIOA_CLK_ENABLE(); /*Configure GPIO pin Output Level */ HAL_GPIO_WritePin(GPIOA, GPIO_PIN_0|GPIO_PIN_1|GPIO_PIN_4|GPIO_PIN_5 |GPIO_PIN_6|GPIO_PIN_7, GPIO_PIN_RESET); /*Configure GPIO pins : PA0 PA1 PA4 PA5 PA6 PA7 */ GPIO_InitStruct.Pin = GPIO_PIN_0|GPIO_PIN_1|GPIO_PIN_4|GPIO_PIN_5 |GPIO_PIN_6|GPIO_PIN_7; GPIO_InitStruct.Mode = GPIO_MODE_OUTPUT_PP; GPIO_InitStruct.Pull = GPIO_NOPULL; GPIO_InitStruct.Speed = GPIO_SPEED_FREQ_LOW; HAL_GPIO_Init(GPIOA, &GPIO_InitStruct); } /** * @brief This function is executed in case of error occurrence. * @retval None */ void Error_Handler(void) { /* USER CODE BEGIN Error_Handler_Debug */ /* User can add his own implementation to report the HAL error return state */ __disable_irq(); while (1) { } /* USER CODE END Error_Handler_Debug */ } #ifdef USE_FULL_ASSERT /** * @brief Reports the name of the source file and the source line number * where the assert_param error has occurred. * @param file: pointer to the source file name * @param line: assert_param error line source number * @retval None */ void assert_failed(uint8_t *file, uint32_t line) { /* USER CODE BEGIN 6 */ /* User can add his own implementation to report the file name and line number, ex: printf("Wrong parameters value: file %s on line %d\r\n", file, line) */ /* USER CODE END 6 */ } #endif /* USE_FULL_ASSERT */ /************************ (C) COPYRIGHT STMicroelectronics *****END OF FILE****/
I use C/C++ loop to create delay without using the STM32CubeIDE Hal_Delay function. However the delay time is not known.
Simulating Program In Proteus |
Click here to download its source file.
No comments:
Post a Comment