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

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

相关文章

C#实现系统信息监控与获取功能

《C#实现系统信息监控与获取功能》在C#开发的众多应用场景中,获取系统信息以及监控用户操作有着广泛的用途,比如在系统性能优化工具中,需要实时读取CPU、GPU资源信息,本文将详细介绍如何使用C#来实现... 目录前言一、C# 监控键盘1. 原理与实现思路2. 代码实现二、读取 CPU、GPU 资源信息1.

Oracle查询优化之高效实现仅查询前10条记录的方法与实践

《Oracle查询优化之高效实现仅查询前10条记录的方法与实践》:本文主要介绍Oracle查询优化之高效实现仅查询前10条记录的相关资料,包括使用ROWNUM、ROW_NUMBER()函数、FET... 目录1. 使用 ROWNUM 查询2. 使用 ROW_NUMBER() 函数3. 使用 FETCH FI

在C#中获取端口号与系统信息的高效实践

《在C#中获取端口号与系统信息的高效实践》在现代软件开发中,尤其是系统管理、运维、监控和性能优化等场景中,了解计算机硬件和网络的状态至关重要,C#作为一种广泛应用的编程语言,提供了丰富的API来帮助开... 目录引言1. 获取端口号信息1.1 获取活动的 TCP 和 UDP 连接说明:应用场景:2. 获取硬

C#使用HttpClient进行Post请求出现超时问题的解决及优化

《C#使用HttpClient进行Post请求出现超时问题的解决及优化》最近我的控制台程序发现有时候总是出现请求超时等问题,通常好几分钟最多只有3-4个请求,在使用apipost发现并发10个5分钟也... 目录优化结论单例HttpClient连接池耗尽和并发并发异步最终优化后优化结论我直接上优化结论吧,

Java内存泄漏问题的排查、优化与最佳实践

《Java内存泄漏问题的排查、优化与最佳实践》在Java开发中,内存泄漏是一个常见且令人头疼的问题,内存泄漏指的是程序在运行过程中,已经不再使用的对象没有被及时释放,从而导致内存占用不断增加,最终... 目录引言1. 什么是内存泄漏?常见的内存泄漏情况2. 如何排查 Java 中的内存泄漏?2.1 使用 J

JAVA系统中Spring Boot应用程序的配置文件application.yml使用详解

《JAVA系统中SpringBoot应用程序的配置文件application.yml使用详解》:本文主要介绍JAVA系统中SpringBoot应用程序的配置文件application.yml的... 目录文件路径文件内容解释1. Server 配置2. Spring 配置3. Logging 配置4. Ma

2.1/5.1和7.1声道系统有什么区别? 音频声道的专业知识科普

《2.1/5.1和7.1声道系统有什么区别?音频声道的专业知识科普》当设置环绕声系统时,会遇到2.1、5.1、7.1、7.1.2、9.1等数字,当一遍又一遍地看到它们时,可能想知道它们是什... 想要把智能电视自带的音响升级成专业级的家庭影院系统吗?那么你将面临一个重要的选择——使用 2.1、5.1 还是

C#使用yield关键字实现提升迭代性能与效率

《C#使用yield关键字实现提升迭代性能与效率》yield关键字在C#中简化了数据迭代的方式,实现了按需生成数据,自动维护迭代状态,本文主要来聊聊如何使用yield关键字实现提升迭代性能与效率,感兴... 目录前言传统迭代和yield迭代方式对比yield延迟加载按需获取数据yield break显式示迭

高效管理你的Linux系统: Debian操作系统常用命令指南

《高效管理你的Linux系统:Debian操作系统常用命令指南》在Debian操作系统中,了解和掌握常用命令对于提高工作效率和系统管理至关重要,本文将详细介绍Debian的常用命令,帮助读者更好地使... Debian是一个流行的linux发行版,它以其稳定性、强大的软件包管理和丰富的社区资源而闻名。在使用

Ubuntu系统怎么安装Warp? 新一代AI 终端神器安装使用方法

《Ubuntu系统怎么安装Warp?新一代AI终端神器安装使用方法》Warp是一款使用Rust开发的现代化AI终端工具,该怎么再Ubuntu系统中安装使用呢?下面我们就来看看详细教程... Warp Terminal 是一款使用 Rust 开发的现代化「AI 终端」工具。最初它只支持 MACOS,但在 20