概述:
在C語言中,使用printf()進行格式化輸出非常友善,例如,printf("%d\n",a)可以将a的值以十進制的格式輸出,然後換行。printff()函數的原型為:int printf(const char *format,...),在函數參數中的...表示可變參數,即輸入參數的個數不确定(例如,printf("%d\n",a)和printf("%d%d\n",a,b)都可以使函數進行正确的輸出),這種輸入參數不确定的函數就叫可變參數。在UART中能不能自己寫一個類似于 printf() 的函數(例如,Uart0_Printf("%d",a))呢?下面咱們就試圖找到實作這種函數的方法。
實驗内容:
本實驗要實作的功能就是利用uart實作發送可變參數的功能。
實驗代碼結構如下圖:
實驗一共包含3個檔案:main.c uart.c uart.h
main.c檔案内容:
#include<s3c2440.h>
#include"uart.h"
int main()
{
unsigned int a=10; //系統時鐘初始化,FCLK=400MHz,HCLK=100MHz,PCLK=50MHz
Uart0_Init(115200); //初始化并設定波特率為115 200
while(1)
{
Uart0_Printf("Uart0_Printf test output is:%d\n",a);
}
}
uart.c檔案内容:
#include<s3c2440.h>
#include<stdarg.h>
#include"uart.h"
#define PCLK 50000000
#define UART_BRD (int)((PCLK/(baudrate*16))-1)
void Uart0_Init(unsigned int baudrate)
{
GPHCON&=~((3<<4)|(3<<6)); //GPH2--TXD0;GPH3--RXD0
GPHCON|=((2<<4)|(2<<6)); //設定GPH2、GPH3為TXD0、RXD0功能
GPHUP=0x00; //上拉電阻使能
ULCON0|=0x03; //設定資料發送格式:8個資料位,1個停止位,無校驗位
UCON0=0x05; //發送模式和接收模式都使用查詢模式
UBRDIV0=UART_BRD; //設定波特率,其中波特率作為一個參數傳遞到該初始化函數
URXH0=0; //将URXH0清零
}
void putc(unsigned char c)
{
UTXH0=c;
while(!(UTRSTAT0&(1<<2))); //等待發送完成
}
unsigned char getc(void)
{
unsigned char c;
while(!(UTRSTAT0&(1<<0))); //查詢是否接收到有效資料
c=URXH0;
return c;
}
static void Uart0_SendByte(int data)
{
if(data=='\n') //注意,在超級終端中使用的換行符是'\r',是以當遇到'\n'時
{ //需要将其轉換為'r'
while(!(UTRSTAT0&(1<<2))); //等待發送完成
UTXH0='\r';
}
while(!(UTRSTAT0&(1<<2))); //等待發送完成完成後,将新發送
UTXH0=data; //的資料寫入發送寄存器
}
static void Uart0_SendString(char *pt)
{
while(*pt)
{
Uart0_SendByte(*pt++);
}
}
void Uart0_Printf(const char *fmt,...)
{
va_list ap; //定義了一個指向可變參數清單指針
char string[50]; //存儲要發送的内容
va_start(ap,fmt); //是參數清單指針ap指向函數參數清單中的第一個可變參數
vsprintf(string,fmt,ap);
va_end(ap); //清空參數清單
Uart0_SendString(string); //将該緩沖區中的資料列印到序列槽中
}
下面重點講解一下Uart0_Printf(const char *fmt,...)
void Uart0_Printf(const char *fmt,...)
{
va_list ap; //定義了一個指向可變參數清單指針
char string[50]; //存儲要發送的内容
va_start(ap,fmt); //是參數清單指針ap指向函數參數清單中的第一個可變參數
vsprintf(string,fmt,ap);
va_end(ap); //清空參數清單
Uart0_SendString(string); //将該緩沖區中的資料列印到序列槽中
}
了解這個函數需要了解下面的基礎知識:
可變參數的清單分為兩部分:固定參數和個數可變的可變參數。函數中至少有一個固定參數;可變參數由于個數不确定,聲明用"..."表示。
● va_list ap:定義了一個指向可變參數清單指針。
● va_start(ap,argN):使參數清單指針ap指向函數清單中的第一個可變參數,argN是最後一個固定參數。例如,當函數的聲明是void va_test(char a,char b,...),則它的固定參數依次是a,b,最後一個固定參數argN為c,是以就是a_start(ap,c)。
● va_end(ap):清空參數清單,并置參數指針ap無效,該宏的作用是結束可變參數的擷取。
● vsprintf()函數原型為int vsprintf(char *string,char *format,va_list param),其作用是将param按格式format寫入字元串string中。
是以,上述函數的基本流程是:
● 先開辟一塊區域存儲可變參數
● 然後,調用vsprintf()函數将可變參數按照指定的格式複制到緩沖區中。
● 最後,調用Uart0_SendString()函數将該緩沖區中的資料列印到序列槽中。
将以上3個檔案,複制完後,再編譯調試後,可以看到序列槽的效果:
我将該工程文檔上傳到:可自行下載下傳點選打開連結
注意:老生長談
①将S3C2440.s中的CLOCK_SETUP EQU 0 修改為 CLOCK_SETUP EQU 1
②複制并修改Ext_RAM檔案。