单片机
返回首页

单片机使用printf函数进行串口打印输出

2015-10-12 来源:eefocus

以前在使用keil编写51单片机程序的时候,经常使用printf函数打印输出一些关键的过程数据到电脑,方便监控程序的运行状况。最近使用IAR for MSP430 调试 MSP430G2553 程序的时候,发现了一些小问题,MSP430G2553单片机并没有按照我的预期输出数据到电脑。

带着疑惑我查看了keil的帮助文件里面的printf函数说明,原来printf函数最终是调用putchar函数来实现打印输出字符的。

putchar,该函数将制定的表达式的值所对应的字符输出到标准输出终端上。表达式可以是字符型或整型,它每次只能输出一个字符。我们来看keil标准函数库里的putchar函数的函数体。

根据说明提示,我在D:KeilC51LIB文件夹里面找到了putchar.c文件

#include

#define XON 0x11

#define XOFF 0x13

char putchar (char c) {

if (c == ' ') {

if (RI) {

if (SBUF == XOFF) {

do {

RI = 0;

while (!RI);

}

while (SBUF != XON);

RI = 0;

}

}

while (!TI);

TI = 0;

SBUF = 0x0d;

}

if (RI) {

if (SBUF == XOFF) {

do {

RI = 0;

while (!RI);

}

while (SBUF != XON);

RI = 0;

}

}

while (!TI);

TI = 0;

return (SBUF = c);

}

#if 0 // comment out versions below

char putchar (char c) {

if (c == ' ') {

while (!TI);

TI = 0;

SBUF = 0x0d;

}

while (!TI);

TI = 0;

return (SBUF = c);

}

char putchar (char c) {

while (!TI);

TI = 0;

return (SBUF = c);

}

#endif

由说明文件可以看出,我们可以改写这个底层的putchar函数来适应不同的硬件。keil里面的putchar函数是默认用串行口输出信息的,我们可以自由定义成另外的输出模块,比如自定义IO向1602液晶输出信息。

keil的printf函数大致搞明白了,回头再研究下IAR for MSP430,可惜,我没有能够查看该软件标准函数库里的printf.c 和 putchar.c,不过我觉得程序没有通过串口向电脑打印输出信息,是因为底层的putchar函数没有定义为通过MSP430G2553的UART进行输出,那如果我自己重定向一个putchar函数,覆盖掉标准函数库里面的putchar,是不是就能够输出了呢。于是编写putchar函数如下:

int putchar(int c)

{

if(c == ' ')

{

while(UCA0STAT & UCBUSY);

UCA0TXBUF = ' ';

}

while(UCA0STAT & UCBUSY);

UCA0TXBUF = c;

return c;

}

编译后,输出完全正确。大功告成。

将测试程序向大家展示一下,希望能帮到大家。

////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////

uart.c

////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////

#include

void UartInit()

{

BCSCTL1 = CALBC1_1MHZ; // Set DCO

DCOCTL = CALDCO_1MHZ;

BCSCTL2 &= ~(DIVS_3);

P1SEL = BIT1 + BIT2 ; // P1.1 = RXD, P1.2=TXD

P1SEL2 = BIT1 + BIT2; // P1.1 = RXD, P1.2=TXD

UCA0CTL1

= UCSSEL_2; // SMCLK

UCA0BR0 = 104; // 1MHz 9600

UCA0BR1 = 0; // 1MHz 9600

UCA0MCTL = UCBRS0; // Modulation UCBRSx = 1

UCA0CTL1 &= ~UCSWRST; // **Initialize USCI state machine**

UC0IE

= UCA0RXIE; // Enable RX int

}

int putchar(int c)

{

if(c == ' ')

{

while(UCA0STAT & UCBUSY);

UCA0TXBUF = ' ';

}

while(UCA0STAT & UCBUSY);

UCA0TXBUF = c;

return c;

}

////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////

main.c

////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////

#include

#include 'uart.h'

#include 'stdio.h'

void main()

{

float value = 123.123456789;

char *string='http://www.hao123.com';

WDTCTL = WDTPW + WDTHOLD;

UartInit();

printf('value = %f %s ',value,string);

while(1);

}

进入单片机查看更多内容>>
相关视频
  • RISC-V嵌入式系统开发

  • SOC系统级芯片设计实验

  • 云龙51单片机实训视频教程(王云,字幕版)

  • 2022 Digi-Key KOL 系列: 你见过1GHz主频的单片机吗?Teensy 4.1开发板介绍

  • TI 新一代 C2000™ 微控制器:全方位助力伺服及马达驱动应用

  • MSP430电容触摸技术 - 防水Demo演示

精选电路图
  • 光控音效发生器电路

  • 如何利用ESP8266制作一个简单的四轴飞行器

  • 优化电路板布局的简单方法

  • 分享一个电网倾角计电路

  • 使用NE555和磁簧开关的橱柜照明电路

  • 一种构建12V和230V双直流电源的简单方法

    相关电子头条文章