Ten post będzie kontynuacją wcześniejszego, dotyczącego komunikacji poprzez USART. Tym razem przedstawię odwrotny sposób transmisji czyli od komputera do mikrokontrolera.
Na samym początku po włączeniu zegarów w programie CubeMx, należy włączyć obsługę USART2 w trybie asynchronicznym, po czym uruchomić przerwania NVIC dla niego.
Wygenerowana funkcja dla ustawienia taktowania zegara oraz uruchomienia USART-u wygląda następująco:
Program
Na samym początku po włączeniu zegarów w programie CubeMx, należy włączyć obsługę USART2 w trybie asynchronicznym, po czym uruchomić przerwania NVIC dla niego.
Wygenerowana funkcja dla ustawienia taktowania zegara oraz uruchomienia USART-u wygląda następująco:
- /** System Clock Configuration
- */
- void SystemClock_Config(void)
- {
- RCC_OscInitTypeDef RCC_OscInitStruct;
- RCC_ClkInitTypeDef RCC_ClkInitStruct;
- __PWR_CLK_ENABLE();
- __HAL_PWR_VOLTAGESCALING_CONFIG(PWR_REGULATOR_VOLTAGE_SCALE1);
- RCC_OscInitStruct.OscillatorType = RCC_OSCILLATORTYPE_HSE;
- RCC_OscInitStruct.HSEState = RCC_HSE_ON;
- RCC_OscInitStruct.PLL.PLLState = RCC_PLL_ON;
- RCC_OscInitStruct.PLL.PLLSource = RCC_PLLSOURCE_HSE;
- RCC_OscInitStruct.PLL.PLLM = 8;
- RCC_OscInitStruct.PLL.PLLN = 336;
- RCC_OscInitStruct.PLL.PLLP = RCC_PLLP_DIV2;
- RCC_OscInitStruct.PLL.PLLQ = 4;
- HAL_RCC_OscConfig(&RCC_OscInitStruct);
- RCC_ClkInitStruct.ClockType = RCC_CLOCKTYPE_HCLK|RCC_CLOCKTYPE_SYSCLK
- |RCC_CLOCKTYPE_PCLK1|RCC_CLOCKTYPE_PCLK2;
- RCC_ClkInitStruct.SYSCLKSource = RCC_SYSCLKSOURCE_PLLCLK;
- RCC_ClkInitStruct.AHBCLKDivider = RCC_SYSCLK_DIV1;
- RCC_ClkInitStruct.APB1CLKDivider = RCC_HCLK_DIV4;
- RCC_ClkInitStruct.APB2CLKDivider = RCC_HCLK_DIV2;
- HAL_RCC_ClockConfig(&RCC_ClkInitStruct, FLASH_LATENCY_5);
- HAL_SYSTICK_Config(HAL_RCC_GetHCLKFreq()/1000);
- HAL_SYSTICK_CLKSourceConfig(SYSTICK_CLKSOURCE_HCLK);
- /* SysTick_IRQn interrupt configuration */
- HAL_NVIC_SetPriority(SysTick_IRQn, 0, 0);
- }
- /* USART2 init function */
- void MX_USART2_UART_Init(void)
- {
- huart2.Instance = USART2;
- huart2.Init.BaudRate = 9600;
- huart2.Init.WordLength = UART_WORDLENGTH_8B;
- huart2.Init.StopBits = UART_STOPBITS_1;
- huart2.Init.Parity = UART_PARITY_NONE;
- huart2.Init.Mode = UART_MODE_TX_RX;
- huart2.Init.HwFlowCtl = UART_HWCONTROL_NONE;
- huart2.Init.OverSampling = UART_OVERSAMPLING_16;
- HAL_UART_Init(&huart2);
- }
Następnie jeśli projekt był robiony nie na modelu płytki STM32F4 - Discovery, tylko na zwykłym mikrokontrolerze to należy uruchomić dodatkowo wszystkie diody wbudowane.
- void MX_GPIO_Init(void)
- {
- GPIO_InitTypeDef GPIO_InitStruct;
- /* GPIO Ports Clock Enable */
- __GPIOH_CLK_ENABLE();
- __GPIOA_CLK_ENABLE();
- __GPIOD_CLK_ENABLE();
- /*Configure GPIO pin Output Level */
- HAL_GPIO_WritePin(GPIOD, GPIO_PIN_12|GPIO_PIN_13|GPIO_PIN_14|GPIO_PIN_15, GPIO_PIN_RESET);
- /*Configure GPIO pins : PD12 PD13 PD14 PD15 */
- GPIO_InitStruct.Pin = GPIO_PIN_12|GPIO_PIN_13|GPIO_PIN_14|GPIO_PIN_15;
- GPIO_InitStruct.Mode = GPIO_MODE_OUTPUT_PP;
- GPIO_InitStruct.Pull = GPIO_NOPULL;
- GPIO_InitStruct.Speed = GPIO_SPEED_FREQ_LOW;
- HAL_GPIO_Init(GPIOD, &GPIO_InitStruct);
- }
Kolejnym etapem jest przygotowanie funkcji obsługujące odebrane dane oraz zdeklarowanie zmiennej globalnej przechowujące zdeklarowane dane.
Zdeklarowana zmienna:
- volatile uint8_t Odebrane;
Funkcja działająca:
- void HAL_UART_RxCpltCallback(UART_HandleTypeDef *huart)
- {
- uint8_t Wyswietl[30];
- uint16_t rozmiar = 0;
- //Odebranie znaku i sprawdzenie czy odpowiada znakom ponizej
- switch(Odebrane)
- {
- case '0':
- {
- rozmiar = sprintf(Wyswietl, "Wylaczenie wszystkich diod\n\r");
- HAL_GPIO_WritePin(GPIOD, GPIO_PIN_12, GPIO_PIN_RESET);
- HAL_GPIO_WritePin(GPIOD, GPIO_PIN_13, GPIO_PIN_RESET);
- HAL_GPIO_WritePin(GPIOD, GPIO_PIN_14, GPIO_PIN_RESET);
- HAL_GPIO_WritePin(GPIOD, GPIO_PIN_15, GPIO_PIN_RESET);
- break;
- }
- case '1':
- {
- rozmiar = sprintf(Wyswietl, "Swieci dioda zielona: \n\r");
- HAL_GPIO_WritePin(GPIOD, GPIO_PIN_12, GPIO_PIN_SET);
- break;
- }
- case '2':
- {
- rozmiar = sprintf(Wyswietl, "Swieci dioda pomaranczowa: \n\r");
- HAL_GPIO_WritePin(GPIOD, GPIO_PIN_13, GPIO_PIN_SET);
- break;
- }
- case '3':
- {
- rozmiar = sprintf(Wyswietl, "Swieci dioda czerwona: \n\r");
- HAL_GPIO_WritePin(GPIOD, GPIO_PIN_14, GPIO_PIN_SET);
- break;
- }
- case '4':
- {
- rozmiar = sprintf(Wyswietl, "Swieci dioda niebieska: \n\r");
- HAL_GPIO_WritePin(GPIOD, GPIO_PIN_15, GPIO_PIN_SET);
- break;
- }
- default:
- {
- rozmiar = sprintf(Wyswietl, "Odebrano zly znak: '%c'\n\r", Odebrane);
- break;
- }
- }
- //Wyslanie danych
- HAL_UART_Transmit_IT(&huart2, Wyswietl, rozmiar);
- //Czekanie na odebranie danych
- HAL_UART_Receive_IT(&huart2, &Odebrane, 1);
- }
Następnie należy wprowadzić wywołanie funkcji odbierającej pojedynczy znak w funkcji main().
- HAL_UART_Receive_IT(&huart2, &Odebrane, 1);
Poniżej okno programu terminal:
Rys. 1. Okno programu terminal
Drugi program będzie przedstawiał w jaki sposób dokonać zapisu całego ciągu znaków. Dane wysłane z komputera będą ponownie wysyłane z mikrokontrolera do komputera.
Aby komunikacja przeszła bezproblemowo należy zdefiniować tablice przechowującą przesłane znaki.
- uint8_t Odebrane[5];
- void HAL_UART_RxCpltCallback(UART_HandleTypeDef *huart)
- {
- uint8_t Wyswietl[20];
- uint16_t rozmiar = 0;
- rozmiar = sprintf(Wyswietl, "Dane: %s",Odebrane);
- //Wyslanie danych
- HAL_UART_Transmit_IT(&huart2, Wyswietl, rozmiar);
- //Czekanie na odebranie danych
- HAL_UART_Receive_IT(&huart2, Odebrane, 5);
- }
- HAL_UART_Receive_IT(&huart2, Odebrane, 5);