历史上的今天
返回首页

历史上的今天

今天是:2025年08月06日(星期三)

正在发生

2019年08月06日 | 在STM32平台上实现printf和scanf函数(带回显)

2019-08-06 来源:eefocus

首先我们来看这两者之间的区别, Keil专用于嵌入式程序仿真编译书写的平台, 支持C99, 不支持标准输入输出, 不支持文件操作, 带有标准库和各类处理器的固件库. POJ专用于程序测试, 支持C99, 支持标准输入输出, 不支持文件操作, 自带C, C++标准库.

 

其实早在Keil开始发布的时候, 附带的ST件器例程里就有对printf函数的实现. 具体的工程如图所示:

 


 

软件仿真的运行结果:

 

但不管怎样, 都不能用于POJ中, 原因有以下几点:

 ①         在main函数中配置了系统的时钟

②         在main函数中配置了串口

③         在main.c文件中包含了很多POJ上不需要的函数

④         必须调用ST的固件库

 

也就是说, 如果能做到以上四点, 那么几乎就可以将Keil的代码用于POJ中了. 我们试着解释一下整个工程的工作流程:

系统上电→STM32F10X.S→main()→RCC_Configuration()→NVIC_Configuration()→GPIO_Configuration()→USART_Init()→USART_Cmd()→printf();

其中RRC_Configuration(), NVIC_Configuration(), GPIO_Configuration(), USART_Init(), USART_Cmd()等函数都是ST的固件库, 不能被POJ系统所识别.

    正如我们所见, 在执行printf函数之前, 先运行了一个汇编文件和一些系统配置专用的函数. 完全可以把配置函数的功能写到汇编文件STM32F10X.S里. 这样系统在进入main函数之前就已经配置好了. 这样在main.c文件里需要做的事情就是:

#include

 

Int main()

{

    Printf(“ this is a test!n”);

    Return (0);

}

这样的代码在POJ系统上是可以运行的. 同时ST例程里的这个函数也需要放到头文件stdio.h里.

/************************************************************************

* Function Name  : PUTCHAR_PROTOTYPE

* Description    : Retargets the C library printf function to the USART.

* Input          : None

* Output         : None

* Return         : None

************************************************************************/

PUTCHAR_PROTOTYPE

{

  /* Write a character to the USART */

  USART_SendData(USARTx, (u8) ch);

 

  /* Loop until the end of transmission */

  while(USART_GetFlagStatus(USARTx, USART_FLAG_TXE) == RESET)

  {

  }

 

  return ch;

}

 

下面是经过我修改的STM32F10X.S的核心代码:

Reset_Handler   PROC

                EXPORT  Reset_Handler             [WEAK]

                IMPORT  __main

        

        LDR    R1, =RCC    ;HSE_ON

        LDR    R0,=0x03030083

        STR    R0,[R1,#0x00]

 

LOOP1

        LDR    R0, =RCC    ;HSE_GOOD

        LDR    R0,[R0,#0x00]

        ANDS.W    R0,#0x20000

        BEQ    LOOP1

        

        LDR    R0, =FLASH    ;FLASH_PREFETCH_ON

        LDR    R0,[R0,#0x00]

        ORR    R0,#0x10

        LDR    R1, =FLASH

        STR    R0,[R1,#0x00]    ;SYS_CLK = 8MHz => FLASH_DEL=0

        

                    ;APB1_CLK = APB2_CLK = AHB_CLK = SYS_CLK

;        LDR    R0, =RCC

;        LDR    R1, =0x001D0405

;        STR    R1,[R0,#0x04]

        

        LDR    R1, =RCC    ;APB2_PERI_ALL_ON

        MOV    R0,#0xFFFF

        STR    R0,[R1,#0x18]

        

        LDR    R1, =GPIOA    ;(B<<4) + (4<<8) == 0x4B0

        MOV    R0,#0x4B0

        STR    R0,[R1,#0x04]    ;GPIOA.10:STIN, GPIOA.9:XPP50

        

        LDR    R1, =USART1    ;9600

        MOV    R0,#0x341

        STR    R0,[R1,#0x08]

        

        LDR    R1, =USART1    ;RX + TX, SUART1_ON

        MOV    R0,#0x200C

        STR    R0,[R1,#0x0C]

        

        MOV    R0,#0x01

        LDR    R1,=TI

        STR    R0,[R1,#0x00]

        LDR    R1,=RI

        STR    R0,[R1,#0x00]

        

                LDR     R0, =__main

                BX      R0

                ENDP

    主要做的工作就是在进行调用__main代码前,做好系统时钟配置,系统IO配置,USART模式配置并允许. 

为了彻底摆脱固件库,并且让scanf支持回显功能,把stdio.h改成这样:

#define SBUF    USART1->DR                //接收发送寄存器

#define TI        (*(u32*)0x4227001c)        //发送结束标志

#define RI        (*(u32*)0x42270014)        //接收标志

……

/**************************************************************************

* Function Name  : PUTCHAR_PROTOTYPE

* Description    : Retargets the C library printf function to the USART.

* Input          : None

* Output         : None

* Return         : None

**************************************************************************/

PUTCHAR_PROTOTYPE

{

  /* Write a character to the USART */

  SBUF = ch;

 

  /* Loop until the end of transmission */

  while(!TI)

  {

      ;

  }

 

  return ch;

}

 

 

void Usend(u8 ch)

{

    SBUF = ch;

   

    while (!TI)

    {

        ;

    }

}

 

GETCHAR_PROTOTYPE

{

    int temp = 0;

 

Gagain:

        

    while (!RI)

    {

        ;

    }

   

    temp = SBUF;

    RI = 0;

 

    if (temp == 0x0d)

    {

        Usend(0x0a);

        Usend(0x0d);

    }

    else if (temp == 0x08)

    {

        goto Gagain;

    }

    else

    {

        Usend(temp);

    }    

   

    return (temp);    

}

经过这番修理,现在工程效果如下:

 

整个工程简洁了很多,main.c文件也清爽了.

再看软件仿真运行结果:



现在已经支持回显功能了.最后我们正式把结果应用到实际中,以
POJ上的第一道题为例.

 

A+B Problem

Time Limit: 1000MS

 

Memory Limit: 10000K

Total Submissions: 257447

 

Accepted: 139582

Description

Calculate a+b

Input

Two integer a,b (0<=a,b<=10)

Output

Output a+b

Sample Input

1 2

Sample Output

3

 



复制main.c中的代码到提交窗口, 点提交, 直接通过



推荐阅读

史海拾趣

安信可(Ai)公司的发展小趣事

在不断创新和突破的过程中,安信可不断推出具有创新性的产品。例如,公司推出的WiFi+BLE蓝牙二合一的Ai-WB1系列模组,凭借其高可靠度、高集成度、应用灵活、高性价比等特点,深受市场欢迎。这款产品不仅为智能设备提供了无线通信连接能力,还广泛应用于工业、商业、消费等各类市场,进一步拓展了安信可的应用领域。

创都(CAX)公司的发展小趣事

在电子行业中,产品质量是企业生存和发展的关键。创都公司深知这一点,因此在品质管理上投入了大量精力。公司建立了严格的质量管理体系,从原材料采购到生产流程控制,再到产品出厂检验,每一个环节都经过严格把关。此外,创都公司还积极引进先进的生产设备和技术,提升生产效率和产品质量。这些措施使得创都公司的产品在市场上享有良好的口碑,赢得了消费者的信任和喜爱。

DuPont公司的发展小趣事

为了进一步拓展电子市场,DuPont公司积极寻求与全球知名电子企业的合作。通过与国际知名品牌的战略合作,DuPont成功进入了多个重要的电子市场领域,如智能手机、平板电脑、汽车电子等。这些合作不仅为DuPont公司带来了广阔的市场空间,也促进了公司在电子领域的技术创新和产品升级。

Dielectric Laboratories公司的发展小趣事

作为一家在电子行业中具有重要地位的企业,DLI始终将创新作为公司的核心竞争力。公司不断投入研发资金,探索新的技术领域,推出具有领先性的产品。例如,在电容器领域,DLI研发了StackiCapTM技术,使得高CV电容器得到进一步发展。这一技术的推出不仅提升了DLI的产品竞争力,也为整个电子行业的发展注入了新的活力。同时,DLI还积极参与国际交流和合作,与全球知名的电子企业建立合作关系,共同推动电子行业的发展。

以上五个故事简要概述了Dielectric Laboratories(DLI)公司在电子行业中的发展历程和成就。通过这些故事,我们可以看到DLI在不断创新、追求卓越的过程中,逐渐成为了电子行业中具有重要影响力的企业。

千志电子(CCO)公司的发展小趣事

随着技术的不断进步和市场的日益成熟,千志电子开始注重电阻产业的深耕。公司不仅专注于电阻的生产,还逐渐向电阻专用设备、原材料等领域延伸。2006年,千志电子成立了深圳市鑫兴志实业有限公司,主要生产电阻相关的生产机器如切割机、焊接机、成型机、涂装机等。同时,千志电子还成立了千志电子科技(湖北)有限公司,生产各类型电阻器、设备及电阻所需原材料如碳棒、线材等。这一战略调整使千志电子形成了从原材料到设备的完整产业链,提高了生产效率和产品质量,进一步巩固了其在电阻行业的领先地位。

FDK AMERICA公司的发展小趣事

在竞争激烈的电子市场中,FDK AMERICA公司始终坚持品质至上的原则。公司严格控制产品质量,从原材料采购到生产流程再到售后服务,每一个环节都力求做到精益求精。这种对品质的执着追求赢得了客户的信赖,也为公司树立了良好的品牌形象。

问答坊 | AI 解惑

控制范围在1000米左右,怎么实现?

本人对RF/无线知之甚少,对于遥控范围在1000米左右的项目,该怎么实现呢?…

查看全部问答>

机器人制作(三)电机的特性与选择

本帖最后由 paulhyde 于 2014-9-15 09:22 编辑 机器人运动的效果好不好,电机的选择很关键  …

查看全部问答>

周立功LPC2200开发板启动代码分析

周立功LPC2200开发板启动代码分析…

查看全部问答>

如何在winCE系统中通过填写注册表安装输入法???

如题,想通过写注册表来安装新的输入法,应该怎么做??以全拼输入法为例,除了写入winpy.ime和kbdus.dll两个文件外,还需要怎么做????另外,这两个文件应该拷贝到WinCE系统的什么位置?????…

查看全部问答>

关于xdata unsigned char MYRAM _at_ 0xdfff的简单问题

xdata unsigned char MYRAM _at_ 0xdfff定义之后,在程序中能不能对MYRAM进行改变?比如 MYRAM++; 我用一个循环向片外数据RAM依次写入数据,用C语言如何实现?…

查看全部问答>

char*作形参引发的问题,可以追加给分

void copywrite(char* string) { } void mystring(char* name) {    copywrite(name); } 1 void main() {   mystring(\"my name is\"); } 2 void main() {   char str[20]=\"my name is\";  &nbs ...…

查看全部问答>

IFIX和S7-400用S7A通讯

我的系统是多台电脑和PLC通信。其中一台完全正常,其他的就有我下面说的问题。IFIX和400通信,我在S7A上看见的通信状态是GOOD。但是在PDB里面,有个向下写的数据块有些数据没有值?。两台上位机和PLC通信应该怎么弄? 还有个问题,就是启动IFIX的时 ...…

查看全部问答>

SPI接口问个问题

spi多个从机的时候,如果片选信号同时设为两个以上的从机使能,那两个从机同时向主机发送数据,该怎么办?spi也没有仲裁机制…

查看全部问答>

STM32数字和模拟部分供电

                                 数字和模拟部分供电想单独供电,但不知道直接数字部分或直接模拟部分,会不会损坏芯片,从手册上上好象不给接数字 ...…

查看全部问答>

做网络通信 LM3S6911 能直接替换 LM3S6432 吗

做网络通信,不改变电路,LM3S6911 能直接替换 LM3S6432 吗?…

查看全部问答>