嵌入式系统程序可移植性设计及性能优化之一――宏定义设计

2023-10-25 19:09

本文主要是介绍嵌入式系统程序可移植性设计及性能优化之一――宏定义设计,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

 

 

嵌入式系统程序可移植性设计及性能优化之一

――――宏定义设计

 

Sailor_forever  sailing_9806@163.com 转载请注明

http://blog.csdn.net/sailor_8318/archive/ 2008/07/16 /2663254.aspx

 

 

【摘要】本节介绍了嵌入式系统程序设计中采用宏定义进行常量定义的必要性。说明了宏常量定义的基本规则以及如何采用依赖关系定义宏常量来保证其可移植性和裁减性。最后介绍了如何利用宏定义实现掩码偏移量等来高效的进行位操作。

 

【关键词】嵌入式,可移植性,宏定义,依赖关系,掩码,偏移量,位操作

 

 

1   宏定义设计... - 1 -

1.1     为何要采用宏定义?... - 1 -

1.2     宏定义的基本规则... - 1 -

1.3     依赖关系定义宏改善移植性... - 1 -

1.4     通过偏移量和掩码进行位操作... - 2 -

 

 

1       宏定义设计

1.1    为何要采用宏定义?

在程序设计过程中,对于经常使用的一些常量,如果将它直接写到程序中去,一旦常量的数值发生变化,就必须逐个找出程序中所有的常量,并逐一进行修改,这样必然会降低程序的可维护性。因此,应尽量通过预处理命令方式将常量定义为特定的字符,这样常量就有了统一的表现形式,不会出现输入错误导致的不一致性。另外宏常量意义明确,大大改善了代码的可读性。

 

只读的变量也可以实现上述宏常量所带来的可移植性、可靠性及可读性等特点,但其要占据存储空间,需要访问内存,相比宏常量的立即数寻址而言效率要低。在C++中提倡用const只读变量来定义常量,是因为这样可以提供更严格的类型安全检查。但由于Cconst只读变量不能用于某些场合,因此在嵌入式C中仍多数采用宏来定义常量。

 

1.2    宏定义的基本规则

下面以一个实例来说明宏定义的基本规则,如用预处理指令#define 声明一个常量,用以表明1年中有多少秒,不考虑润年

#define C_SECONDS_PER_YEAR (60 * 60 * 24 * 365)UL

a)       命名风格,为了与普通变量区分开来,宏定义通常全部大写,多个单元之间用下划线隔开;

b)       整个表达式应括起来,若有参数则应将每个参数都括起来,防止替换扩展后可能带来的异常问题;

c)       常量表达式先合并后再替换。预处理器将为你计算常量表达式的值,因此,直接写出你是如何计算一年中有多少秒而不是计算出实际的值,是更清晰而没有运行性能代价的。

d)       为常量添加类型信息。宏的不足之一在于缺乏类型安全检查,人为的提供类型信息可以有效检查出此类问题。UL告诉编译器这个常量是无符号长整型数,因此将其赋值给u16型变量会出现告警。

 

1.3    依赖关系定义宏改善移植性

嵌入式系统程序的最大特点是硬件平台的多变性,因此需要根据具体的应用情况更改大量配置,而这些配置基本都是由宏定义来实现的,放在特定的头文件中,与其他的代码隔离,在一定程度上改善了代码的可移植性。但有些时候,多个宏定义有严重的依赖关系,增减某个宏会引起其他定义的更改,如何定义这些宏对嵌入式程序的可移植性有很大影响。

 

A

常量分别定义

#define C_DD_MODULE_ID_AOM  (0x00010101)     /* AOM模块ID */

#define C_DD_MODULE_ID_RRCM (0x00010102)        /* RRCM模块ID */

#define C_DD_MODULE_ID_RLC (0x00010103)        /* RLC模块ID */

#define C_DD_MODULE_ID_TRM  (0x00010104)       /* TRM模块ID */

#define C_DD_MODULE_ID_MCP_MIN        (C_DD_MODULE_ID_AOM)                

#define C_DD_MODULE_ID_MCP_MAX     (C_DD_MODULE_ID_TRM)

B

依赖定义

#define C_DD_MODULE_ID_MCP_MIN  (0x00010101) /* MCP最小模块ID */

#defineC_DD_MODULE_ID_AOM (C_DD_MODULE_ID_MCP_MIN)                        /* AOM模块ID */

#define C_DD_MODULE_ID_RRCM      (C_DD_MODULE_ID_AOM + 1)                        /* RRCM模块ID */

#define C_DD_MODULE_ID_RLC  (C_DD_MODULE_ID_RRCM + 1)                        /* RLC模块ID */

#define C_DD_MODULE_ID_TRM (C_DD_MODULE_ID_RLC + 1)                        /* TRM模块ID */

#define C_DD_MODULE_ID_MCP_MAX (C_DD_MODULE_ID_TRM)                /* MCP最大模块ID */

 

A常量分别定义”方式,因为各个宏定义值必须连续,若更改或者删除C_DD_MODULE_ID_AOM,其他的定义基本都受到影响,将严重影响到代码的可扩充性和可裁减性。

 

B依赖定义”方式,其原则是:

a)       base用常量定义;

b)       第一个定义为base

c)       其他的在上一个基础上加1

d)       max项即为最后一项。

这样整体改动起来只需要更改base;在中间删除或添加部分项时只需要更改一个上下衔接处即可;添加则比较简单,只需要在原有最后项后添加即可。

这种方式若改动部分定义对其他定义的影响最小,大大改善了代码的可移植性。

 

1.4    通过偏移量和掩码进行位操作

嵌入式系统经常需要和硬件打交道,而配置硬件寄存器则是系统初始化阶段的重要任务,如何清晰明了的进行配置决定了代码的可读性。尽管可以使用位域,但位域是不可移植的,因此利用宏定义来解决可移植性问题。

 

对于每一个待操作相应位来说,应具备以下几个标识:

待操作的位名称B_NAME

操作位所占据的位宽B_WIDTH_NAME

操作位第一位的偏移量B_SHIFT_NAME

操作位的掩码B_MASK_NAME

 

PRI为例进行说明:

#define PRI  DD_C64_EDMA_OPT_PRI

#define C_WIDTH _DD_C64_EDMA_OPT_PRI          (3)

#define C_SHIFT_DD_C64_EDMA_OPT_PRI          (29)

 

可以手动定义对应位的掩码如下

#define C_MASK_DD_C64_EDMA_OPT_PRI           (0xe0000000)

但更好的方式是利用位宽和偏移量来自动构成掩码

#define BIT_MASK(_name)  (((1U<< C_WIDTH _##_name))-1)<<( C_SHIFT_##_name)

#define C_MASK_DD_C64_EDMA_OPT_PRI  BIT_MASK(PRI)

BIT_MASK(PRI)经过宏替换后即为:

(((1U<<( C_WIDTH _DD_C64_EDMA_OPT_PRI))-1)  /

<<( C_SHIFT_DD_C64_EDMA_OPT_PRI)

 

对于每个位的取值也应该用宏定义来表示,这样清晰明确

#define C_DD_C64_EDMA_OPT_URGENT_PRI         (0x0)

#define C_DD_C64_EDMA_OPT_HIGH_PRI           (0x1)

#define C_DD_C64_EDMA_OPT_MEDIUM_PRI         (0x2)

#define C_DD_C64_EDMA_OPT_LOW_PRI            (0x3)

 

具备了掩码和偏移量即可对各个位进行操作了

#define SET_BITS(_reg,_name_val)/

((_reg)=((_ reg)&~(BIT_MASK(_name))) /

| (((_val)<<( C_SHIFT_##_ name))&(BIT_MASK(_name))))

 

通过如下方式调用设置优先级

SET_BITS(u32Opt, PRI, C_DD_C64_EDMA_OPT_HIGH_PRI);

即实现了将u32OptPRI等位设置为C_DD_C64_EDMA_OPT_HIGH_PRI

 

SET_BITS宏可应用于待操作的位为多位的情况,当待操作的位仅为一位时,可用更简单的操作方式

#define SET_BIT(_reg,_name)   ((_reg) |= (BIT_MASK(_name))

#define CLR_BIT(_reg,_name)   ((_reg) &= ~(BIT_MASK(_name))

 

TCINT为例:

#define TCINT  DD_C64_EDMA_OPT_TCINT

#define C_WIDTH _DD_C64_EDMA_OPT_TCINT  (1)

#define C_SHIFT_DD_C64_EDMA_OPT_PRI          (20)

SET_BIT(u32Opt, TCINT)

CLR_BIT(u32Opt, TCINT)

 

 

这篇关于嵌入式系统程序可移植性设计及性能优化之一――宏定义设计的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/284603

相关文章

Mac系统下卸载JAVA和JDK的步骤

《Mac系统下卸载JAVA和JDK的步骤》JDK是Java语言的软件开发工具包,它提供了开发和运行Java应用程序所需的工具、库和资源,:本文主要介绍Mac系统下卸载JAVA和JDK的相关资料,需... 目录1. 卸载系统自带的 Java 版本检查当前 Java 版本通过命令卸载系统 Java2. 卸载自定

CSS Anchor Positioning重新定义锚点定位的时代来临(最新推荐)

《CSSAnchorPositioning重新定义锚点定位的时代来临(最新推荐)》CSSAnchorPositioning是一项仍在草案中的新特性,由Chrome125开始提供原生支持需... 目录 css Anchor Positioning:重新定义「锚定定位」的时代来了! 什么是 Anchor Pos

基于Python实现一个简单的题库与在线考试系统

《基于Python实现一个简单的题库与在线考试系统》在当今信息化教育时代,在线学习与考试系统已成为教育技术领域的重要组成部分,本文就来介绍一下如何使用Python和PyQt5框架开发一个名为白泽题库系... 目录概述功能特点界面展示系统架构设计类结构图Excel题库填写格式模板题库题目填写格式表核心数据结构

Linux系统中的firewall-offline-cmd详解(收藏版)

《Linux系统中的firewall-offline-cmd详解(收藏版)》firewall-offline-cmd是firewalld的一个命令行工具,专门设计用于在没有运行firewalld服务的... 目录主要用途基本语法选项1. 状态管理2. 区域管理3. 服务管理4. 端口管理5. ICMP 阻断

MyBatis设计SQL返回布尔值(Boolean)的常见方法

《MyBatis设计SQL返回布尔值(Boolean)的常见方法》这篇文章主要为大家详细介绍了MyBatis设计SQL返回布尔值(Boolean)的几种常见方法,文中的示例代码讲解详细,感兴趣的小伙伴... 目录方案一:使用COUNT查询存在性(推荐)方案二:条件表达式直接返回布尔方案三:存在性检查(EXI

python编写朋克风格的天气查询程序

《python编写朋克风格的天气查询程序》这篇文章主要为大家详细介绍了一个基于Python的桌面应用程序,使用了tkinter库来创建图形用户界面并通过requests库调用Open-MeteoAPI... 目录工具介绍工具使用说明python脚本内容如何运行脚本工具介绍这个天气查询工具是一个基于 Pyt

Ubuntu设置程序开机自启动的操作步骤

《Ubuntu设置程序开机自启动的操作步骤》在部署程序到边缘端时,我们总希望可以通电即启动我们写好的程序,本篇博客用以记录如何在ubuntu开机执行某条命令或者某个可执行程序,需要的朋友可以参考下... 目录1、概述2、图形界面设置3、设置为Systemd服务1、概述测试环境:Ubuntu22.04 带图

Python程序打包exe,单文件和多文件方式

《Python程序打包exe,单文件和多文件方式》:本文主要介绍Python程序打包exe,单文件和多文件方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录python 脚本打成exe文件安装Pyinstaller准备一个ico图标打包方式一(适用于文件较少的程

Windows 系统下 Nginx 的配置步骤详解

《Windows系统下Nginx的配置步骤详解》Nginx是一款功能强大的软件,在互联网领域有广泛应用,简单来说,它就像一个聪明的交通指挥员,能让网站运行得更高效、更稳定,:本文主要介绍W... 目录一、为什么要用 Nginx二、Windows 系统下 Nginx 的配置步骤1. 下载 Nginx2. 解压

如何确定哪些软件是Mac系统自带的? Mac系统内置应用查看技巧

《如何确定哪些软件是Mac系统自带的?Mac系统内置应用查看技巧》如何确定哪些软件是Mac系统自带的?mac系统中有很多自带的应用,想要看看哪些是系统自带,该怎么查看呢?下面我们就来看看Mac系统内... 在MAC电脑上,可以使用以下方法来确定哪些软件是系统自带的:1.应用程序文件夹打开应用程序文件夹