當(dāng)前位置:首頁 > 單片機 > 單片機
[導(dǎo)讀]在“51操作系統(tǒng)學(xué)習(xí)筆記(一)”里,已經(jīng)掌握了并行多任務(wù)的實現(xiàn),和利用修改sp內(nèi)容,子程序結(jié)束調(diào)用ret來實現(xiàn)程序跳轉(zhuǎn)。但程序沒有涉及中斷,任務(wù)中有中斷,程序會失去控制。這一單元,要學(xué)習(xí)帶中斷問題的多任務(wù)操作系

在“51操作系統(tǒng)學(xué)習(xí)筆記(一)”里,已經(jīng)掌握了并行多任務(wù)的實現(xiàn),和利用修改sp內(nèi)容,子程序結(jié)束調(diào)用ret來實現(xiàn)程序跳轉(zhuǎn)。但程序沒有涉及中斷,任務(wù)中有中斷,程序會失去控制。

這一單元,要學(xué)習(xí)帶中斷問題的多任務(wù)操作系統(tǒng):"在51單片機下具有延時功能占先式內(nèi)核的操作系統(tǒng)"

在網(wǎng)上找的這個程序比較簡短,也符合本單元的學(xué)習(xí)任務(wù)。

不過有一個麻煩,這個程序用到了52擴展的一些功能,需要先學(xué)習(xí)89c52單片機的一些寄存器和計數(shù)器使用。

全部源代碼如下:程序詳細講解,和源自于http://www.51hei.com/mcu/1325.html

//可以直接拷貝到keil中調(diào)試

#include

#define MAX_TASKS 5

typedef struct os_task_control_table {
unsigned char os_task_wait_tick;
unsigned char os_task_stack_top;
}TCB;

volatile unsigned char int_count;
volatile unsigned char os_en_cr_count;
#define enter_int() EA=0;int_count++;
#define os_enter_critical() EA=0;os_en_cr_count++;//關(guān)中斷,關(guān)中斷次數(shù)增加1
#define os_exit_critical() if(os_en_cr_count>=1){os_en_cr_count--;if(os_en_cr_count==0)EA=1;} //關(guān)閉中斷
unsigned char code os_map_tbl[] = {0x01, 0x02, 0x04, 0x08, 0x10, 0x20, 0x40, 0x80};//任務(wù)的狀態(tài)字

volatile unsigned char os_task_int_tbl;//任務(wù)中斷表
idata volatile TCB os_tcb[MAX_TASKS];//任務(wù)控制表數(shù)組
volatile unsigned char os_task_running_id;
volatile unsigned char os_task_rdy_tbl;//任務(wù)準(zhǔn)備
unsigned char idata os_task_stack[MAX_TASKS][20];//任務(wù)堆棧,每個任務(wù)有20字節(jié)的堆棧

void os_init(void);
void os_task_create(unsigned char task_id ,unsigned int task_point,unsigned char stack_point);
void os_delay(unsigned char ticks);
void os_start(void);
void os_task_switch(void);
void exit_int(void);


//操作系統(tǒng)初始化
void os_init(void) {
EA = 0; //關(guān)閉中斷
ET2 = 1;
T2CON = 0X00;
T2MOD = 0X00; //關(guān)計數(shù)器
RCAP2H = 0x0D8;
RCAP2L = 0x0F0; //計數(shù)器2的初始化
os_task_rdy_tbl = 0; //
os_task_int_tbl = 0xff; //中斷
int_count = 0; //中斷的次數(shù)
os_en_cr_count = 0; //關(guān)閉中斷的次數(shù)
}


//任務(wù)加載
void os_task_create(unsigned char task_id ,unsigned int task_point,unsigned char stack_point) {
os_enter_critical();//關(guān)中斷,程序段中不能被打斷
((unsigned char idata *)stack_point)[0] = task_point;//task_point是16位的,把任務(wù)的程序地址記錄下來
((unsigned char idata *)stack_point)[1] = task_point>>8;//右移8位,高位地址在上面
os_tcb[task_id].os_task_stack_top = stack_point+14;//堆棧結(jié)構(gòu):有兩字節(jié)程序地址,13字節(jié)的其他數(shù)據(jù),所以棧頂部是加14(0-14)
os_task_rdy_tbl |= os_map_tbl[task_id];//任務(wù)的狀態(tài)就緒
os_tcb[task_id].os_task_wait_tick = 0;//裝載任務(wù),任務(wù)的等待時間就是0
os_exit_critical(); //開中斷
}

//系統(tǒng)延時
void os_delay(unsigned char ticks) {
os_enter_critical();//
os_tcb[os_task_running_id].os_task_wait_tick = ticks;
os_task_rdy_tbl &= ~os_map_tbl[os_task_running_id];
os_exit_critical();
os_task_switch(); //換下一個任務(wù)
}


//任務(wù)開始
void os_start(void) {
os_task_running_id = 0; //由任務(wù)0開始
os_tcb[os_task_running_id].os_task_stack_top -= 13; //堆棧頂跳到任務(wù)一的程序地址處的堆棧
EA = 1; //開中斷
SP = os_tcb[os_task_running_id].os_task_stack_top; //把任務(wù)一的程序地址給sp
TR2 = 1; //T2計數(shù)器打開,該計數(shù)器是52擴展的計數(shù)器
} //os_start(void)執(zhí)行完,就實現(xiàn)RET,sp的內(nèi)容賦給pc,會接著執(zhí)行任務(wù)一


//任務(wù)轉(zhuǎn)換
void os_task_switch(void) {
unsigned char i;
EA = 0;
os_tcb[os_task_running_id].os_task_stack_top = SP;
os_task_int_tbl &= ~os_map_tbl[os_task_running_id];
for(i=0; iif(os_task_rdy_tbl&os_map_tbl[i]) {
break;
}
}
os_task_running_id = i;
SP = os_tcb[os_task_running_id].os_task_stack_top;
if(os_task_int_tbl&os_map_tbl[os_task_running_id]) {
__asm POP 7 //Keil51的中斷處理會先讓這13個寄存器入棧,所以這里要出棧
__asm POP 6

__asm POP 5
__asm POP 4
__asm POP 3
__asm POP 2
__asm POP 1
__asm POP 0
__asm POP PSW
__asm POP DPL //13字節(jié)堆棧
__asm POP DPH
__asm POP B
__asm POP ACC
}
EA = 1;
__asm RETI //從中斷服務(wù)返回主程序
}


//中斷結(jié)束
void exit_int(void) {
unsigned char i;
SP -= 2;

if(--int_count == 0)
{ //如果中斷數(shù)=0
os_tcb[os_task_running_id].os_task_stack_top = SP;
os_task_int_tbl |= os_map_tbl[os_task_running_id];
for(i=0; i { //順序檢查任務(wù)是否準(zhǔn)備好
if(os_task_rdy_tbl&os_map_tbl[i])
{
break;
}
}
os_task_running_id = i; //把就緒的任務(wù)號給任務(wù)運行號碼
SP = os_tcb[os_task_running_id].os_task_stack_top; //堆棧指向運行任務(wù)
if(os_task_int_tbl&os_map_tbl[os_task_running_id])
{
__asm POP 7
__asm POP 6 //恢復(fù)任務(wù)寄存器
__asm POP 5
__asm POP 4
__asm POP 3
__asm POP 2
__asm POP 1
__asm POP 0
__asm POP PSW
__asm POP DPL
__asm POP DPH
__asm POP B
__asm POP ACC
}
EA = 1;
__asm RETI
}

//如果中斷數(shù)還不為0
__asm POP 7
__asm POP 6 //恢復(fù)任務(wù)寄存器
__asm POP 5
__asm POP 4
__asm POP 3
__asm POP 2
__asm POP 1
__asm POP 0
__asm POP PSW
__asm POP DPL
__asm POP DPH
__asm POP B
__asm POP ACC
EA=1;
__asm RETI //計數(shù)器2中斷服務(wù)程序結(jié)束,返回主程序
}


//滴答中斷程序,計數(shù)器2的中斷
void timer2_isr(void) interrupt 5 {
unsigned char i;
TF2=0;
enter_int();
for(i=0; iif(os_tcb[i].os_task_wait_tick) {
os_tcb[i].os_task_wait_tick--;
if(os_tcb[i].os_task_wait_tick == 0) {
os_task_rdy_tbl |= os_map_tbl[i];
}
}
}
exit_int(); //
}


//下面是任務(wù)實現(xiàn)代碼
void task_0(void) { //空循環(huán)
while(1) {

}
}

sbit seg2 = P2^5;
sbit seg3 = P2^6;
sbit seg4 = P2^7;

void delay_ms(unsigned int xms){ //只是程序延時
unsigned int x,y;
for(x=xms; x>0; x--)
for(y=248; y>0; y--);
}

unsigned char code table[]={0x3f,0x06,0x5b,0x4f,0x66,0x6d,0x7d,0x07,0x7f,0x6f,0x77,0x7c,0x39,0x5e,0x79,0x71,0x40,0};

void task_1(void) { //提取task_2的等待時間,顯示在數(shù)碼管上
unsigned char gw,sw,bw;
while(1) {
bw = os_tcb[2].os_task_wait_tick/100;
sw = os_tcb[2].os_task_wait_tick%100/10;
gw = os_tcb[2].os_task_wait_tick%10;
P0 = table[bw];
seg2=0;
delay_ms(3);
seg2=1;
P0 = table[sw];
seg3=0;
delay_ms(3);
seg3=1;
P0 = table[gw];
seg4=0;
delay_ms(3);
seg4=1;
}
}

void task_2(void) {
unsigned char i;
while(1) {
i++;
P3 = 0x01<<(i%8);
os_delay(200);
}
}

void task_3(void) {
unsigned char i;
while(1) {
i++;
//P2 = 0x01<<(i%8);
os_delay(7);
}
}

void task_4(void) {
unsigned char i;
while(1) {
i++;
P1 = 0x01<<(i%8);
os_delay(10);
}
}


void main(void) {
os_init();
os_task_create(4,(unsigned int)&task_0,(unsigned char)os_task_stack[4]);
os_task_create(3,(unsigned int)&task_1,(unsigned char)os_task_stack[3]);
os_task_create(2,(unsigned int)&task_2,(unsigned char)os_task_stack[2]);
os_task_create(1,(unsigned int)&task_3,(unsigned char)os_task_stack[1]);
os_task_create(0,(unsigned int)&task_4,(unsigned char)os_task_stack[0]);
os_start();
}


本站聲明: 本文章由作者或相關(guān)機構(gòu)授權(quán)發(fā)布,目的在于傳遞更多信息,并不代表本站贊同其觀點,本站亦不保證或承諾內(nèi)容真實性等。需要轉(zhuǎn)載請聯(lián)系該專欄作者,如若文章內(nèi)容侵犯您的權(quán)益,請及時聯(lián)系本站刪除。
換一批
延伸閱讀

9月2日消息,不造車的華為或?qū)⒋呱龈蟮莫毥谦F公司,隨著阿維塔和賽力斯的入局,華為引望愈發(fā)顯得引人矚目。

關(guān)鍵字: 阿維塔 塞力斯 華為

加利福尼亞州圣克拉拉縣2024年8月30日 /美通社/ -- 數(shù)字化轉(zhuǎn)型技術(shù)解決方案公司Trianz今天宣布,該公司與Amazon Web Services (AWS)簽訂了...

關(guān)鍵字: AWS AN BSP 數(shù)字化

倫敦2024年8月29日 /美通社/ -- 英國汽車技術(shù)公司SODA.Auto推出其旗艦產(chǎn)品SODA V,這是全球首款涵蓋汽車工程師從創(chuàng)意到認證的所有需求的工具,可用于創(chuàng)建軟件定義汽車。 SODA V工具的開發(fā)耗時1.5...

關(guān)鍵字: 汽車 人工智能 智能驅(qū)動 BSP

北京2024年8月28日 /美通社/ -- 越來越多用戶希望企業(yè)業(yè)務(wù)能7×24不間斷運行,同時企業(yè)卻面臨越來越多業(yè)務(wù)中斷的風(fēng)險,如企業(yè)系統(tǒng)復(fù)雜性的增加,頻繁的功能更新和發(fā)布等。如何確保業(yè)務(wù)連續(xù)性,提升韌性,成...

關(guān)鍵字: 亞馬遜 解密 控制平面 BSP

8月30日消息,據(jù)媒體報道,騰訊和網(wǎng)易近期正在縮減他們對日本游戲市場的投資。

關(guān)鍵字: 騰訊 編碼器 CPU

8月28日消息,今天上午,2024中國國際大數(shù)據(jù)產(chǎn)業(yè)博覽會開幕式在貴陽舉行,華為董事、質(zhì)量流程IT總裁陶景文發(fā)表了演講。

關(guān)鍵字: 華為 12nm EDA 半導(dǎo)體

8月28日消息,在2024中國國際大數(shù)據(jù)產(chǎn)業(yè)博覽會上,華為常務(wù)董事、華為云CEO張平安發(fā)表演講稱,數(shù)字世界的話語權(quán)最終是由生態(tài)的繁榮決定的。

關(guān)鍵字: 華為 12nm 手機 衛(wèi)星通信

要點: 有效應(yīng)對環(huán)境變化,經(jīng)營業(yè)績穩(wěn)中有升 落實提質(zhì)增效舉措,毛利潤率延續(xù)升勢 戰(zhàn)略布局成效顯著,戰(zhàn)新業(yè)務(wù)引領(lǐng)增長 以科技創(chuàng)新為引領(lǐng),提升企業(yè)核心競爭力 堅持高質(zhì)量發(fā)展策略,塑強核心競爭優(yōu)勢...

關(guān)鍵字: 通信 BSP 電信運營商 數(shù)字經(jīng)濟

北京2024年8月27日 /美通社/ -- 8月21日,由中央廣播電視總臺與中國電影電視技術(shù)學(xué)會聯(lián)合牽頭組建的NVI技術(shù)創(chuàng)新聯(lián)盟在BIRTV2024超高清全產(chǎn)業(yè)鏈發(fā)展研討會上宣布正式成立。 活動現(xiàn)場 NVI技術(shù)創(chuàng)新聯(lián)...

關(guān)鍵字: VI 傳輸協(xié)議 音頻 BSP

北京2024年8月27日 /美通社/ -- 在8月23日舉辦的2024年長三角生態(tài)綠色一體化發(fā)展示范區(qū)聯(lián)合招商會上,軟通動力信息技術(shù)(集團)股份有限公司(以下簡稱"軟通動力")與長三角投資(上海)有限...

關(guān)鍵字: BSP 信息技術(shù)
關(guān)閉
關(guān)閉