hebut971372

ARM中打印函数print 的几种实现方法

0
阅读(4140)

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等。只是没有对浮点的支持。