2、printf()函数实现思路分析这部分涉及uboot中串口的初始化,参考博客: 《uboot中串口(控制台)初始化详解》;
3、函数调用关系(1)uboot中的printf()函数的用法和利用C库编写C语言程序时是一样的,本身uboot中printf()函数就是移植过来的;
(2)printf()函数涉及到C语言的可变参数,首先对传参里的%d、%c等做处理,得到要输出的字符串;
(3)调用输出字符串的函数puts();
(4)puts()内部实际又是调用的putc(),按一个字符一个字符的输出;
总结:printf()中字符解析部分的代码是通用的,只有putc()函数是依赖硬件,不同的硬件输出字符对应不同的putc()函数;
printf (const char *fmt, ...)
vsprintf(char *buf, const char *fmt, va_list args)
puts (const char *s)
fputs (int file, const char *s)
stdio_devices[file]->puts (s)
4、printf()函数源码
void printf (const char *fmt, ...)
{
va_list args;
uint i;
char printbuffer[CFG_PBSIZE];
va_start (args, fmt);
/*解析字符串*/
i = vsprintf (printbuffer, fmt, args);
va_end (args);
/* 输出解析好的字符串*/
puts (printbuffer);
}
void puts (const char *s)
{
if (gd->flags & GD_FLG_DEVINIT) {
/* Send to the standard output */
fputs (stdout, s);
} else {
/* Send directly to the handler */
serial_puts (s);
}
}
void fputs (int file, const char *s)
{
if (file < MAX_FILES)
stdio_devices[file]->puts (s);
}
5、实际串口发送数据的函数代码实际stdio_devices[file]->puts函数指向serial_puts(const char *s)函数,是在drv_system_init()函数填充的device_t结构体并注册到devlist链表中。
//串口枚举,S5PV210有4个串口
typedef enum {
S5PC11X_UART0,
S5PC11X_UART1,
S5PC11X_UART2,
S5PC11X_UART3,
} S5PC11X_UARTS_NR;
//返回对应串口的寄存器组基地址
static inline S5PC11X_UART * S5PC11X_GetBase_UART(S5PC11X_UARTS_NR nr)
{
return (S5PC11X_UART *)(ELFIN_UART_BASE + (nr*0x400));
}
//发送字符串
void serial_puts(const char *s)
{
/* 调用serial_putc()函数逐个字符发送 */
while (*s) {
serial_putc(*s++);
}
}
//发送单个字符
void serial_putc(const char c)
{
//得到串口的寄存器组基地址
S5PC11X_UART *const uart = S5PC11X_GetBase_UART(UART_NR);
//查询串口的状态是否是空闲态
/* wait for room in the tx FIFO */
while (!(uart->UTRSTAT & 0x2));
//发送一个字符
uart->UTXH = c;
/* 如果发送'\n',再发送'\r',这是处理Windows和linux中换行符的不同*/
if (c == '\n')
serial_putc('\r');
}
欢迎分享,转载请注明来源:内存溢出
评论列表(0条)