0
  • 聊天消息
  • 系統(tǒng)消息
  • 評(píng)論與回復(fù)
登錄后你可以
  • 下載海量資料
  • 學(xué)習(xí)在線課程
  • 觀看技術(shù)視頻
  • 寫文章/發(fā)帖/加入社區(qū)
會(huì)員中心
創(chuàng)作中心

完善資料讓更多小伙伴認(rèn)識(shí)你,還能領(lǐng)取20積分哦,立即完善>

3天內(nèi)不再提示

單片機(jī)main()函數(shù)退出之后,程序會(huì)跑去哪里?

dyquk4xk2p3d ? 來源:TsinghuaJoking ? 2023-03-30 10:01 ? 次閱讀

對(duì)于嵌入式系統(tǒng),如果沒有運(yùn)行RTOS,那么程序開發(fā)中的主函數(shù)main()需要通過某種機(jī)制使其永遠(yuǎn)愉快的運(yùn)行下去,它沒有終點(diǎn)。如果想從main函數(shù)中退出,具體干什么是由所使用的C語言編譯器決定的。

一、問題提出

單片機(jī)led模塊定義函數(shù)中看到一個(gè)有趣的問題。提問者在進(jìn)行基本的C51編程實(shí)驗(yàn),編寫了一個(gè)簡(jiǎn)單的C51程序如下:

#include

voidtest(num){
switch(num){
case1:P2_0=0;P2_1=0;
break;
}
}

voidmain(void){
test(1);
}

程序執(zhí)行完之后,可以看到實(shí)驗(yàn)板上的有兩個(gè)LED被點(diǎn)亮,另外六個(gè)居然微微發(fā)亮。

a8229fca-ce8a-11ed-bfe3-dac502259ad0.png

如果在主程序中,增加一個(gè)無限循環(huán):while(1); ,則電路板上的就不再會(huì)出現(xiàn)“微微點(diǎn)亮”的現(xiàn)象了。

#include

voidtest(num){
switch(num){
case1:P2_0=0;P2_1=0;
break;
}
}

voidmain(void){
test(1);
while(1);
}
a8c10372-ce8a-11ed-bfe3-dac502259ad0.png

上面兩種情況的區(qū)別,在于第二個(gè)程序中主循環(huán) main()函數(shù)始終沒有退出,而第一個(gè)程序,main()函數(shù)退出了。似乎前面LED微微點(diǎn)亮 應(yīng)該與主函數(shù)退出之后,單片機(jī)都干了些啥有關(guān)系。

那么就剩下一個(gè)問題:對(duì)于普通的嵌入式系統(tǒng),C語言編程中main()函數(shù)退出之后,程序去哪兒了?

二、程序去哪兒了?

從上面提問者書寫的代碼來看,應(yīng)該是一位C51的愛好者,使用的是C51的編譯器,在一款C51開發(fā)板上愉快的進(jìn)行實(shí)驗(yàn)。他一開始沒有安裝嵌入式程序開發(fā)的慣例 在主程序void main(void)中利用無限循環(huán)將程序控制在主程序函數(shù)中,就出現(xiàn)了前面實(shí)驗(yàn)結(jié)果中令人迷惑的情況。

注:他是一個(gè)膽大心細(xì)的人,觀察還挺仔細(xì)的。

2.1 盤古開天辟地

對(duì)于C語言編程來說,所有的用戶程序世界是從主程序main()開始的。給用戶程序開天辟地的任務(wù)是由一小段盤古代碼STARTUP.A51。

51單片機(jī)程序執(zhí)行流程(STARTUP.A51管理Main函數(shù)的執(zhí)行)

下面截取了STARTUP.A51 代碼的一段,可以看到盤古在單片機(jī)RESET之后做了點(diǎn)準(zhǔn)備工作(初始化全局變量、堆棧指針)之后,就直接跳轉(zhuǎn)至:?C_START

NAME?C_STARTUP

?C_C51STARTUPSEGMENTCODE
?STACKSEGMENTIDATA

RSEG?STACK
DS1

EXTRNCODE(?C_START)
PUBLIC?C_STARTUP

CSEGAT0
?C_STARTUP:LJMPSTARTUP1

RSEG?C_C51STARTUP

STARTUP1:

IFIDATALEN<>0
MOVR0,#IDATALEN-1
CLRA
IDATALOOP:MOV@R0,A
DJNZR0,IDATALOOP
ENDIF

IFXDATALEN<>0
MOVDPTR,#XDATASTART
MOVR7,#LOW(XDATALEN)
IF(LOW(XDATALEN))<>0
MOVR6,#(HIGH(XDATALEN))+1
ELSE
MOVR6,#HIGH(XDATALEN)
ENDIF
CLRA
XDATALOOP:MOVX@DPTR,A
INCDPTR
DJNZR7,XDATALOOP
DJNZR6,XDATALOOP
ENDIF

IFPPAGEENABLE<>0
MOVPPAGE_SFR,#PPAGE
ENDIF

IFPDATALEN<>0
MOVR0,#LOW(PDATASTART)
MOVR7,#LOW(PDATALEN)
CLRA
PDATALOOP:MOVX@R0,A
INCR0
DJNZR7,PDATALOOP
ENDIF

IFIBPSTACK<>0
EXTRNDATA(?C_IBP)

MOV?C_IBP,#LOWIBPSTACKTOP
ENDIF

IFXBPSTACK<>0
EXTRNDATA(?C_XBP)

MOV?C_XBP,#HIGHXBPSTACKTOP
MOV?C_XBP+1,#LOWXBPSTACKTOP
ENDIF

IFPBPSTACK<>0
EXTRNDATA(?C_PBP)
MOV?C_PBP,#LOWPBPSTACKTOP
ENDIF

MOVSP,#?STACK-1
LJMP?C_START

END

上面的代碼也被博文51單片機(jī)程序執(zhí)行流程(STARTUP.A51)中進(jìn)行逐步調(diào)試跟蹤驗(yàn)證過:

a92449c8-ce8a-11ed-bfe3-dac502259ad0.png

2.2 世界盡頭

由于進(jìn)入main()函數(shù)是長(zhǎng)跳轉(zhuǎn),所以main函數(shù)是不會(huì)正常返回到啟動(dòng)程序STARTUP.A51,那么程序去哪了?

在博文單片機(jī)C語言while(1)的問題中作者對(duì)于KEIL編譯器和PIC的MAPLAB編譯器對(duì)于main函數(shù)的最后時(shí)光進(jìn)行了反匯編查看。

Keil編譯器

在main函數(shù)的最后,程序增加了一下幾行代碼:

MOVR0,#0x7F
CLRA
MOV@R0,A
DJNZR0,(3)
MOVSP,#0x0C
LJMPmain

這幾條語句,前4條,是將我們單片機(jī)的內(nèi)存的前128個(gè)地址清零,第5條,是定義堆棧,第6條,是將程序重新跳轉(zhuǎn)到main函數(shù)的首行進(jìn)行執(zhí)行。

MAPLAB編譯器

PIC 單片機(jī)語言程序進(jìn)行跟蹤,發(fā)現(xiàn)main() 函數(shù)最后一條語句為 reset,也就是單片機(jī)直接復(fù)位,這是 MAPLAB編譯器根據(jù) PIC 單片機(jī)特點(diǎn)增加的復(fù)位語句。

總結(jié)

對(duì)于嵌入式系統(tǒng),如果沒有運(yùn)行RTOS,那么程序開發(fā)中的主函數(shù)(main())需要通過某種機(jī)制使其永遠(yuǎn)愉快的運(yùn)行下去,它沒有終點(diǎn)。如果想從main函數(shù)中退出,具體干什么是由所使用的C語言編譯器決定的。

審核編輯:湯梓紅

聲明:本文內(nèi)容及配圖由入駐作者撰寫或者入駐合作網(wǎng)站授權(quán)轉(zhuǎn)載。文章觀點(diǎn)僅代表作者本人,不代表電子發(fā)燒友網(wǎng)立場(chǎng)。文章及其配圖僅供工程師學(xué)習(xí)之用,如有內(nèi)容侵權(quán)或者其他違規(guī)問題,請(qǐng)聯(lián)系本站處理。 舉報(bào)投訴
  • C語言
    +關(guān)注

    關(guān)注

    180

    文章

    7594

    瀏覽量

    135861
  • 函數(shù)
    +關(guān)注

    關(guān)注

    3

    文章

    4277

    瀏覽量

    62323
  • RTOS
    +關(guān)注

    關(guān)注

    21

    文章

    809

    瀏覽量

    119361
  • main
    +關(guān)注

    關(guān)注

    0

    文章

    38

    瀏覽量

    6128
  • maplab
    +關(guān)注

    關(guān)注

    0

    文章

    3

    瀏覽量

    1252

原文標(biāo)題:?jiǎn)纹瑱C(jī)main()函數(shù)退出之后,程序會(huì)跑去哪里?

文章出處:【微信號(hào):良許Linux,微信公眾號(hào):良許Linux】歡迎添加關(guān)注!文章轉(zhuǎn)載請(qǐng)注明出處。

收藏 人收藏

    評(píng)論

    相關(guān)推薦

    單片機(jī)main函數(shù)在中斷函數(shù)里執(zhí)行?

    最近看了硬漢分享的一個(gè)內(nèi)容:為什么復(fù)位中斷服務(wù)程序里面直接調(diào)用的main函數(shù),難道所有程序都在復(fù)位中斷里面執(zhí)行的?
    發(fā)表于 10-24 11:04 ?919次閱讀

    請(qǐng)問單片機(jī)main函數(shù)是執(zhí)行一次,還是會(huì)反復(fù)執(zhí)行?

    ; //電機(jī)運(yùn)行一段時(shí)間motor = 0;//電機(jī)停}按程序邏輯,電機(jī)應(yīng)該是運(yùn)行一會(huì)就停止的,但是下載到板子之后,電機(jī)一直運(yùn)行。在程序最后加了while死循環(huán)
    發(fā)表于 03-20 08:42

    請(qǐng)問單片機(jī)main函數(shù)是執(zhí)行一次還是會(huì)反復(fù)執(zhí)行?

    請(qǐng)問單片機(jī)main函數(shù)是執(zhí)行一次,還是會(huì)反復(fù)執(zhí)行?
    發(fā)表于 10-19 06:25

    請(qǐng)問單片機(jī)main函數(shù)是執(zhí)行一次還是會(huì)反復(fù)執(zhí)行?

    motor = 0;//電機(jī)停 } 按程序邏輯,電機(jī)應(yīng)該是運(yùn)行一會(huì)就停止的,但是下載到板子之后,電機(jī)一直運(yùn)行。 在程序最后加了while死循環(huán)之后
    發(fā)表于 10-23 06:20

    AVR單片機(jī)的延時(shí)函數(shù)程序免費(fèi)下載

    本文檔的主要內(nèi)容詳細(xì)介紹的是AVR單片機(jī)的延時(shí)函數(shù)程序免費(fèi)下載
    發(fā)表于 11-16 08:00 ?7次下載
    AVR<b class='flag-5'>單片機(jī)</b>的延時(shí)<b class='flag-5'>函數(shù)</b><b class='flag-5'>程序</b>免費(fèi)下載

    單片機(jī)學(xué)習(xí)教程之C程序的簡(jiǎn)單介紹

    包含頭文件:現(xiàn)學(xué)習(xí)的是C51 單片機(jī),所以需要的頭文件是#include《reg51.h》程序的主函數(shù), main 函數(shù)
    發(fā)表于 11-19 17:24 ?12次下載

    深度剖析C語言的main函數(shù)

    main的返回值 main函數(shù)的返回值用于說明程序退出狀態(tài)。如果返回0,則代表程序正常
    的頭像 發(fā)表于 12-18 17:07 ?4352次閱讀

    STM32程序無法進(jìn)入main函數(shù)的解決方法

    很多人在基于STM32單片機(jī)項(xiàng)目開發(fā)過程中,會(huì)遇到STM32程序無法進(jìn)入main的現(xiàn)象,在這篇文將分享STM32程序無法進(jìn)入
    的頭像 發(fā)表于 07-22 16:18 ?1.5w次閱讀

    單片機(jī) keil c語言字符比較函數(shù)的使用 使用strcmp();

    (const char *str1, const char *str2) 把 str1 所指向的字符串和 str2 所指向的字符串進(jìn)行比較。菜鳥的鏈接32單片機(jī)中的使用示例1.要在單片機(jī)中使用我們要先添加頭文件:string.h不然程序
    發(fā)表于 11-21 15:06 ?17次下載
    <b class='flag-5'>單片機(jī)</b> keil  c語言字符比較<b class='flag-5'>函數(shù)</b>的使用 使用strcmp();

    STM32系列單片機(jī)在進(jìn)入main函數(shù)前都在干些什么?

    在剛開始學(xué)習(xí)單片機(jī)的時(shí)候,一直以為程序啟動(dòng)后就直接進(jìn)入到了main函數(shù),但是隨著學(xué)習(xí)的深入才發(fā)現(xiàn),程序在進(jìn)入
    發(fā)表于 11-23 18:21 ?12次下載
    STM32系列<b class='flag-5'>單片機(jī)</b>在進(jìn)入<b class='flag-5'>main</b><b class='flag-5'>函數(shù)</b>前都在干些什么?

    單片機(jī) keil調(diào)試的時(shí)候進(jìn)入不了main函數(shù)

    單片機(jī) keil調(diào)試的時(shí)候進(jìn)入不了main函數(shù)
    發(fā)表于 12-03 20:51 ?6次下載
    <b class='flag-5'>單片機(jī)</b> keil調(diào)試的時(shí)候進(jìn)入不了<b class='flag-5'>main</b><b class='flag-5'>函數(shù)</b>

    C語言編程中main()函數(shù)退出之后程序去哪兒了

    對(duì)于嵌入式系統(tǒng),如果沒有運(yùn)行RTOS,那么程序開發(fā)中的主函數(shù)main()需要通過某種機(jī)制使其永遠(yuǎn)愉快的運(yùn)行下去,它沒有終點(diǎn)。如果想從main函數(shù)
    的頭像 發(fā)表于 03-22 17:44 ?2669次閱讀

    C語言編程中main函數(shù)退出程序去哪兒了?

    PIC 單片機(jī)語言程序進(jìn)行跟蹤,發(fā)現(xiàn)main() 函數(shù)最后一條語句為 reset,也就是單片機(jī)直接復(fù)位,這是 MAPLAB編譯器根據(jù) PIC
    發(fā)表于 05-31 15:28 ?398次閱讀
    C語言編程中<b class='flag-5'>main</b><b class='flag-5'>函數(shù)</b><b class='flag-5'>退出</b>后<b class='flag-5'>程序</b>去哪兒了?

    main函數(shù)不一定就是程序入口

    寫個(gè)測(cè)試代碼,代碼中有main函數(shù),也有test函數(shù),test就是剛才我們說的入口,不過得指定退出方式,要不然程序運(yùn)行的時(shí)候會(huì)出問題。
    的頭像 發(fā)表于 06-15 17:12 ?611次閱讀

    單片機(jī)main函數(shù)結(jié)束干嘛去了?

    單片機(jī)main函數(shù)結(jié)束干嘛去了?
    的頭像 發(fā)表于 10-18 17:37 ?657次閱讀
    <b class='flag-5'>單片機(jī)</b><b class='flag-5'>main</b><b class='flag-5'>函數(shù)</b>結(jié)束干嘛去了?