ээээ... как показывает практика - использование очередей в уарте не есть самый лучший способ передачи данных по нескольким причинам
- необходимо распределеить память под очередь
- передача одного байта вызывает большой overhead
- еще какие-то-там-я-и-не-помню.
Более того, несмотря на то, что прерывания при низких скоростях приходят довольно-таки редко, для их обработки, тем не менее, требуется выполнить несколько действий.
В stm32 можно использовать DMA на уарте и это позволяет решить проблему оверхеда. Однако без прерываний никак по-любому, однако они будут случаться только тогда, когда приемный буфер заполнен, или когда передача прекращена. При это в оси будет использоваться только один семафор.
Код для stm32l152 ниже.
Может кому пригодится еще. у меня работает в оч критической задаче.
Код
#include "stm32l1xx.h"
#include "stm32l1xx_rcc.h"
#include "stm32l1xx_tim.h"
#include "stm32l1xx_gpio.h"
#include "stm32l1xx_spi.h"
#include "stm32l1xx_dma.h"
#include "stm32l1xx_usart.h"
#include "misc.h"
#include "FreeRTOS.h"
#include "semphr.h"
#include "task.h"
#include "queue.h"
#define USART2_RX_BUF_SIZE 128
static char usart2_rx_buffer[USART2_RX_BUF_SIZE];
xSemaphoreHandle xSemaphoreUSART2;
void
usart2_init(void)
{
USART_InitTypeDef USART_InitStructure;
GPIO_InitTypeDef GPIO_InitStructure;
NVIC_InitTypeDef NVIC_InitStructure;
DMA_InitTypeDef DMA_InitStructure;
RCC_APB1PeriphClockCmd(RCC_APB1Periph_USART2, ENABLE);
USART_InitStructure.USART_BaudRate = 57600;
USART_InitStructure.USART_WordLength = USART_WordLength_8b;
USART_InitStructure.USART_StopBits = USART_StopBits_1;
USART_InitStructure.USART_Parity = USART_Parity_No;
USART_InitStructure.USART_HardwareFlowControl = USART_HardwareFlowControl_None;
USART_InitStructure.USART_Mode = USART_Mode_Rx | USART_Mode_Tx;
USART_Init(USART2, &USART_InitStructure);
USART_DMACmd(USART2, USART_DMAReq_Tx | USART_DMAReq_Rx, ENABLE);
/* Enable USART1 IDLE line interrupts because all others are handeled via DMA */
USART_ITConfig(USART2, USART_IT_IDLE , ENABLE);
GPIO_PinAFConfig(GPIOA, GPIO_PinSource2, GPIO_AF_USART2 );
GPIO_PinAFConfig(GPIOA, GPIO_PinSource3, GPIO_AF_USART2 );
/* Configure USART1 Tx, RX (PA.02 PA.03) as alternate function push-pull */
GPIO_InitStructure.GPIO_Pin = GPIO_Pin_2 | GPIO_Pin_3;
GPIO_InitStructure.GPIO_Speed = GPIO_Speed_40MHz;
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF;
GPIO_InitStructure.GPIO_OType = GPIO_OType_PP;
GPIO_InitStructure.GPIO_PuPd = GPIO_PuPd_UP;
GPIO_Init(GPIOA, &GPIO_InitStructure);
/* Enable the USART1 Interrupt */
NVIC_InitStructure.NVIC_IRQChannel = USART2_IRQn;
NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 12;
NVIC_InitStructure.NVIC_IRQChannelSubPriority = 12;
NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;
NVIC_Init(&NVIC_InitStructure);
DMA_DeInit(DMA1_Channel6);
DMA_InitStructure.DMA_PeripheralBaseAddr = (uint32_t)&USART2->DR;
DMA_InitStructure.DMA_MemoryBaseAddr = (uint32_t)usart2_rx_buffer;
DMA_InitStructure.DMA_DIR = DMA_DIR_PeripheralSRC;
DMA_InitStructure.DMA_BufferSize = USART2_RX_BUF_SIZE;
DMA_InitStructure.DMA_PeripheralInc = DMA_PeripheralInc_Disable;
DMA_InitStructure.DMA_MemoryInc = DMA_MemoryInc_Enable;
DMA_InitStructure.DMA_PeripheralDataSize = DMA_PeripheralDataSize_Byte;
DMA_InitStructure.DMA_MemoryDataSize = DMA_MemoryDataSize_Byte;
DMA_InitStructure.DMA_Mode = DMA_Mode_Circular;
DMA_InitStructure.DMA_Priority = DMA_Priority_High;
DMA_InitStructure.DMA_M2M = DMA_M2M_Disable;
DMA_Init(DMA1_Channel6, &DMA_InitStructure);
USART_DMACmd(USART2, USART_DMAReq_Rx, ENABLE);
NVIC_InitStructure.NVIC_IRQChannel = DMA1_Channel6_IRQn;
NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 12;
NVIC_InitStructure.NVIC_IRQChannelSubPriority = 11;
NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;
NVIC_Init(&NVIC_InitStructure);
/* Enable interrupts on channel 6 */
DMA_ITConfig(DMA1_Channel6, DMA_IT_TC, ENABLE);
DMA_ITConfig(DMA1_Channel6, DMA_IT_HT, ENABLE);
vSemaphoreCreateBinary( xSemaphoreUSART2 );
xSemaphoreTake( xSemaphoreUSART2, ( portTickType ) portMAX_DELAY );
USART_DMACmd(USART2, USART_DMAReq_Tx, ENABLE);
USART2->SR = 0;
DMA_Cmd(DMA1_Channel6, ENABLE);
USART_Cmd(USART2, ENABLE);
}
int
usart2_send(void *buffer, int size)
{
DMA_InitTypeDef DMA_InitStructure;
if(size > 65535)
return -2;
// if (DMA1_Channel7->CNDTR)
// return -1;
while(DMA1_Channel7->CNDTR);
// now configure transmit line:
// do not enable interrupts
DMA_DeInit(DMA1_Channel7);
DMA_InitStructure.DMA_PeripheralBaseAddr = (uint32_t)&USART2->DR;
DMA_InitStructure.DMA_MemoryBaseAddr = (uint32_t)buffer;
DMA_InitStructure.DMA_DIR = DMA_DIR_PeripheralDST;
DMA_InitStructure.DMA_BufferSize = size;
DMA_InitStructure.DMA_PeripheralInc = DMA_PeripheralInc_Disable;
DMA_InitStructure.DMA_MemoryInc = DMA_MemoryInc_Enable;
DMA_InitStructure.DMA_PeripheralDataSize = DMA_PeripheralDataSize_Byte;
DMA_InitStructure.DMA_MemoryDataSize = DMA_MemoryDataSize_Byte;
DMA_InitStructure.DMA_Mode = DMA_Mode_Normal;
DMA_InitStructure.DMA_Priority = DMA_Priority_High;
DMA_InitStructure.DMA_M2M = DMA_M2M_Disable;
DMA_Init(DMA1_Channel7, &DMA_InitStructure);
DMA_Cmd(DMA1_Channel7, ENABLE);
return 0;
}
void USART2_IRQHandler(void)
{
portBASE_TYPE xHigherPriorityTaskWoken = pdFALSE;
int statusreg = USART2->SR;
char uart_messages;
uart_messages = 0;
if(statusreg & (1<<4) ) // IDLE. ONLY this interrupt is enabled!!!
{
uart_messages = 3;
USART2->DR; // dummy read to clear idle flag
}
if(uart_messages)
{
xSemaphoreGiveFromISR( xSemaphoreUSART2, &xHigherPriorityTaskWoken );
}
portEND_SWITCHING_ISR(xHigherPriorityTaskWoken);
}
void
DMA1_Channel6_IRQHandler(void)
{
portBASE_TYPE xHigherPriorityTaskWoken;
DMA1->IFCR = 7 << 20; // clear DMA interrupt flags and continue
xSemaphoreGiveFromISR( xSemaphoreUSART2, &xHigherPriorityTaskWoken );
portEND_SWITCHING_ISR(xHigherPriorityTaskWoken);
}
static int rx_read;
int
uart2_get_char(portTickType block_time)
{
int res;
while(1)
{
if (USART2_RX_BUF_SIZE - DMA1_Channel6->CNDTR != rx_read)
{
res = usart2_rx_buffer[rx_read];
rx_read++;
if(rx_read == USART2_RX_BUF_SIZE)
rx_read = 0;
return res & 0xff;
}
if (pdFALSE == xSemaphoreTake( xSemaphoreUSART2, ( portTickType ) block_time ))
return -1;
}
return 0;
}