精品博文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等。只是没有对浮点的支持。
相关文章
- 459元国产新科技尔英B760M主板酷睿I7-12700处理器降压幅度100%+
- 魔兽世界怀旧服:狂暴战整合WA,卡英勇WA
- 「图钉神机」放假了?那就配台电脑打游戏吧
- 测试员必备:Linux下安装JDK 1.8你必须知道的那些事
- abee AS Enclosure W1机箱评测:复刻经典,简约又好用
- NZXT KRAKEN X53 RGB一体式水冷散热器评测:内外俱备的寒冰武士
- 8G显存是原罪,还是光追来背锅?RTX5060金属大师对比评测
- 订单爆满工人回流!韩国造船巨头产能利用率同比大增
- 技嘉AORUS WATERFORCE X 360水冷评测:兼顾冷与静,呈现力与美
- 精品博文ARM中打印函数print 的几种实现方法