c標準式庫中 的printf & sscanf 是標準輸出入 指的是PC monitor & keyboard
所以在單晶片使用時,就要修改PRINTF & SCANF 的底層 ,以 keil 操作stm32為例
比較簡單的做法是修改fputc & fgetc 兩個函數,因為這兩個分別是printf & scanf的底層實作.
只要在main.c OR uart.c 加入以下兩個函數,就可使用printf & scanf
/*******************************************************************************
* Function Name : int fputc(int ch, FILE *f)
* Description :
* Input : int ch, FILE *f
* Output :
* Return : int ch
*******************************************************************************/
int fputc(int ch, FILE *f)
{
USART1->DR = (u8) ch;
/* Loop until the end of transmission */
while(USART_GetFlagStatus(USART1, USART_FLAG_TXE) == RESET)
{
}
return ch;
}
/*******************************************************************************
Function Name : fputc
Description : redefine getc function,use scanf with uart input
Input : FILE *f
Output :
* Return : int ch
********************************************************************************/
int fgetc(FILE *f)
{
while (USART_GetFlagStatus(USART1, USART_FLAG_RXNE) == RESET)
{}
return (int)USART_ReceiveData(USART1);
}
另外一種方法就是
2、添加Retarget.c,实现在KEIL下使用printf函数,以LPC2478为例
首先在Keil安装目录下面ARM/Startup/Retarget.c找到Retarget.c文件将其复制到你的工程文件夹下面;并将其加入到工程中
在uart.c中添加如下代码
// Implementation of sendchar (also used by printf function to output data)
int sendchar (int ch) { // Write character to Serial Port
while (!(U0LSR & 0x20));
return (U0THR = ch);
} int getkey (void) { // Read character from Serial Port
while (!(U0LSR & 0x01));
return (U0RBR);
}
int sendchar (int ch) { // Write character to Serial Port
while (!(U0LSR & 0x20));
return (U0THR = ch);
} int getkey (void) { // Read character from Serial Port
while (!(U0LSR & 0x01));
return (U0RBR);
}
这样,只要在需要用printf的文件里#include 就可以了,printf会通过Retarget中的fputc函数调用sendchar来实现串口数据的输出。
事实上,和第一种的方式是一样的。
3、自定义printf函数,以AVR为例
前面介绍的是在KEIL编译器上使用printf函数,但不是所有的编译器平台都能适用,因此有时候我们需要自定义printf函数,下面以AVR在GCC下为例
在usart.c中添加如下代码
#include <stdio.h>
#include <stdarg.h>
/*********************************************************/
//向串口usart0发送一个字节函数
void Uart0_putchar( unsigned char sdbyte)
{
UDR0=sdbyte;
while(!(UCSR0A&0x40));
UCSR0A|=0x40;
}////////////////////////////////////////////////////////
void Uart0_printf(char *str,...)
{
char buf[128];
unsigned char i = 0;
va_list ptr;
va_start(ptr,str);
vsprintf(buf,str,ptr);
while(buf[i])
{
Uart0_putchar(buf[i]);
i++;
}
}
#include <stdarg.h>
/*********************************************************/
//向串口usart0发送一个字节函数
void Uart0_putchar( unsigned char sdbyte)
{
UDR0=sdbyte;
while(!(UCSR0A&0x40));
UCSR0A|=0x40;
}////////////////////////////////////////////////////////
void Uart0_printf(char *str,...)
{
char buf[128];
unsigned char i = 0;
va_list ptr;
va_start(ptr,str);
vsprintf(buf,str,ptr);
while(buf[i])
{
Uart0_putchar(buf[i]);
i++;
}
}
有了printf格式化输出函数,调试起来就方便多了。
說個題外話,當你使用stm32的uart 中斷時.中斷函式請將USART_ClearITPendingBit(USART1,USART_IT_RXNE);
加在scanf的函式後面,不然的話第一次接收的第一個字元都會接收不到.
說個題外話,當你使用stm32的uart 中斷時.中斷函式請將USART_ClearITPendingBit(USART1,USART_IT_RXNE);
加在scanf的函式後面,不然的話第一次接收的第一個字元都會接收不到.
沒有留言:
張貼留言