OpenEdv-开源电子网

 找回密码
 立即注册
正点原子全套STM32/Linux/FPGA开发资料,上千讲STM32视频教程免费下载...
查看: 8446|回复: 13

OSStatInit()卡主了是什么情况?~不行了!真的想不通啊!

[复制链接]

1

主题

10

帖子

0

精华

新手上路

积分
34
金钱
34
注册时间
2013-10-22
在线时间
0 小时
发表于 2014-4-4 15:07:02 | 显示全部楼层 |阅读模式
程序运行到OSStatInit()时就不在运行了,将OSStatInit()注释掉后每个任务都只运行一次后就不再运行了,但是这两种情况下,卡主后都还能响应中断,不知道是怎么回事?

这个程序的目的就是:通过按键0,1进入中断,在中断中向SW任务(sw_task)发送消息,SW根据消息来挂起和解挂AD采集任务(adc_task)和温度采集任务(wen_task),按键0,采集AD,按键1,采集温度。

我之前用寄存器版本移植UCOS,运行这个程序没有任何问题,现在用库函数移植UCOS,运行这个程序就出现了这个问题,原子哥能不能帮忙看一下:

main.c代码如下:

#include"sys.h"
#include"delay.h"
#include"usart.h"
#include "led.h"
#include "adc.h"
#include "lcd.h"
#include"exti.h"
#include"includes.h"

//START 任务
#define START_TASK_PRIO       10 //设置任务优先级
#define START_STK_SIZE   512 //设置任务堆栈大小
OS_STK START_TASK_STK[START_STK_SIZE];     //任务堆栈
void start_task(void *pdata);              //任务函数
    
//ADC任务
#define ADC_TASK_PRIO       7  //设置任务优先级
#define ADC_STK_SIZE       512 //设置任务堆栈大小
OS_STK ADC_TASK_STK[ADC_STK_SIZE];   //任务堆栈
void adc_task(void *pdata);   //任务函数

//WEN任务
#define WEN_TASK_PRIO       8  //设置任务优先级
#define WEN_STK_SIZE   512 //设置任务堆栈大小
OS_STK WEN_TASK_STK[WEN_STK_SIZE];   //任务堆栈
void wen_task(void *pdata);   //任务函数

//切换任务
#define SW_TASK_PRIO          5  //设置任务优先级
#define SW_STK_SIZE   512 //设置任务堆栈大小
OS_STK SW_TASK_STK[SW_STK_SIZE];   //任务堆栈
void sw_task(void *pdata);   //任务函数

//CPU利用率显示
#define CPU_TASK_PRIO       9  //设置任务优先级
#define CPU_STK_SIZE   512 //设置任务堆栈大小
OS_STK CPU_TASK_STK[CPU_STK_SIZE];   //任务堆栈
void cpu_task(void *pdata);   //任务函数

OS_EVENT *myMBox;//消息邮箱

#define READ_TIMES                  40  //一次读取ADC数据的个数
#define LOST_VAL                        3 //去掉最大值和最小值的个数

int main(void)
{  
NVIC_Configuration();
delay_init();     //延时初始化
uart_init(9600); //串口初始化为9600
LED_Init();   //初始化LED
LCD_Init();
Adc_Init();
EXTIX_Init();
OSInit();   
  OSTaskCreate(start_task,(void *)0,(OS_STK *)&START_TASK_STK[START_STK_SIZE-1],START_TASK_PRIO );//创建起始任务
OSStart();  
}

void start_task(void *pdata)
{
    u8 err;
OS_CPU_SR cpu_sr=0;
pdata = pdata;
POINT_COLOR=BRED;//设置字体为红色
LCD_ShowString(100,90,"CPU: 00 %"); 
POINT_COLOR=BLUE;//设置字体为蓝色
LCD_ShowString(60,130,"ADC_CH0_VAL:");      
LCD_ShowString(60,150,"ADC_CH0_VOL:0.000V");
LCD_ShowString(60,170,"TEMP_VAL:");      
LCD_ShowString(60,190,"TEMP_VOL:0.000V");      
LCD_ShowString(60,210,"TEMPERATE:00.00C");
myMBox=OSMboxCreate((void*)0); //创建消息邮箱
OSStatInit(); //初始化统计任务.这里会延时1秒钟左右,若要使用统计任务必须调用该函数
  OS_ENTER_CRITICAL(); //进入临界区(无法被中断打断)    
  OSTaskCreate(sw_task,(void *)0,(OS_STK*)&SW_TASK_STK[SW_STK_SIZE-1],SW_TASK_PRIO);
OSTaskCreate(adc_task,(void *)0,(OS_STK*)&ADC_TASK_STK[ADC_STK_SIZE-1],ADC_TASK_PRIO);   
  OSTaskCreate(wen_task,(void *)0,(OS_STK*)&WEN_TASK_STK[WEN_STK_SIZE-1],WEN_TASK_PRIO);
OSTaskCreate(cpu_task,(void *)0,(OS_STK*)&CPU_TASK_STK[CPU_STK_SIZE-1],CPU_TASK_PRIO);   
OSTaskSuspend(START_TASK_PRIO); //挂起起始任务.
OS_EXIT_CRITICAL(); //退出临界区(可以被中断打断)
}

void sw_task(void *pdata)
{
u8 bz1=0;
u8 bz2=0;
LCD_ShowString(110,70,"START");
while(1)
{
u32 *sign;
u8 err;
sign=(u32*)OSMboxPend(myMBox,0,&err);
switch(*sign)
{
case 1:
bz1=!bz1;
if(bz1)
{
OSTaskSuspend(ADC_TASK_PRIO);
}
else
{
OSTaskResume(ADC_TASK_PRIO);
}
break;
case 2:
bz2=!bz2;
if(bz2)
{
OSTaskSuspend(WEN_TASK_PRIO);
}
else
{
OSTaskResume(WEN_TASK_PRIO);
}
break;
}
OSTimeDly(10);
}
}

void adc_task(void *pdata)
{
while(1)
{
u8 i,j;
u16 adc[READ_TIMES];  //数组用于保存AD的值
u16 adcx=0;           //AD的平均值
u16 temp,temp1,temp3;
float temp2;
u32 sum=0;
for(i=0;i<READ_TIMES;i++) //多次读取AD值
{
adc=Get_Adc(ADC_CH0);
}
   
for(i=0;i<READ_TIMES-1; i++)//排序
{
for(j=i+1;j<READ_TIMES;j++)
{
if(adc>adc[j])//升序排列
{
temp=adc;
adc=adc[j];
adc[j]=temp;
}
}
}
for(i=LOST_VAL;i<READ_TIMES-LOST_VAL;i++) //计算平均值
{
sum+=adc;
adcx=sum/(READ_TIMES-2*LOST_VAL);  
}
//屏幕上显示
LCD_ShowNum(156,130,adcx,4,16);//显示ADC的值
temp2=(float)adcx*(3.3/4096);
adcx=temp2;
LCD_ShowNum(156,150,adcx,1,16);//显示电压值整数部分
temp2-=adcx;
temp2*=1000;
LCD_ShowNum(172,150,temp2,3,16);//显示电压值小数部分
LED0=!LED0;

// //将电压值以ASCII码发送出去
// temp3=temp2;         //temp2为float,temp,temp3为整型u16
// temp=adcx+48;
// USART1->DR=temp; //发送整数位ASCII码
// while((USART1->SR&0X40)==0);//等待发送结束
// USART1->DR='.';
// while((USART1->SR&0X40)==0);
// temp1=100;
// for(i=0;i<3;i++) //发送小数位ASCII码
// {
// temp=temp3/temp1;
// USART1->DR=temp+48;
// while((USART1->SR&0X40)==0);//等待发送结束
// temp3=temp3%temp1;
// temp1=temp1/10;
// }
// USART1->DR='V';
// while((USART1->SR&0X40)==0);//等待发送结束
// USART1->DR=' ';
// while((USART1->SR&0X40)==0);//等待发送结束
OSTimeDly(25);
};
}

void wen_task(void *pdata)
{
while(1)
{
u8 i,j;
u16 adc[READ_TIMES];  //数组用于保存AD的值
u16 adcx=0;           //AD的平均值
u16 temp,temp1,temp3;
float temp2,temperate;
u32 sum=0;
LED1=!LED1;
for(i=0;i<READ_TIMES;i++) //多次读取AD值
{
adc=Get_Adc(ADC_WEN);
}
   
for(i=0;i<READ_TIMES-1; i++)//排序
{
for(j=i+1;j<READ_TIMES;j++)
{
if(adc>adc[j])//升序排列
{
temp=adc;
adc=adc[j];
adc[j]=temp;
}
}
}
for(i=LOST_VAL;i<READ_TIMES-LOST_VAL;i++) //计算平均值
{
sum+=adc;
adcx=sum/(READ_TIMES-2*LOST_VAL);  
}
LCD_ShowNum(132,170,adcx,4,16);//显示ADC的值
temp2=(float)adcx*(3.3/4096);
temperate=temp2;//保存温度传感器的电压值
adcx=temp2;
LCD_ShowNum(132,190,adcx,1,16);     //显示电压值整数部分
temp2-=(u8)temp;     //减掉整数部分  
LCD_ShowNum(148,190,temp2*1000,3,16);//显示电压小数部分
 
temperate=(1.43-temperate)/0.0043+25;//计算出当前温度值  
LCD_ShowNum(140,210,(u8)temperate,2,16); //显示温度整数部分
temperate-=(u8)temperate;  
LCD_ShowNum(164,210,temperate*100,2,16);//显示温度小数部分
LED1=!LED1;
OSTimeDly(50);
};
}

void cpu_task(void *pdata)
{
while(1)
{
LCD_ShowNum(140,90,OSCPUUsage,2,16);   //显示CPU利用率
OSTimeDly(100);
}
}

正点原子逻辑分析仪DL16劲爆上市
回复

使用道具 举报

530

主题

11万

帖子

34

精华

管理员

Rank: 12Rank: 12Rank: 12

积分
165353
金钱
165353
注册时间
2010-12-1
在线时间
2108 小时
发表于 2014-4-4 22:25:11 | 显示全部楼层
把这个任务里面的while(1)里面的内容全部删了,只剩下delay试试。
我是开源电子网www.openedv.com站长,有关站务问题请与我联系。
正点原子STM32开发板购买店铺http://openedv.taobao.com
正点原子官方微信公众平台,点击这里关注“正点原子”
回复 支持 反对

使用道具 举报

1

主题

10

帖子

0

精华

新手上路

积分
34
金钱
34
注册时间
2013-10-22
在线时间
0 小时
 楼主| 发表于 2014-4-4 22:51:45 | 显示全部楼层
回复【2楼】正点原子:
------------------------------
可能是我系统移植的问题,我试了一下最简单的任务也只运行了几次,后面就不能运行了,但是UCOSII这个文件夹我直接用“MINISTM32 扩展实验30 UCOSII消息邮箱测试”中的UCOSII文件夹,结果还是一样,这不知道出了什么问题
回复 支持 反对

使用道具 举报

530

主题

11万

帖子

34

精华

管理员

Rank: 12Rank: 12Rank: 12

积分
165353
金钱
165353
注册时间
2010-12-1
在线时间
2108 小时
发表于 2014-4-4 23:21:11 | 显示全部楼层
回复【3楼】469219945:
---------------------------------
直接下载我们的例程,总可以跑吧?
我是开源电子网www.openedv.com站长,有关站务问题请与我联系。
正点原子STM32开发板购买店铺http://openedv.taobao.com
正点原子官方微信公众平台,点击这里关注“正点原子”
回复 支持 反对

使用道具 举报

1

主题

10

帖子

0

精华

新手上路

积分
34
金钱
34
注册时间
2013-10-22
在线时间
0 小时
 楼主| 发表于 2014-4-5 00:14:45 | 显示全部楼层
回复【4楼】正点原子:
---------------------------------
可以
回复 支持 反对

使用道具 举报

1

主题

10

帖子

0

精华

新手上路

积分
34
金钱
34
注册时间
2013-10-22
在线时间
0 小时
 楼主| 发表于 2014-4-5 13:02:53 | 显示全部楼层
回复【4楼】正点原子:
---------------------------------
为什么连这么简单的程序都跑不了啊?见网上有人分析可能死在空闲任务了!
真不知道怎么解决!
#include"sys.h"
#include"delay.h"
#include"usart.h"
#include "led.h"
#include "adc.h"
#include "lcd.h"
#include"exti.h"
#include"includes.h"

//START 任务
#define START_TASK_PRIO       9 //设置任务优先级
#define START_STK_SIZE   512 //设置任务堆栈大小
OS_STK START_TASK_STK[START_STK_SIZE];     //任务堆栈
void start_task(void *pdata);              //任务函数
     
//ADC任务
#define ADC_TASK_PRIO        6  //设置任务优先级
#define ADC_STK_SIZE        512 //设置任务堆栈大小
OS_STK ADC_TASK_STK[ADC_STK_SIZE];    //任务堆栈
void adc_task(void *pdata);    //任务函数

OS_EVENT *myMBox;//消息邮箱

int main(void)
{     
delay_init();      //延时初始化
NVIC_Configuration();
LED_Init();     //初始化LED
OSInit();   
  OSTaskCreate(start_task,(void *)0,(OS_STK *)&START_TASK_STK[START_STK_SIZE-1],START_TASK_PRIO );//创建起始任务
OSStart();  
}

void start_task(void *pdata)
{
    u8 err;
OS_CPU_SR cpu_sr=0;
pdata = pdata;
   OS_ENTER_CRITICAL(); //进入临界区(无法被中断打断)    
OSTaskCreate(adc_task,(void *)0,(OS_STK*)&ADC_TASK_STK[ADC_STK_SIZE-1],ADC_TASK_PRIO);        
OSTaskSuspend(START_TASK_PRIO); //挂起起始任务.
OS_EXIT_CRITICAL(); //退出临界区(可以被中断打断)
}
void adc_task(void *pdata)

while(1)
{
LED1=!LED1;
OSTimeDly(100);
};
}
回复 支持 反对

使用道具 举报

530

主题

11万

帖子

34

精华

管理员

Rank: 12Rank: 12Rank: 12

积分
165353
金钱
165353
注册时间
2010-12-1
在线时间
2108 小时
发表于 2014-4-5 13:37:08 | 显示全部楼层
回复***5楼*** 469219945:
---------------------------------
你的systick配置正常么?
你仿真看看systick的中断,是不是一直可以进入?
我是开源电子网www.openedv.com站长,有关站务问题请与我联系。
正点原子STM32开发板购买店铺http://openedv.taobao.com
正点原子官方微信公众平台,点击这里关注“正点原子”
回复 支持 反对

使用道具 举报

1

主题

10

帖子

0

精华

新手上路

积分
34
金钱
34
注册时间
2013-10-22
在线时间
0 小时
 楼主| 发表于 2014-4-5 14:30:16 | 显示全部楼层
回复【7楼】正点原子:
---------------------------------
刚才试了一下,假如将SYSTEM文件夹换成“MINISTM32 扩展实验30 UCOSII消息邮箱测试”中的SYSTEM文件夹就可以了,我之前用的是战舰库函数版本的SYSTEM文件夹,就不行,但是我读了一下原理上没什么错误啊,只是前一个SYSTEM文件夹是用寄存器写的,后一个是用库函数写的,我的板子是mini的,原子哥,这是怎么回事啊?
回复 支持 反对

使用道具 举报

530

主题

11万

帖子

34

精华

管理员

Rank: 12Rank: 12Rank: 12

积分
165353
金钱
165353
注册时间
2010-12-1
在线时间
2108 小时
发表于 2014-4-5 19:01:18 | 显示全部楼层
回复【8楼】469219945:
---------------------------------
SYSTEM文件夹要支持UCOSII的版本的才可以,且在sys.h里面,要设置SYSTEM_SUPPORT_UCOS 为1,才行。
我是开源电子网www.openedv.com站长,有关站务问题请与我联系。
正点原子STM32开发板购买店铺http://openedv.taobao.com
正点原子官方微信公众平台,点击这里关注“正点原子”
回复 支持 反对

使用道具 举报

1

主题

10

帖子

0

精华

新手上路

积分
34
金钱
34
注册时间
2013-10-22
在线时间
0 小时
 楼主| 发表于 2014-4-5 21:00:12 | 显示全部楼层
回复【9楼】正点原子:
---------------------------------
谢谢原子哥啊!搞定了!
回复 支持 反对

使用道具 举报

1

主题

10

帖子

0

精华

新手上路

积分
34
金钱
34
注册时间
2013-10-22
在线时间
0 小时
 楼主| 发表于 2014-4-5 21:15:07 | 显示全部楼层
回复【9楼】正点原子:
---------------------------------
还有个问题需要问一下原子哥,任务堆栈的大小事根据什么分配的?怎么才能分配的合适,使任务既能正常工作,又节约资源!
回复 支持 反对

使用道具 举报

530

主题

11万

帖子

34

精华

管理员

Rank: 12Rank: 12Rank: 12

积分
165353
金钱
165353
注册时间
2010-12-1
在线时间
2108 小时
发表于 2014-4-6 00:05:18 | 显示全部楼层
回复【11楼】469219945:
---------------------------------
这个我是靠自己估计的,一般最小是64,然后设置到不死机为标准
我是开源电子网www.openedv.com站长,有关站务问题请与我联系。
正点原子STM32开发板购买店铺http://openedv.taobao.com
正点原子官方微信公众平台,点击这里关注“正点原子”
回复 支持 反对

使用道具 举报

3

主题

16

帖子

0

精华

新手上路

积分
47
金钱
47
注册时间
2014-2-11
在线时间
9 小时
发表于 2016-8-24 14:22:47 | 显示全部楼层
回复 支持 反对

使用道具 举报

0

主题

5

帖子

0

精华

新手上路

积分
28
金钱
28
注册时间
2019-6-5
在线时间
7 小时
发表于 2020-9-2 10:42:26 | 显示全部楼层
我也遇到了这样的情况,然后发现我在库函数注释掉这个函数之后,没有重新写一个;
新写一个就好了

void SysTick_Handler(void)
{       
        OSIntEnter();                                                        //进入中断
        OSTimeTick();                                               //调用ucos的时钟服务程序               
        OSIntExit();                                                        //触发任务切换软中断
}
回复 支持 反对

使用道具 举报

您需要登录后才可以回帖 登录 | 立即注册

本版积分规则



关闭

原子哥极力推荐上一条 /2 下一条

正点原子公众号

QQ|手机版|OpenEdv-开源电子网 ( 粤ICP备12000418号-1 )

GMT+8, 2024-11-25 13:51

Powered by OpenEdv-开源电子网

© 2001-2030 OpenEdv-开源电子网

快速回复 返回顶部 返回列表