【精品博文】ARM中打印函数print 的几种实现方法

1 利用C库函数printf

步骤:

1) 首先需要包含头文件stdio.h。

2) 然后定义文件句柄。实际上就是一个int型变量封装在结构体中。

struct __FILE{int handle;};

3)定义FILE __stdout; FILE即为__FILE,通过stdio.h宏定义。

4) 实现函数

int fputc(int ch, FILE *f){ char tempch = ch; sendchar(tempch); return ch;}

5) 实现函数

int sendchar (int ch){

if (ch == '\n') {

while (!(console_tty_f->lsr & UART_LSR_THRE));

console_tty_f->dll_fifo = 0x0d;

}

while (!(console_tty_f->lsr & UART_LSR_THRE));

return (console_tty_f->dll_fifo = ch);

}

由以上代码可见,printf为阻塞函数,采用等待发完的办法,可能影响其它进程。如果编写非等待的打印函数,可以采用第二种方法。

2 利用C库函数vsprintf和变参函数

步骤:

1) 包含头文件stdio.h和stdarg.h。

2) 编写变参数函数。

void print(const char *lpszFormat, ...){char szBuffer[PRINT_BUF]={0};

va_list args; int ret;     va_start(args, lpszFormat);

ret = vsprintf(szBuffer, lpszFormat, args);

va_end(args);

uart[console_uart].put_2_ring(console_uart,szBuffer, ret);

}

由此可见,利用库函数vsprintf格式化输入字符串,然后在空闲时发送。

3 自行完成参数提取,格式化

步骤:

1) 定义可变参数列表typedef char *va_list;

2) 定义地址对齐宏

#define _AUPBND (sizeof (int) - 1)

#define  _ADNBND  (sizeof (int) - 1)

#define _bnd(X, bnd) (((sizeof (X)) + (bnd)) & (~(bnd)))

3) 定义可变长参数提取宏

#define va_start(ap, A) (void) ((ap) = (((char *) &(A)) + (_bnd (A,_AUPBND))))

#define va_arg(ap,T) (*(T *)(((ap) += (_bnd (T, _AUPBND))) - (_bnd (T,_ADNBND)))

#define va_end(ap) (void) 0

4)编写变参数函数。如方法2第2步。

5) 实现vsprintf函数。实现源码较多,如linux等。只是没有对浮点的支持。

(0)

相关推荐