嵌入式系統上的簡易printf windstorm on 2008-12-30,17:02 1,097 views Comments (3) 此文針對嵌入式軟件的業余初學者,高手請自行路過打醬油。 嵌入式中,調試手段通常有兩種,一是遠程gdb,一是直接printf。如果是調試自己玩的小板子,用gdb有點大張旗鼓了,大多數情況下printf就可以搞定。不過printf的問題是stdiolib的size太大,稍微有點程序,加上幾個常用的庫,比如stdio和string,超過16k甚至32k(已經大于一些低端芯片的flash容量了)是很正常的事情,而且通常比較慢,程序越多,越麻煩。道理很簡單,標準C語言庫的規范中,Printf()必須處理大量的數據格式,包括字符串、字符、(各種長度的有符號和無符號)數字,以及浮點值。而且格式字符串還要包括用于更改文本對齊、基數、間距、字段寬度和精度的調節器和指示器。符合這個規范的代碼必然會是冗長和繁重的。一些嵌入式系統庫倒是提供了一些之針對整數的printf,但還是有問題,首先是還是太大,其次是你沒有自己的調整權限。 其實printf也就是IO的調用包裝而已,我們完全可以自己寫一個簡易版本的printf滿足自己的需要,并隨時根據需要裁剪。具體來說,printf在這里要起的作用就是將調試字符串從嵌入式目標空閑的串口壓出,并在運行于宿主工作站的終端模擬器上顯示結果。下面就簡單介紹一下,如何來自己寫一個簡易printf函數。 要寫printf,首先要知道什么是可變參數傳遞,我們來看看標準庫里面,是如何定義可變參數實現的: 關于可變參數的原理,網上有一些文章,總結來說,就是我們可以通過Intel80×86機器的對齊特性來獲得所有的參數,因為在Intel80×86機器上,每個變量的地址都要是sizeof(int)的倍數,這樣能提升CPU運行的效率。也就是說,所有參數的首地址都要是4的倍數,就算你是char型的,那浪費3個byte也要安排你占第四個坑。 好,由于C語言傳遞參數時是用push指令從右到左將參數逐個壓棧,因此我們通過棧指針跳4n格來訪問第n個參數,不要忘了,參數的地址都是字對齊的。這里,我們用#define _bnd(X, bnd) (((sizeof (X)) + (bnd)) &(~(bnd)))來計算類型為X的參數在棧中占據的字對齊后的字節數。bnd是sizeof (acpi_native_int) –1,acpi_native_unit在32位機的定義是: typedef u32 acpi_native_uint; 所以( ~(bnd))就是0xfffffffc 。 因此,_bnd(X,bnd) 宏在32位機下就是 ( (sizeof(X) + 3)&0xfffffffc ) 很明顯,其作用是–倘若sizeof(X)不是4的整數倍,將其變為4的整數倍。 va_start(ap,A) 負責初始化參數指針ap,將函數參數A右邊第一個參數的地址賦給ap,這個第一個參數通常就是printf里面的”%x%d%f%d”。 va_arg(ap,T) 可以獲得ap指向參數的值,并使ap指向下一個參數,T用來指明當前參數類型。 在這里,上述代碼還是麻煩,而且sizeof我們也不能直接用,所以我們不如干脆直接寫一個不那么麻煩而有針對性的可變參數操作定義: #define sizeof(x) ((char *)(&x+1) - (char *)(&x))有了這幾個定義,print函數就好寫了,為了節省空間,這個簡單的print()只支持“%s”,“%d”和”%c”格式的分類符,暫時不需要其他功能,比如格式對齊之類的,當然,可以根據自己的需要擴展這個函數。 int print( const char *fmt, ... )這里面有一些函數,uart_putc是串口驅動程序,給串口送東西的,uart_puts是簡單的多重putc包裝。uart_putints則需要做一些atoi的轉換,一個比較簡單但是有效的atoi程序宏定義如下: #define ATOI(X, result) \ |