新聞中心

EEPW首頁 > 消費電子 > 設(shè)計應(yīng)用 > 一篇看懂嵌入式Linux

一篇看懂嵌入式Linux

作者: 時間:2018-07-30 來源:網(wǎng)絡(luò) 收藏

跟桌面一樣,是一個操作系統(tǒng)。從走過來的童鞋往往習慣于直接控制寄存器,事必躬親,從零開始實現(xiàn)想要的功能。而在的世界里,我們首先要拋棄這個思想,應(yīng)把它作為最后沒辦法的辦法。

本文引用地址:http://www.butianyuan.cn/article/201807/384397.htm

就像我們想要在windows系統(tǒng)中編寫一個程序,首先想到的不是操作CPU芯片的寄存器,而是學習Windows API一樣。我們在linux編程時,首先想到的應(yīng)該是使用現(xiàn)成的驅(qū)動或軟件或Linux API。沒有的話看看能不能修改一下現(xiàn)成的資源為己所用。還是不行的話才考慮自己從頭開始寫。

嵌入式Linux大廈是由很多層組成的,當我們想找一個人時,首先要明確他在那一層樓。同樣的,我們遇到問題時,首先要知道是哪個方面的問題,然后才有可能知道到哪里尋找答案。下面我們把這座大廈進行一下拆解。

我們平時使用Linux系統(tǒng)的話,最常用的工具就是Shell(或者用windows中常見的說法:命令行),初學者接觸Linux的第一個東西往往也是shell。也許你已經(jīng)知道,把shell命令組合起來寫成一個文件,亦即shell編程,也是一門大學問,它能做的事很多很強大,但僅限于對Linux系統(tǒng)的操作。

我們一定不會用shell命令去編寫一個顯示屏程序,或者一個GPS導航程序。而且作為嵌入式Linux開發(fā)來說,shell不可能作為最終產(chǎn)品工作的平臺,因為我們不能要求用戶在屏幕中輸入代碼來實現(xiàn)功能。因此我認為對嵌入式開發(fā)來說,shell命令無需深究,掌握基本操作就夠了。

shell基本操作主要包括:獲取命令幫助,到達指定目錄,查看目錄內(nèi)容,權(quán)限修改,文件的復制粘貼等基本操作,文件搜索,文件內(nèi)容查看和編輯,系統(tǒng)關(guān)機重啟……(這些只是最基本的,后面再慢慢學別的命令,比如學習進程編程時,再學習進程相關(guān)的命令;學習C語言編程時,再學各種編譯和調(diào)試命令也不遲)

學習嵌入式Linux,我們的最終目的是制作一套嵌入式系統(tǒng)來實現(xiàn)功能。往往需要用C/C++或Python等其他語言來編寫程序,但是編程之前我們要先明確一些基本概念。

最基本的,當我們編寫程序時,首先要明確嵌入式Linux分為用戶空間和內(nèi)核空間。用戶空間是應(yīng)用程序運行的空間,內(nèi)核空間就是操作系統(tǒng)和驅(qū)動程序運行的空間。這是從軟件的角度來說的,對應(yīng)于ARM芯片來說,就是芯片的不同“工作模式”。這兩個空間是通過“地理隔離”實現(xiàn)互相完全獨立的,它們各自的程序使用不同的內(nèi)存地址區(qū)間,各自使用自己的頭文件(有些頭文件在兩個空間內(nèi)甚至是重名的,要注意區(qū)分)、各自調(diào)用屬于自己空間的函數(shù)(哪怕實現(xiàn)的功能相同,比如printf()和printk()),而且不能互相直接訪問(用指針也不行)。(意味著學習這兩部分的編程時要學習兩套獨立的知識體系)

內(nèi)核空間相關(guān)的東西有:Linux內(nèi)核源碼、內(nèi)核編譯和配置、內(nèi)核移植、文件系統(tǒng)、Busybox、設(shè)備驅(qū)動程序編寫、中斷編程……

用戶空間相關(guān)的東西有:Shell、應(yīng)用程序編譯和調(diào)試、進程、線程、文件IO編程、網(wǎng)絡(luò)通信相關(guān)、Qt圖形界面編程……

如果你僅僅要開發(fā)應(yīng)用程序,那你就可以遠離內(nèi)核空間那些東西了。對你來說,驅(qū)動程序、底層硬件、操作系統(tǒng)的工作方式等都是透明的,你寫的程序在別的芯片上也能跑得很好。

但如果你想要開發(fā)驅(qū)動程序,或者定制自己的操作系統(tǒng),或者你想向一片“全裸”芯片中寫入操作系統(tǒng),并使它正常運行起來,那就得學習內(nèi)核空間的知識了。

如果你想讓“全裸”芯片運行起來,還會遇到一塊比內(nèi)核更底層的東西,Bootloader。它是在內(nèi)核啟動前運行的一段程序,用來初始化硬件、建立內(nèi)存空間映射等,與芯片的品牌、型號極其相關(guān)。我們通常對一些現(xiàn)成的Bootloader進行修改來滿足需求,常見的Bootloader有U-Boot、Vivi等。

再多說一句,如果想從零開始做一個嵌入式設(shè)備,還有更底層的問題需要解決和學習:電路設(shè)計、PCB布線等。

因此,我們看到的嵌入式Linux書籍就可以粗略分成兩個方向:一類講嵌入式Linux應(yīng)用程序編程,另一類講如何搭建一個完整的嵌入式Linux平臺。分別對應(yīng)的就是用戶空間和內(nèi)核空間的事情。

雖然用戶空間和內(nèi)核空間是獨立的,但就像Windows提供了API,允許我們對系統(tǒng)進行操作一樣,用戶空間的程序也可以通過系統(tǒng)調(diào)用來訪問內(nèi)核(就是一些的C語言函數(shù))。但由于系統(tǒng)調(diào)用非?;A(chǔ),所以有時使用起來很麻煩。比如說一個簡單的給變量分配內(nèi)存空間的操作,就需要動用多個系統(tǒng)調(diào)用。Linux定義一些庫函數(shù)(API)來將系統(tǒng)調(diào)用組合成某些常用的功能,以方便我們編程(同樣是C語言函數(shù))。因此,我們在讀別人的程序時,就要區(qū)分其中的函數(shù)是系統(tǒng)調(diào)用,還是庫函數(shù),還是C/C++標準庫中的函數(shù),還是用戶自己定義的函數(shù)。如果是前三者,就可以到各個地方搜索相應(yīng)的資料,這樣學習起來就快很多。

那么shell程序和我們用C/C++編寫的程序有什么區(qū)別呢?事實上,我們在shell中寫的每一個命令,都對應(yīng)了一個程序,在程序內(nèi)部就是通過調(diào)用各種API來實現(xiàn)相應(yīng)功能的。因此用shell能實現(xiàn)的功能,理論上都能用C語言實現(xiàn)。

作為嵌入式Linux開發(fā)初學者,簡單熟悉了shell以后,就可以開始進行一些C語言編程的嘗試了。

我們最早接觸編程一般都是在大學的編程課上,而且往往用的是Visual C++ 6.0。竊以為這是讓我對編程原理長期困惑不解的罪魁禍首!啥是環(huán)境變量?為啥要設(shè)置include路徑,lib路徑?為啥一點編譯按鈕就會出來那么多后綴名不同的文件?這些很基礎(chǔ)很重要的問題都被VC6.0這個外殼掩蓋了。但哪怕你在Linux中使用gcc編譯一個最簡單程序,一定就會像我一樣馬上明白把一個.c的源文件變成一個可執(zhí)行文件,中間究竟發(fā)生了什么事情。如果你再用gdb調(diào)試一個程序,就會明白得更多一點。

關(guān)于C/C++編程的基本工具,我們需要學習的有:vim等代碼編輯器、diff等文件比較的shell命令、gcc等編譯器、gdb等調(diào)試工具、交叉編譯等。這里需要特別提到一個重要工具(網(wǎng)站):github,根據(jù)百度的解釋,它是一個“分布式的版本控制系統(tǒng)”,初學者還用不到版本控制,那就可以單純把它當成一個開放的源代碼庫。這個網(wǎng)站里有大量優(yōu)秀的源代碼供學習和使用。

學習了基本的編程方法,我們就該接觸Linux的API等內(nèi)容了。畢竟,我們的嵌入式系統(tǒng)要與設(shè)備進行交互,只用C/C++標準庫是不夠的。在此之前,需要建立一個Linux的重要概念:一切皆文件。甚至硬件設(shè)備對Linux系統(tǒng)來說,也是文件。這樣對設(shè)備的操作就等同于對文件進行讀、寫,或讀寫以外的操作。這部分內(nèi)容在各種書籍資料中通常以“文件IO編程”命名,作為一個章節(jié)來寫。我覺得這是應(yīng)當?shù)谝粋€來學的東西,因為看到自己能隨意操控文件和外設(shè)是一件讓人很振奮的事情!成就感是繼續(xù)學習的一大動力!

另外一個重要內(nèi)容是,理解進程和線程。通過學習這個部分,能管中窺豹地大致領(lǐng)略到Linux系統(tǒng)如何進行調(diào)度,你的程序是怎么在Linux中運行的。這是操作系統(tǒng)原理的內(nèi)容,但作為非軟件專業(yè)出身的人,沒辦法,只能自學了。

其他應(yīng)用程序編程如網(wǎng)絡(luò)編程、Qt圖形編程等就不一一說明了。

驅(qū)動程序可能是我們將來接觸內(nèi)核空間遇到的第一個內(nèi)容。不過暫時還沒什么特別想說的。內(nèi)核空間距離初學者還是有點遠的……以后再來學這部分內(nèi)容。



關(guān)鍵詞: 嵌入式 單片機 Linux

評論


相關(guān)推薦

技術(shù)專區(qū)

關(guān)閉