历史上的今天
返回首页

历史上的今天

今天是:2024年08月27日(星期二)

正在发生

2019年08月27日 | 【STM32H7教程】第9章 STM32H7重要知识点数据类型,变量和堆栈

2019-08-27 来源:eefocus

9.1   初学者重要提示

1、 如果对C语言不熟练的话,可以阅读C语言书:C Primer Plus(第五版)中文版.pdf


论坛下载:http://forum.armfly.com/forum.php?mod=viewthread&tid=91219。


 2、 为了更好的学习本章知识点,可以看之前做的视频教程第10章,针对H7也将在今年发布视频教程:


http://forum.armfly.com/forum.php?mod=viewthread&tid=15408 。


9.2   数据类型

了解数据类型之前要对ANSI C和ISO C的发展史有个了解,特别是C89,C99和C11的由来。


9.2.1      ANSI C和ISO C历史

  1983年,美国国家标准协会(ANSI)组成了一个委员会来创立C语言的标准。因为这个标准是1989年发布的,所以一般简称C89标准。有些人也把C89标准叫做ANSI C。

  在1990年,ANSI C89标准被国际标准化组织(ISO)和国际电工委员会(IEC)采纳为国际标准,名叫ISO/IEC 9899:1990 - Programming languages  C,有些人简称C90标准。因此,C89和C90通常指同一个标准,一般更常用C89这种说法。

  在2000年3月,国际标准化组织(ISO)和国际电工委员会(IEC)采纳了第二个C语言标准,名叫ISO/IEC 9899:1999 - Programming languages -- C,简称C99标准。

  在2011年12月,国际标准化组织(ISO)和国际电工委员会(IEC)采纳了第三个C语言标准,名叫ISO/IEC 9899:2011 - Information technology -- Programming languages -- C,简称C11标准。它是C程序语言的最新标准。

 


对于我们常用的编译器MDK和IAR而已,C89,C99和C11均支持。


9.2.2      ARM架构(含Cortex-M系列)数据类型  

ARM架构(含Cortex-M系列)的数据类型定义如下:


9.2.3      头文件stdint.h对数据类型的定义

stdint.h是C99中引进的一个标准C库的头文件。目前大部分单片机C编译器均支持,IAR和MDK都支持。下面是部分内容(来自MDK)。


/* exact-width signed integer types */

typedef   signed          char int8_t;

typedef   signed short     int int16_t;

typedef   signed           int int32_t;

typedef   signed       __INT64 int64_t;

 

/* exact-width unsigned integer types */

typedef unsigned          char uint8_t;

typedef unsigned short     int uint16_t;

typedef unsigned           int uint32_t;

typedef unsigned       __INT64 uint64_t;

 

/* 7.18.1.2 */

 

/* smallest type of at least n bits */

/* minimum-width signed integer types */

typedef   signed          char int_least8_t;

typedef   signed short     int int_least16_t;

typedef   signed           int int_least32_t;

typedef   signed       __INT64 int_least64_t;

 

/* minimum-width unsigned integer types */

typedef unsigned          char uint_least8_t;

typedef unsigned short     int uint_least16_t;

typedef unsigned           int uint_least32_t;

typedef unsigned       __INT64 uint_least64_t;

 

/* 7.18.1.3 */

 

/* fastest minimum-width signed integer types */

typedef   signed           int int_fast8_t;

typedef   signed           int int_fast16_t;

typedef   signed           int int_fast32_t;

typedef   signed       __INT64 int_fast64_t;

 

    /* fastest minimum-width unsigned integer types */

typedef unsigned           int uint_fast8_t;

typedef unsigned           int uint_fast16_t;

typedef unsigned           int uint_fast32_t;

typedef unsigned       __INT64 uint_fast64_t;

以MDK5.26为例,stdint.h位于如下路径:


Keil_v526ARMARMCCinclude


以IAR8.X为,stdint.h位于如下路径:


IAR SystemsEmbedded Workbench 8.1armincc


9.2.4      程序中推荐的变量命名方式

看程序的时候,经常会看到各种各样的变量命名方式,例如声明一个8位无符号整数,常见的有如下几种写法:


u8 err;


U8 err;


INT8U err;


UINT8 err;


uint8 err;


uint8_t err;


当大家阅读别人写的程序时,往往会看到风格各异的定义方式,移植部分程序时也不知道采用哪种方式更合适。


我们推荐大家采用最后一种定义方式,这种方法符合C99规范,ST的固件库都是采用的这种类型定义方式。像我们开发板配套的STM32例程,从2009年最初的版本开始就一直沿用C99的标准写法来定义整数。


 


  知识点拓展

针对变量声明问题,之前专门发过一个详细的帖子:


http://forum.armfly.com/forum.php?mod=viewthread&tid=501


9.3   局部变量和全局变量

9.3.1      局部变量

在一个函数内部定义的变量是内部变量,它只在本函数范围内有效,也就是说只有在本函数内才能使用它们,在此函数以外是不能使用这些变量的,这称为局部变量。


使用局部变量注意以下问题:


  不同函数中可以使用相同名字的变量,它们代表不同的对象,互不干扰。

  形式参数也是局部变量。

  局部变量的作用域在函数内部。

9.3.2      全局变量

     在函数内部定义的变量是局部变量,而在函数之外定义的变量称为外部变量,也就是全局变量。使用全局变量的注意事项:


  全局变量可以为本文件中其他函数所共用。它的有效范围为从定义变量的位置开始到本源文件结束。

  设置全局变量的作用是增加了函数间数据联系的渠道。

 如果在同一个源文件中,外部变量和局部变量同名,则在局部变量的作用范围内,外部变量被“屏蔽”,即外部变量将不起作用。

9.3.3      使用全局变量的缺点

    程序设计中,建议不要创建太多的全局变量,主要是出于以下三点考虑:


  全局变量在程序的执行过程中都占用存储单元,而不是仅在需要时才占用存储单元。

  函数的通用性降低了,因为函数在执行时要依赖于其所在的外部变量。如果将一个函数移植到另一个文件中,还要将有关的外部变量及其值一起移植过去。

  使用全局变量过多,会降低程序的清晰性,特别是多个函数都调用此变量时。

9.3.4      变量的存储类别

从变量的作用域来分,可以分为全局变量和局部变量,而从变量值存在的时间来看,可以分为静态存储方式和动态存储方式。


  静态存储方式:指在程序运行期间由系统分配固定的存储空间方式。

  动态存储方式:在程序运行期间根据需要进行动态的分配存储空间方式。

全局变量存储在静态存储区中,动态存储区可以存放以下数据:


  函数形式参数,在调用函数时给形参分配存储空间。

  局部变量(未加static声明的局部变量)。

  函数调用时的现场保护和返回地址等。

9.3.5      用static声明局部或者全局变量

有时候希望函数中的局部变量的值在函数调用结束后不消失而保留原值,即占用的存储单元不释放,在下一次该函数调用时,该变量已有值,就是上一次函数调用结束时的值。这时可以使用关键字static进行声明。


用static声明一个变量的作用:


  对局部变量用static声明,则使用该变量在整个程序执行期间不释放,为其分配的的空间始终存在。

  全局变量用static声明,则该变量的作用域只限于本文件模块(即被声明的文件中)。

9.4   堆栈

9.4.1      堆栈作用

栈(stack)空间,用于局部变量,函数调时现场保护和返回地址,函数的形参等。


堆(heap)空间,主要用于动态内存分配,也就是说用 malloc,calloc, realloc等函数分配的变量空间是在堆上。以STM32H7为例,堆栈是在startup_stm32h743xx.s文件里面设置:


9.4.2      寄存器组(堆栈指针寄存器)

Cortex – M7/M4/M3 处理器拥有R0-R15的通用寄存器组。其中R13作为堆栈指针SP。SP有两个,但在同一时刻只能有一个可以用。


  主堆栈指针(MSP):这是缺省的堆栈指针,它由OS内核、异常服务例程以及所有需要特权访问的应用程序代码来使用。

  进程堆栈指针(PSP):用于常规的应用程序代码(不处于异常服务例程中时)。

另外以下两点要注意:


  大多数情况下的应用,只需使用指针MSP,而PSP多用于 RTOS 中。

  R13 的最低两位被硬线连接到0,并且总是读出0,这意味着堆栈总是4字节对齐的。

9.4.3      Cortex-M7/M4/M3向下生长的满栈

这个知识点在以后用H7移植RTOS时,非常有用。


9.4.4      堆栈的基本操作

这里对入栈和出栈做个简单的介绍。PUSH入栈操作:SP先自减 4,再存入新的数值:

POP出栈操作:先从SP指针处读出上一次被压入的值,再把SP指针自增 4:


9.5   局部变量,全局变量和堆栈实例

通过下面的实例可以对局部变量,全局变量和堆栈有个感性的认识:


uint32_t a = 0;    //全局初始化区, 可以被其他c文件 extern 引用

static uint32_t ss = 0;    //静态变量,只允许在本文件使用

uint8_t *p1;         //全局未初始化区

int main(void)

{

uint32_t b;             //栈

uint8_t s[] = "abc";    //栈

uint8_t *p2;            //栈

uint8_t *p3 = "123456"; //123456在常量区,p3在栈上。

static uint32_t c =0;   //全局(静态)初始化区

p1 = (uint8_t *)malloc(10); //在堆区申请了10个字节空间

p2 = (uint8_t *)malloc(20); //在堆区申请了20个字节空间

strcpy(p1, "123456");      /* 123456字符串(结束符号是0(),总长度7)放在常量区,

                              编译器可能会将它与p3所指向的"123456"优化成一个地方 */

}

通过查看MAP文件,可以看全局变量在RAM中的位置:


Symbol Name                      Value        Ov Type        Size    Object(Section)

a                               0x20000000     Data            4         main.o(.data)

p1                              0x2000000c     Data            4         main.o(.data)

ss                              0x20000004     Data            4         main.o(.data)

而局部变量要调整状态进入main函数里面查看:


9.6   总结

C语言的基础知识点要掌握牢靠,对于后面学习HAL库源码大有裨益。


推荐阅读

史海拾趣

驰芯微(CHIPWISE)公司的发展小趣事

驰芯微公司与明然科技的合作是公司在技术创新方面的又一重要突破。双方共同研发的MCU控制芯片成功应用于主动悬架项目,并在奇瑞的星途瑶光、瑞虎9等车型上取得了重要进展。这一合作不仅提升了国产芯片在高端配置车型中的应用水平,也进一步推动了驰芯微在汽车电子领域的发展。

FutureWafer公司的发展小趣事

近年来,国际环境复杂多变,某些国家对我国进行“芯片限制”。然而,这一限制反而为驰芯微带来了机遇。在“芯片限制”的背景下,更多国内车企开始寻求与国内芯片企业的合作。驰芯微凭借其在车规级芯片领域的丰富经验和卓越性能,成功获得了多家车企的青睐。公司抓住这一机遇,加大了与车企的合作力度,进一步巩固了其在行业中的地位。

Array Microsystems Inc公司的发展小趣事

为了支持公司的快速发展和技术创新,Array Microsystems Inc高度重视人才引进和团队建设。公司积极招聘业内优秀的技术人才和管理人才,为他们提供良好的工作环境和福利待遇。同时,公司还注重员工的培训和晋升机制,鼓励员工不断学习和成长。这些措施使得Array Microsystems Inc拥有了一支高素质、专业化的团队,为公司的发展提供了有力的人才保障。

迈翔科技(COILMX)公司的发展小趣事

迈翔科技(COILMX)的创始人看到了电子行业,特别是电感器市场的巨大潜力。1996年,公司正式成立,初期主要专注于SMD电感的设计、制造和销售。公司从香港起步,逐步在深圳设立工厂,引进先进的生产设备和技术,为未来的发展奠定了坚实的基础。

晶群科技(Gem-micro)公司的发展小趣事

面对未来,迈翔科技有着明确的发展规划和目标。公司将继续加大技术研发投入,不断提升产品性能和质量;同时,公司也将积极拓展新的应用领域和市场,寻求更多的合作机会和发展空间。未来,迈翔科技将致力于成为全球领先的电感器制造商之一。

以上五个故事概述了迈翔科技在电子行业中的发展历程和关键事件。这些故事展示了公司从创业初期到成为行业领导者的艰辛历程和不懈努力。

Datalogic公司的发展小趣事

在2022年,Datalogic推出了PowerScan 9600系列工业手持式扫描枪。这款扫描枪具有坚固的设计、IP67等级防护和出色的性能,能够在严酷的工业环境中稳定运行。PowerScan 9600系列的推出进一步提升了Datalogic在电子行业中的产品竞争力。

问答坊 | AI 解惑

请您帮忙,给点建议或程序都行。

我正在开发一个键盘,我使用51单片机 控制像机和云台的,上面有一些按键和一个摇杆。功能按键的功能如数字键、控制摄像机变焦等功能键、还有如调预至位、复位等功能。摇杆就是控制云台转动的方向,可以上下左右等水平360度波动,这样可以控制云台水 ...…

查看全部问答>

遇到了一个诡异的问题。

VC代码移植到EVC4.0.编译没问题,但画面不能正常显示。图形在ONDRAW函数里画的,2万行左右。 于是我一点点的添加,开始是正常的,后来就不正常了,有行数限制? 有时候多初始化一个矩形就不行,有时候把矩形初始化赋值的代码注释起来,就能继续添 ...…

查看全部问答>

不能正确读取开关量

连接如图,用128的pe口读取八个开关,如果全部不连通没有问题,如果全部接通也没有问题,问题是:如果只接通SW8,则PE02和PE03均为零,如果接通SW6,则PE04和PE05均为零,这是为什么呀??245有问题?还是?…

查看全部问答>

哈哈~突然发现自己太有才了有木有啊,3个IO口控制一个12864+4个LED~上图

如题如题。。。 一片小小的244芯片帮了我的大忙啊,说多无用,有图有真相。…

查看全部问答>

电路在空载时出现的问题

请教大家,电路在空载时为什么会产生间隙震荡呢?…

查看全部问答>

信号放大中遇到的问题

谁能告诉我做信号放大在什么条件下需要多个电路?谢谢了,嘿嘿…

查看全部问答>

请教各位OrCAD Capture CIS的一个小问题

不是很熟啊,小问题请教一下 制作元器件 用SpreadSheet放置管脚 Top和Right方向管脚序号 左到右,上到下顺序,如何改成右到左和下到上啊。没查到。 谢谢。…

查看全部问答>

32K晶振焊接方法

本人菜鸟一枚,业余爱好从taobao淘的3套launchpad(20元一套),到手2个月一直不知道怎么焊接那个小小的32768的晶振,搜遍全网也没个教程,只有天朝之外的youtube有一篇视频教程,无奈啊。今天终于被逼无奈自己尝试焊接,用我那双颤抖的双手居然焊 ...…

查看全部问答>

为什么我的CCS 5.2不能退出DEBUG ?

调试目标C2000,可以进入DEBUG模式,可以进行正常调试,但是,退出就不行了,一点退出(就是那个红方块)就死着不动。只有关闭了。 每次这样很麻烦。且,原来编译时的状态不保存。…

查看全部问答>

FPGA实现UART收发器

module UART ( rst,clk50M,send_en,rev_en,TXD,TXD_data,RXD,RXD_data); input rst,clk50M,RXD,send_en,rev_en; input [7:0] TXD_data; output TXD; reg TXD; reg [7:0] TXD_cnt; output [7:0] RXD_data; reg [7:0] RXD_data; reg [7:0] RXD_cnt; ...…

查看全部问答>