本文為RISC-V嵌入式開發(fā)準備篇2:嵌入式開發(fā)的特點介紹。
本文的目的是對嵌入式開發(fā)的特點進行簡單的科普與回顧,為后續(xù)詳細介紹“RISC-V GCC工具鏈”和“RISC-V匯編語言程序設計”打下基礎。注:本文力求通俗易懂,主要面向初學者,對嵌入式開發(fā)有所了解的讀者可以忽略此文。
在本號上次發(fā)表的文章《編譯過程簡介》中介紹過,嵌入式系統(tǒng)的程序編譯過程和開發(fā)有其特殊性,譬如:
嵌入式系統(tǒng)需要使用交叉編譯與遠程調試的方法進行開發(fā)。
需要自己定義引導程序。
需要注意減少代碼體積(Code Size)。
需要移植printf從而使得嵌入式系統(tǒng)也能夠打印輸入。
使用Newlib作為C運行庫。
每個特定的嵌入式系統(tǒng)都需要配套的板級支持包。
下文將分別予以介紹。
1 交叉編譯和遠程調試
在本號上次發(fā)表的文章《編譯過程簡介》中介紹了如何在Linux系統(tǒng)的PC電腦上開發(fā)一個Hello World程序,對其進行編譯,然后運行在此電腦上。在這種方式下,我們使用PC電腦上的編譯器編譯出該PC電腦本身可執(zhí)行的程序,這種編譯方式稱之為本地編譯。
嵌入式平臺上往往資源有限,嵌入式系統(tǒng)(譬如常見ARM MCU或8051單片機)的存儲器容量通常只在幾KB到幾MB之間,且只有閃存而沒有硬盤這種大容量存儲設備,在這種資源有限的環(huán)境中,不可能將編譯器等開發(fā)工具安裝在嵌入式設備中,所以無法直接在嵌入式設備中進行軟件開發(fā)。因此,嵌入式平臺的軟件一般在主機PC上進行開發(fā)和編譯,然后將編譯好的二進制代碼下載至目標嵌入式系統(tǒng)平臺上運行,這種編譯方式屬于交叉編譯。
交叉編譯可以簡單理解為,在當前編譯平臺下,編譯出來的程序能運行在體系結構不同的另一種目標平臺上,但是編譯平臺本身卻不能運行該程序,譬如,在x86平臺的PC電腦上編寫程序并編譯成能運行在ARM平臺的程序,編譯得到的程序在x86平臺上不能運行,必須放到ARM平臺上才能運行。
與交叉編譯同理,在嵌入式平臺上往往也無法運行完整的調試器,因此當運行于嵌入式平臺上的程序出現(xiàn)問題時,需要借助主機PC平臺上的調試器來對嵌入式平臺進行調試。這種調試方式屬于遠程調試。
常見的交叉編譯和遠程調試工具是GCC和GDB。在本號上次發(fā)表的文章《編譯過程簡介》中介紹了如何使用Linux自帶的GCC本地編譯一個Hello World程序并運行。但是,GCC不僅能作為本地編譯器,還能作為交叉編譯器;同理GDB不僅可以作為本地調試器,還可以作為遠程調試器。
當作為交叉編譯器之時,GCC通常有不同的命名,譬如:
arm-none-eabi-gcc和arm-none-eabi-gdb是面向裸機(Bare-Metal)ARM平臺的交叉編譯器和遠程調試器。
所謂裸機(Bare-Metal)是嵌入式領域的一個常見形態(tài),表示不運行操作系統(tǒng)的系統(tǒng)
而riscv-none-embed-gcc和riscv-none-embed-gdb是面向裸機RISC-V平臺的交叉編譯器和遠程調試器。
本號后續(xù)發(fā)文《RISC-V GCC工具鏈的介紹》將介紹RISC-V GCC工具鏈的更多信息。
2 移植newlib或newlib-nano作為C運行庫
newlib是一個面向嵌入式系統(tǒng)的C運行庫。相對于本號上次發(fā)表的文章《編譯過程簡介》中介紹的glibc,newlib實現(xiàn)了大部分的功能函數(shù),但體積卻小很多。newlib獨特的體系結構將功能實現(xiàn)與具體的操作系統(tǒng)分層,使之能夠很好地進行配置以滿足嵌入式系統(tǒng)的要求。由于專為嵌入式系統(tǒng)設計,newlib具有可移植性強、輕量級、速度快、功能完備等特點,已廣泛應用于各種嵌入式系統(tǒng)中。
由于嵌入式操作系統(tǒng)和底層硬件的多樣性,為了能夠將C/C++語言所需要的庫函數(shù)實現(xiàn)與具體的操作系統(tǒng)和底層硬件進行分層,newlib的所有庫函數(shù)都建立在20個樁函數(shù)的基礎上,這20個樁函數(shù)完成具體操作系統(tǒng)和底層硬件相關的功能:
I/O和文件系統(tǒng)訪問(open、close、read、write、lseek、stat、fstat、fcntl、link、unlink、rename);
擴大內存堆的需求(sbrk);
獲得當前系統(tǒng)的日期和時間(gettimeofday、times);
各種類型的任務管理函數(shù)(execve、fork、getpid、kill、wait、_exit);
這20個樁函數(shù)在語義、語法上與POSIX(Portable Operating System Interface of UNIX)標準下對應的20個同名系統(tǒng)調用完全兼容。
所以,如果需要移植newlib至某個目標嵌入式平臺,成功移植的關鍵是在目標平臺下找到能夠與newlib樁函數(shù)銜接的功能函數(shù)或者實現(xiàn)這些樁函數(shù)。本號后續(xù)發(fā)文《基于HBird-E-SDK平臺的軟件開發(fā)與運行》將介紹蜂鳥E200的HBird-E-SDK平臺如何實現(xiàn)移植實現(xiàn)newlib的樁函數(shù)。
注意:newlib的一個特殊版本newlib-nano版本進一步為嵌入式平臺減少了代碼體積(Code Size),因為newlib-nano提供了更加精簡版本的malloc和printf函數(shù)的實現(xiàn),并且對庫函數(shù)使用GCC的-Os(側重代碼體積的優(yōu)化)選項進行編譯優(yōu)化。
3 嵌入式引導程序和中斷異常處理
在本號上次發(fā)表的文章《編譯過程簡介》中介紹了如何在Linux系統(tǒng)的PC電腦上開發(fā)一個Hello World程序,對其進行編譯,然后運行在此電腦上。在這種方式下,程序員僅僅只需要關注Hello World程序本身,程序的主體由main函數(shù)組織而成,程序員可以無需關注Linux操作系統(tǒng)在運行該程序的main函數(shù)之前和之后需要做什么。事實上,在Linux操作系統(tǒng)中運行應用程序(譬如簡單的Hello World)時,操作系統(tǒng)需要動態(tài)地創(chuàng)建一個進程、為其分配內存空間、創(chuàng)建并運行該進程的引導程序,然后才會開始執(zhí)行該程序的main函數(shù),待其運行結束之后,操作系統(tǒng)還要清除并釋放其內存空間、注銷該進程等。
從上述過程中可以看出,程序的引導和清除這些“臟活累活”都是由Linux這樣的操作系統(tǒng)來負責進行。但是在嵌入式系統(tǒng)中,程序員除了開發(fā)以main函數(shù)為主體的功能程序之外,還需要關注如下兩個方面:
引導程序:
嵌入式系統(tǒng)上電后需要對系統(tǒng)硬件和軟件運行環(huán)境進行初始化,這些工作往往由用匯編語言編寫的引導程序完成。
引導程序是嵌入式系統(tǒng)上電后運行的第一段軟件代碼。引導程序對于嵌入式系統(tǒng)非常關鍵,引導程序所執(zhí)行的操作依賴于所開發(fā)的嵌入式系統(tǒng)的軟硬件特性,一般流程包括:初始化硬件、設置異常和中斷向量表、把程序拷貝到片上SRAM中、完成代碼的重映射等,最后跳轉到main函數(shù)入口。
本號后續(xù)發(fā)文《基于HBird-E-SDK平臺的軟件開發(fā)與運行》將結合HBird-E-SDK平臺的引導程序實例了解引導程序的更多細節(jié)。
中斷異常處理
中斷和異常是嵌入式系統(tǒng)非常重要的一個環(huán)節(jié),因此,嵌入式系統(tǒng)軟件還必須正確地配置中斷和異常處理函數(shù)。有關RISC-V架構的中斷和異常的詳細信息,請參見RISC-V中文書籍《手把手教你設計CPU——RISC-V處理器篇》 中第13章內容《不得不說的故事——中斷和異?!贰?/p>
本號后續(xù)發(fā)文《基于HBird-E-SDK平臺的軟件開發(fā)與運行》將結合HBird-E-SDK程序實例了解如何配置中斷和異常處理函數(shù)。
4 嵌入式系統(tǒng)鏈接腳本
在本號上次發(fā)表的文章《編譯過程簡介》中介紹了如何在Linux系統(tǒng)的PC電腦上開發(fā)一個Hello World程序,對其進行編譯,然后運行在此電腦上。在這種方式下,程序員也無需關心編譯過程中的“鏈接”這一步驟所使用的鏈接腳本,無需為程序分配具體的內存空間。
但是在嵌入式系統(tǒng)中,程序員除了開發(fā)以main函數(shù)為主體的功能程序之外,還需要關注“鏈接腳本”為程序分配合適的存儲器空間,譬如程序段放在什么區(qū)間、數(shù)據(jù)段放在什么區(qū)間等等。
本號后續(xù)發(fā)文《基于HBird-E-SDK平臺的軟件開發(fā)與運行》將結合HBird-E-SDK的“鏈接腳本”實例了解更多細節(jié)。
5 減少代碼體積
嵌入式平臺上往往存儲器資源有限,嵌入式系統(tǒng)(譬如常見的ARM MCU或8051單片機)的存儲器容量通常只在幾KB到幾MB之間,且只有閃存而沒有硬盤這種大容量存儲設備,在這種資源有限的環(huán)境中,程序的代碼體積(Code Size)顯得尤其重要,因此,有效地降低降低代碼體積(Code Size)是嵌入式軟件開發(fā)必須要考慮的問題,常見的方法如:
使用newlib-nano作為C運行庫以取得較小代碼體積(Code Size)的C庫函數(shù)。
盡量少使用C語言的大型庫函數(shù),譬如在正式發(fā)行版本的程序中避免使用printf和scanf等函數(shù)。
如果在開發(fā)的過程中一定需要使用printf函數(shù),可以使用某些自己實現(xiàn)的閹割版printf函數(shù)(而不是C運行庫中提供的printf函數(shù))以生成較小的代碼體積。
除此之外,在C/C++語言的語法和程序開發(fā)方面也有眾多技巧以取得更小的代碼體積(Code Size)。
本號后續(xù)發(fā)文《基于HBird-E-SDK平臺的軟件開發(fā)與運行》將結合HBird-E-SDK平臺實例了解更多“減少代碼體積”的實現(xiàn)細節(jié)。減小代碼體積(Code Size)的方法很多,本文在此不做一一贅述,請初學的讀者自行查閱相關資料進行學習。
6 支持printf函數(shù)
在本號上次發(fā)表的文章《編譯過程簡介 》中介紹了如何在Linux系統(tǒng)的PC電腦上開發(fā)一個Hello World程序,程序中使用C語言的標準庫函數(shù)printf打印了一個“Hello World”字符串。該程序在Linux系統(tǒng)里面運行的時候字符串被成功的輸出到了Linux的終端界面上。在這個過程中,程序員無需關心Linux系統(tǒng)到底是如何將printf函數(shù)的字符串輸出到Linux終端上的。事實上,如《編譯過程簡介》中所述,在Linux本地編譯的程序會鏈接使用Linux系統(tǒng)的C運行庫glibc,而glibc充當了應用程序和Linux操作系統(tǒng)之間的接口,glibc提供的 printf 函數(shù)就會調用如sys_write等操作系統(tǒng)的底層系統(tǒng)調用函數(shù),從而能夠將“字符串”輸出到Linux終端上。
從上述過程中可以看出,由于有glibc的支持,所以printf函數(shù)能夠在Linux系統(tǒng)中正確的進行輸出。但是在嵌入式系統(tǒng)中,printf的輸出卻不那么容易了,基于如下幾個原因:
嵌入式系統(tǒng)使用newlib作為C運行庫,而newlib的C運行庫所提供的printf函數(shù)最終依賴于如本文中所介紹的newlib樁函數(shù)write,因此必須實現(xiàn)此write函數(shù)才能夠正確的執(zhí)行printf函數(shù)。
嵌入式系統(tǒng)往往沒有“顯示終端”存在,譬如常見的單片機其作為一個黑盒子一般的芯片,根本沒有顯示終端。因此,為了能夠支持顯示輸出,通常需要借助單片機芯片的UART接口將printf函數(shù)的輸出重新定向到主機PC的COM口上,然后借助主機PC的串口調試助手顯示出輸出信息。同理,對于scanf輸入函數(shù),也需要通過主機PC的串口調試助手獲取輸入然后通過主機PC的COM口發(fā)送給單片機芯片的UART接口。
從以上兩點可以看出,嵌入式平臺的UART接口非常重要,往往扮演了輸出管道的角色,為了能夠將printf函數(shù)的輸出定向到UART接口,需要實現(xiàn)newlib的樁函數(shù)write,使其通過編程UART的相關寄存器將字符通過UART接口輸出。本號后續(xù)發(fā)文《基于HBird-E-SDK平臺的軟件開發(fā)與運行》將結合HBird-E-SDK平臺移植printf函數(shù)的實例了解更多細節(jié)。
7 提供板級支持包
對于特定的嵌入式硬件平臺,為了方便用戶在硬件平臺上開發(fā)嵌入式程序,硬件平臺一般會提供板級支持包(Board Support Package,BSP)。板級支持包所包含的內容沒有絕對的標準,通常說來,其必須包含如下內容:
底層硬件設備的地址分配信息
底層硬件設備的驅動函數(shù)
系統(tǒng)的引導程序
中斷和異常處理服務程序
系統(tǒng)的鏈接腳本
如果使用newlib作為C運行庫,一般還提供newlib樁函數(shù)的實現(xiàn)。
由于板級支持包往往會將很多底層的基礎設施和移植工作搭建好,因此應用程序開發(fā)人員通常都無需關心本文第1.2節(jié)至第1.6節(jié)中描述的內容,能夠從底層細節(jié)中被解放出來避免重復建設而出錯。本號后續(xù)發(fā)文《基于HBird-E-SDK平臺的軟件開發(fā)與運行》將結合HBird-E-SDK平臺的BSP實例了解更多細節(jié)。
-
嵌入式系統(tǒng)
+關注
關注
41文章
3624瀏覽量
129751 -
RISC-V
+關注
關注
45文章
2322瀏覽量
46592
原文標題:RISC-V嵌入式開發(fā)準備篇2:嵌入式開發(fā)的特點介紹
文章出處:【微信號:real_farmer,微信公眾號:硅農亞歷山大】歡迎添加關注!文章轉載請注明出處。
發(fā)布評論請先 登錄
相關推薦
評論