dc31-addon-addon-badge-firm.../bootloader_hk32f030m_uart_x.../code/src/usart.c

142 lines
3.4 KiB
C

/*
* usart.c
*
* Created on: Jun 23, 2023
* Author: true
*/
#include "hk32f030m.h"
#include "usart.h"
#include <string.h>
#define BAUDRATE 115200
#define USART_DEVICE USART1
#define USART_TX_GPIO_CLK RCC_AHBPeriph_GPIOD
#define USART_TX_PORT GPIOD
#define USART_TX_PIN 1
#define USART_TX_PINSRC GPIO_PinSource1
#define USART_RX_GPIO_CLK RCC_AHBPeriph_GPIOB
#define USART_RX_PORT GPIOB
#define USART_RX_PIN 4
#define USART_RX_PINSRC GPIO_PinSource4
// inefficient, but we have time and space
#define RX_BUF_LEN 1040
uint8_t rxbuf[RX_BUF_LEN];
uint16_t rxptr = 0;
uint16_t rxlen = 0;
/**
* @brief Initializes USART peripheral at fixed baud rate.
*/
void comm_init()
{
// enable GPIO clock
// RCC_AHBPeriphClockCmd(USART_TX_GPIO_CLK | USART_RX_GPIO_CLK, ENABLE);
// attach USART peripheral to pins
// GPIO_PinAFConfig(USART_TX_PORT, USART_TX_PINSRC, GPIO_AF_1);
// GPIO_PinAFConfig(USART_RX_PORT, USART_RX_PINSRC, GPIO_AF_1);
GPIOB->AFR[0] = 0x00010000;
GPIOD->AFR[0] = 0x00000010;
/* these have been configured in user_io.c to save space
USART_RX_PORT->MODER &= ~(GPIO_MODER_MODER0 << 2 * USART_RX_PIN);
USART_RX_PORT->MODER |= (GPIO_Mode_AF << 2 * USART_RX_PIN); // alt function
USART_TX_PORT->MODER &= ~(GPIO_MODER_MODER0 << 2 * USART_TX_PIN);
USART_TX_PORT->OTYPER &= ~(GPIO_OTYPER_OT_0 << USART_TX_PIN); // push pull
USART_TX_PORT->OSPEEDR &= ~(GPIO_OSPEEDR_OSPEEDR0 << 2 * USART_TX_PIN); // slow
USART_TX_PORT->MODER |= (GPIO_Mode_AF << 2 * USART_TX_PIN); // alt function
*/
// enable USART clock
// RCC_APB2PeriphClockCmd(RCC_APB2Periph_USART1, ENABLE);
// configure USART peripheral manually
// we do this as the library bloats flash space (division, verbose, ...)
USART1->CR1 = 0; // disable USART
USART1->CR2 = 0; // 1 stop bit
USART1->CR3 = 0;
USART1->CR1 = USART_Mode_Rx | USART_Mode_Tx; // 8 bits, no parity, tx/rx enabled
USART1->BRR = 0x45; // 115200 baud rate @8MHz (per datasheet)
// enable USART
USART1->CR1 = USART_Mode_Rx | USART_Mode_Tx | USART_CR1_UE;
}
/**
* @brief Receives data from USART in a polling manner.
* @param *data: Array to save the received data.
* @param length: Size of the data.
* @return status: Report about the success of the receiving.
*/
BL_comm_status comm_rx(uint8_t *data, uint16_t length)
{
uint16_t i;
uint16_t timeout;
for (i = 0; i < length; i++) {
timeout = COMM_TIMEOUT;
while (!(USART1->ISR & USART_ISR_RXNE) && timeout) {
timeout--;
}
if (timeout) {
// we didn't time out and now there is new data waiting for us
data[i] = USART1->RDR;
} else {
// no data returned in time
return COMM_ERROR;
}
}
return COMM_OK;
}
/**
* @brief Transmits a single char to UART.
* @param *data: The char.
* @return status: Report about the success of the transmission.
*/
BL_comm_status comm_tx_byte(uint8_t data)
{
USART_SendData(USART1, data);
while (!(USART1->ISR & USART_ISR_TXE));
return COMM_OK;
}
/**
* @brief Transmits a null-terminated string over USART.
* @param *data: String array.
* @return status: Report about the success of the transmission.
*/
BL_comm_status comm_tx_str(char *data)
{
uint8_t i;
uint16_t len = 0;
len = strlen((const char *)data);
for (i = 0; i < len; i++) {
comm_tx_byte(data[i]);
}
while (!(USART1->ISR & USART_ISR_TC));
return COMM_OK;
}