使用Microblaze创建计时器的最佳方法是什么,这样我可以让它在更传统的脚本中更像delay_ms()
或sleep()
这样的函数?
很容易,我可以像这样创建一个愚蠢的函数:
void delay_ms(int i) {
//mind that I am doing this on the top of my head
for(delays=0; delay<(i*((1/frequency of the device)/2)); delays++) {
}
}
...但是只有处理器处理才能完成,直到它完成,而实际上我需要它让函数允许我在一段时间内停止一个进程,而另一个进程继续工作。
这样的事情是可能的,毫无疑问,但这个问题最简单的解决方案是什么?
(我使用的是Spartan-3A,但我相信该解决方案适用于不同的套件,FPGA。)
答案 0 :(得分:2)
<强> TL; DR 强>
使用微型操作系统,例如FreeRTOS。
答案错误
好吧,如果你没有操作系统,没有任务换向,但有一个外部计时器,你可以 使用以下方法:
启用硬件计时器中断,并管理由此中断驱动的计数器:
你应该有像
这样的东西/**timer.c**/
/* The internal counters
* each task have its counter
*/
static int s_timers[NUMBER_OF_TASKS] = {0,0};
/* on each time tick, decrease timers */
void timer_interrupt()
{
int i;
for (i = 0; i < NUMBER_OF_TASKS; ++i)
{
if (s_timer[i] > 0)
{
s_timer[i]--;
}
}
}
/* set wait counter:
* each task says how tick it want to wait
*/
void timer_set_wait(int task_num, int tick_to_wait)
{
s_timer[task_num] = tick_to_wait;
}
/**
* each task can ask if its time went out
*/
int timer_timeout(int task_num)
{
return (0 == s_timer[task_num]);
}
一旦你有像计时器这样的东西(上面的代码很容易完成), 计划你的任务:
/**task-1.c**/
/*TASK ID must be valid and unique in s_timer */
#define TASK_1_ID 0
void task_1()
{
if (timer_timeout(TASK_1_ID))
{
/* task has wait long enough, it can run again */
/* DO TASK 1 STUFF */
printf("hello from task 1\n");
/* Ask to wait for 150 ticks */
timer_set_wait(TASK_1_ID, 150);
}
}
/**task-2.c**/
/*TASK ID must be valid and unique in s_timer */
#define TASK_2_ID 1
void task_2()
{
if (timer_timeout(TASK_2_ID))
{
/* task has wait long enough, it can run again */
/* DO TASK 2 STUFF */
printf("hello from task 2\n");
/* Ask to wait for 250 ticks */
timer_set_wait(TASK_2_ID, 250);
}
}
安排(这里一个大词)任务:
/** main.c **/
int main()
{
/* init the program, like set up the timer interruption */
init()
/* do tasks, for ever*/
while(1)
{
task_1();
task_2();
}
return 0;
}
我认为我所描述的是一种不应该被认真使用的蹩脚解决方案。
我给出的代码充满了问题,比如如果任务变得缓慢执行会发生什么......
相反,你 - 可以 - 应该使用一些RT Os,比如FreeRTOS,这对这类问题非常有帮助。