历史上的今天
今天是:2024年10月11日(星期五)
2019年10月11日 | STM32F1xx使用FIFO实现USART串口发送中断
2019-10-11 来源:eefocus
fifo就不要造轮子了,用现成的就行了。linux内核中有目前人类写出的基于c语言的最强FIFO,请自行搜索学习《巧夺天工的kfifo》,或者我的另一篇博文《整数的环回特性》。
直接把最常用的几个函数拷贝到STM32工程文件里,顺便把kfifo结构体中的自旋锁成员给屏蔽掉,这玩意只在多核才有用,在单核的32上没有作用,直接注释掉就行。然后把源码中自旋上锁、自旋解锁分别改成STM32的开中断、关中断(或者改成进入临界段、退出临界段,参考我的另一篇博文:《STM32使用中断屏蔽寄存器BASEPRI保护临界段+中断分组+抢占/响应优先级概念》),用以保护FIFO的读写索引。还要一个至关重要的min宏需要移植,此宏很有讲究,请一定要参考我的另一篇博文进行min宏的移植《求最小值的宏:#define min(x,y) x > y? y: x 中的陷阱》,否则在fifo写索引溢出时会触发bug。
typedef struct kfifo {
uint8_t *buffer; /* the buffer holding the data */
uint16_t size; /* the size of the allocated buffer */
uint16_t in; /* data is added at offset (in % size) */
uint16_t out; /* data is extracted from off. (out % size) */
//spinlock_t *lock; /* protects concurrent modifications */
}gfifo_t;
uint16_t __kfifo_put(struct kfifo *fifo,
const uint8_t *buffer, uint16_t len);
uint16_t __kfifo_get(struct kfifo *fifo,
uint8_t *buffer, uint16_t len);

要理解STM32的USART发送中断,首先要了解两个概念:发送数据寄存器DR、移位寄存器,我们发送数据时就是把数据写入DR就不管了,硬件一旦发现DR中有数据,就会自动把DR中的数据放到移位寄存器中,然后硬件逻辑才一位一位地把数据发出去。也就是说:DR空并不意味着发送已完成,移位寄存器空才是真正的发送完成。
STM32的USART发送中断有两个:
(1)“发送完成TC”中断,意思是移位寄存器已发送完成
(2)"数据寄存器空TXE"中断,要注意这个中断!一上电数据寄存器DR中是没有数据的,所以,一旦开启TXE中断(当然,开全局中断也得是开着的),就会立即进入中断服务函数。这就指示我们:不要在初始化中开启TXE中断,而是要在打算发数据时才开。
以下是F103C8T6的USART1初始化
//串口1初始化
void USART1_Init(u32 bound)
{
//GPIO端口设置
GPIO_InitTypeDef GPIO_InitStructure;
USART_InitTypeDef USART_InitStructure;
NVIC_InitTypeDef NVIC_InitStructure;
RCC_APB2PeriphClockCmd(RCC_APB2Periph_USART1|RCC_APB2Periph_GPIOA, ENABLE); //使能USART1,GPIOA时钟
//USART1_TX GPIOA.9
GPIO_InitStructure.GPIO_Pin = GPIO_Pin_9; //PA.9
GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_PP; //复用推挽输出
GPIO_Init(GPIOA, &GPIO_InitStructure);//初始化GPIOA.9
//USART1_RX GPIOA.10初始化
GPIO_InitStructure.GPIO_Pin = GPIO_Pin_10;//PA10
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IN_FLOATING;//浮空输入
GPIO_Init(GPIOA, &GPIO_InitStructure);//初始化GPIOA.10
//Usart1 NVIC 配置
NVIC_InitStructure.NVIC_IRQChannel = USART1_IRQn;//指出中断通道为UASRT1
NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority=3 ;//抢占优先级3
NVIC_InitStructure.NVIC_IRQChannelSubPriority = 3; //子优先级3
NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE; //IRQ通道使能
NVIC_Init(&NVIC_InitStructure); //根据指定的参数初始化VIC寄存器
//USART 初始化设置
USART_InitStructure.USART_BaudRate = bound;//串口波特率
USART_InitStructure.USART_WordLength = USART_WordLength_8b;//字长为8位数据格式
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(USART1, &USART_InitStructure); //初始化串口1
USART_ITConfig(USART1, USART_IT_RXNE, ENABLE);//开启串口接收寄存器非空中断
//USART_ITConfig(USART1, USART_IT_TC, ENABLE);//开启串口发送完成中断(移位寄存器发送完成)
//USART_ITConfig(USART1, USART_IT_TXE, ENABLE);//开启串口发送寄存器空中断(只要DR空就中断,实际上,这句话会导致立马中断,因为此时DR确实是空的,所以一般只在发送时才enable它,而不会在初始化时就enable它)
USART_Cmd(USART1, ENABLE); //使能串口1
}
使用FIFO进行中断式发送的步骤如下:
①把想发的n个字节数据填入FIFO
②开启TXE中断
/*
通过中断发送数据
返回:1成功,0失败
*/
int16_t uart1_send_by_int(const uint8_t *data, uint16_t len)
{
if(get_fifo_unused_size(&uart1TxFifo) >= len)//只有空闲区>len,才执行发送程序
{
gfifo_put(&uart1TxFifo, data, len);
}
else
{
rtt_printf("uart1 SendFifo has no spacern");//程序走到这里,意味着FIFO缓冲不足,会出现发送丢失
return 0;
}
USART_ITConfig(USART1, USART_IT_TXE, ENABLE);
return 1;
}
执行完上述发送函数后,硬件发现DR寄存器中没有数据,会立即进入TXE中断,接下来我们写TXE中断的服务函数:
//串口1的所有中断服务
void USART1_IRQHandler(void) //串口1中断服务程序
{
if(USART_GetITStatus(USART1, USART_IT_TXE) != RESET) //数据寄存器DR空中断TXE
{
if(get_fifo_used_size(&uart1TxFifo) > 0)//main调用链中操作uart1TxFifo的地方必须禁掉本中断(或全局中断)
{
uint8_t sendCh;
//从FIFO中取出一个字节并发送,这个字节一旦被从DR移入移位寄存器,就会再次进入本中断
gfifo_get(&uart1TxFifo, &sendCh, 1);
USART1->DR = sendCh;
}
else
{
USART_ITConfig(USART1, USART_IT_TXE, DISABLE);//FIFO中的所有数据都已发完,关中断
}
}
}
史海拾趣
|
本帖最后由 paulhyde 于 2014-9-15 09:50 编辑 哪位高人能帮帮我啊!急需!!!谢了先……有的话请发到我的邮箱:cocokaola@126.com … 查看全部问答> |
|
我现在需要工控机(IPC)控制30个继电器并行工作,要求延时精度在1毫秒, windows够精度的方法都太耗资源,不能同时开30个,只好想办法硬件解决…… 不知道有没有这类现成的继电器输出板卡可以选? 另外单片机最多有内置多少个定时器的?或者只 ...… 查看全部问答> |
|
使用430F437发现32K晶体起振速度很慢,引起LCD显示开机不正常。 使用430F437发现32K晶体起振速度很慢,引起LCD显示开机不正常。 是什么原因引起的,我怎么改电容都不行啊。… 查看全部问答> |
|
sitara 板子已收到,还没拆开看,晚上上图 ___________________________________________________ 对不住大家了,晚上光线不好,照了几张惨不忍睹,就不上传了。 屏幕色彩不太好,其他都很不错。 [ 本帖最后由 lcofjp 于 2012-11-20 19:50 编辑 ...… 查看全部问答> |
|
[Linux资料]Linux驱动程序工作原理简介 Linux驱动程序工作原理简介 一、linux驱动程序的数据结构 1 二、设备节点如何产生? 2 三、应用程序是如何访问设备驱动程序的? &n ...… 查看全部问答> |
|
昨天DE1到手,就准备装开发环境,因为QII之前有装是13.1,所以打算装个EDS就可以开工了,到altera下载了个SOCEDS 14.1版本,等下完安装的时候提示这个软件只能装在64位系统上,一下蛋疼了,现在用的xp,装了诸多的软件 ...… 查看全部问答> |




