gpt4 book ai didi

c - 定时器中断和 UART 接收中断

转载 作者:太空宇宙 更新时间:2023-11-04 04:23:50 26 4
gpt4 key购买 nike

我是编程新手,我无法让中断按照我希望它们为我的应用程序工作的方式工作。我想通过 UART 将串行数据发送到 PSoC,每秒存储一次值,然后回显存储的值。我正在使用 RX 中断(RX FIFO 非空,优先级 2)和 TC 定时器中断(优先级 3)。附件是 TopDesign 配置。目前,我正试图让这段代码工作(只是一个示例代码,看看我是否能让中断正常工作)。我向 PSoC 发送了一个包含字符“o”的字符串,我应该只读取“o”和“-”,但代码总是卡在其中一个中断中,而另一个中断无法正常工作。谁能告诉我我做错了什么?非常感激!该板为CY8CKIT-042。

#include <project.h>//Contains
#include <stdlib.h>
#include <string.h>
#include <stdbool.h>
uint16 ms_count = 0;
uint8 ch;

CY_ISR_PROTO(Timer_ISR);
CY_ISR_PROTO(RX_ISR);

CY_ISR(Timer_ISR){//Every millisecond, the code goes here
ms_count++;
if (ms_count == 1000){//Every second
ms_count = 0;
LED_Write(!LED_Read());
while(ch != 'o')UART_UartPutChar('-');
}
}

CY_ISR(RX_ISR){
uint8 status = UART_rx_ClearInterrupt();//Clear interrupt flag
uint8 sub;
sub = UART_UartGetChar();
if (sub != 0u){//Make sure grabbed character is not an empty
ch = sub;
if (ch == 'o'){
UART_UartPutChar(ch);
}
}
}

int main()
{
/* Start SCB UART TX+RX operation */
Timer_1_Start();
Time_ISR_StartEx(Timer_ISR);
RX_ISR_StartEx(RX_ISR);
CyGlobalIntEnable;
/* Start SCB UART TX+RX operation */
UART_Start();
UART_UartPutString("fdssf\n");

for(;;)
{

}
}

最佳答案

I want to send serial data over the UART to the PSoC, store the values every second, and then echo back the stored values. I am using a RX interrupt (RX FIFO not empty, priority 2) and a timer interrupt with the TC (priority 3).

中断服务程序 (ISR) 的一个重要原则是使它们尽可能短。另一个是确保它们不会阻塞。正如 Hans Passant 在评论中指出的那样,您的 Timer_ISR 正在阻塞 while 循环。它会不断地发送垃圾邮件,将“-”字符放入 UART,并且不允许发生任何其他事情。这是个坏主意。恕我直言,在这里启用嵌套中断并不是一个好主意。

实际上看起来您已经在这一行中回显了字符“o”:

    if (ch == 'o'){
UART_UartPutChar(ch);
}

您实际上并没有等待一秒钟来回显此处的“o”,它会立即发生。您只需在 if 语句中添加 UART_UartPutChar('-') 即可立即发送破折号字符。听起来您现在正在编写一个简单的测试应用程序只是为了让事情正常运行,我不会费心等到计时器 ISR 触发以回显简单测试应用程序的“o”或“-”。

或者,如果您只想在字母“o”是最近读取的字母时每秒打印一次破折号,则可以将 while 循环替换为简单的 if声明:

if(ch == 'o')UART_UartPutChar('-');

请记住,只有当“o”是最近读取的字符时才有效,因为您要用读取的每个新字符(包括新行)覆盖 ch 变量。我不确定这样做的目的是什么,但确实如此。

您将来在使用 ISR 时可能会考虑的事情是让 ISR 设置一个标志并让主循环监视该标志,然后对其使用react,例如:

volatile int flag = 0;
CY_ISR(Timer_ISR){//Every millisecond, the code goes here
if (some event) {
flag=1;
}
void main() {
while (1) {
if (flag) {
flag=0;
do_something_in_a_long_loop();
}
}
}

(阅读有关 volatile here 的更多信息)。

这只是解决长循环问题的一种方法。此外,是否要在长代码块之前或之后清除标志取决于您的应用程序。如果您的代码块太长并且事件发生得太频繁,这确实有可能(很可能)丢失事件。在这种情况下,开始使用线程或实时操作系统可能是个好主意。您如何设计程序显然取决于您要解决的问题。

警告:在 ISR 和主代码之间共享全局变量时,或者在处理多线程应用程序时,很容易遇到 race conditions。 .您需要了解这些潜在的陷阱以及正确处理它们的各种策略。我建议拿起一本书或参加有关嵌入式编程和/或实时操作系统的类(class)。欢迎来到非常复杂的嵌入式和实时编程世界!

关于c - 定时器中断和 UART 接收中断,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/43673744/

26 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com