终于解决了MDK 3.80a中不能使用printf()函数的问题(5楼semihosting(半主机模式)的个人

2019-12-22 13:45发布

刚开始学stm32,顺着gpio、uart。。。的顺序慢慢爬
初始化的方法学习了马老师的STM32_Init.h大法,自己英文还可以,加上avr的基础还不错,所以gpio和时钟配置都很顺利
碰到uart就头大了,看到各种例程里都是printf()函数,自己也想用,毕竟是avr想用却开销不了的东西。但是我自己写的程序里一旦出现printf,单片机的不干活了。查论坛首先发现要重定义fputc函数,照做了,还是不行。
后来怀疑是uart1初始化问题,用自己写的put_c函数却没问题。
后来又发现一种说法,需要避免使用semihosting(半主机模式),我也把代码加进去了(改fputc去掉了),还是不行。
再一想,重定义fputc是绝对必须的,加上了之后问题解决,成功使用printf("(敏感词0373) ");输出了,哈哈
***************************************************************************************************

以上废话,可以不看。
简单地说:想在mdk 3.80a中用printf,需要同时重定义fputc函数和避免使用semihosting(半主机模式),
论坛里应该有完整介绍这个的帖子,但是我没搜到,也许是沉了。重发出来希望能帮上像我这样的菜鸟们。

需要添加以下代码


#pragma import(__use_no_semihosting)
/******************************************************************************
*标准库需要的支持函数
******************************************************************************/
struct __FILE
{
int handle;
/* Whatever you require here. If the only file you are using is */
/* standard output using printf() for debugging, no file handling */
/* is required. */
};
/* FILE is typedef’ d in stdio.h. */
FILE __stdout;

/// <summary>
/// 定义_sys_exit()以避免使用半主机模式
/// </summary>
/// <param name="x"></param>
/// <returns></returns>
_sys_exit(int x)
{
x = x;
}



int fputc(int ch, FILE *f)
{
    //USART_SendData(USART1, (u8) ch);
    USART1->DR = (u8) ch;
   
    /* Loop until the end of transmission */
    while(USART_GetFlagStatus(USART1, USART_FLAG_TXE) == RESET)
    {
    }

    return ch;
}

以上代码来自goway 和 yugen 的发言,在此表示感谢



********************************************************************************************
更新:

根据四楼的帖子,发现在Options里选上microlib之后,就不用关闭半主机模式了。
友情提示: 此问题已得到解决,问题已经关闭,关闭后问题禁止继续编辑,回答。
该问题目前已经被作者或者管理员关闭, 无法添加新回复
39条回答
first_blood
1楼-- · 2019-12-25 18:31
回复【楼主位】ilawp
-----------------------------------------------------------------------

楼主好啊,这篇帖子很有启发。
不知是不是MDK版本升级的问题,我精简成下述的版本printf()使用正常:
/*********************************/
struct __FILE
{
        int handle;
};

FILE __stdout;
      
int fputc(int ch, FILE *f)
{
        while ((USART1->SR & 0x40) == 0);
        USART1->DR = ch & 0x1FF;
        return ch;
}
/*********************************/

这是使用正常情况下能精简到的最小版本
用微库会出现让人难以理解的符号。。。


MDK (原文件名:MDK.jpg)
jingjvwawa
2楼-- · 2019-12-25 23:20
 精彩回答 2  元偷偷看……
zhaochuanyuAVR
3楼-- · 2019-12-26 03:58
mark,,,
jayzah
4楼-- · 2019-12-26 09:41
是打钩就解决了的,先Mark先,等到遇到解决不了再回头来看。
ken_ly
5楼-- · 2019-12-26 10:38
MDK sprintf使用问题
乱世奸雄
6楼-- · 2019-12-26 15:14
顶27楼,方法可行。

一周热门 更多>