驱动中位域操作简析

2024-04-17 18:32
文章标签 操作 驱动 简析 中位域

本文主要是介绍驱动中位域操作简析,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

文章来源:http://blog.mcuol.com/User/lvembededsys/Article/6994_1.htm

 

在SM501的驱动中,寄存器非常多,每个寄存器位域的定义也特别多。 驱动采用了一套非常奇妙的宏来操作寄存器的位域。觉得有必要总结一下。
1)相关 定义
包括3 个方面:寄存器,寄存器位域,寄存器位域的可取值。
其中,寄存器位域的表示用以0为起始索引的 高位:低位 来表示。这点非常重要,因为后面的宏定义就是利用了其中的冒号,来构造一个条件操作符。
三种定义在标识符安排上的关系:寄存器为R,则R中某位域F定义为R_F,则该位域某个值V定义为R_F_V.:

比如:
#define SYSTEM_CTRL                                     0x000000
#define SYSTEM_CTRL_DPMS                                31:30
#define SYSTEM_CTRL_DPMS_VPHP                           0
#define SYSTEM_CTRL_DPMS_VPHN                           1
#define SYSTEM_CTRL_DPMS_VNHP                           2
#define SYSTEM_CTRL_DPMS_VNHN                           3

2)低层宏操作
#define _F_START(f)             (0 ? f)
#define _F_END(f)               (1 ? f)
#define _F_SIZE(f)              (1 + _F_END(f) - _F_START(f))
#define _F_MASK(f)              (((1 << _F_SIZE(f)) - 1) << _F_START(f))
#define _F_NORMALIZE(v, f)      (((v) & _F_MASK(f)) >> _F_START(f))
#define _F_DENORMALIZE(v, f)    (((v) << _F_START(f)) & _F_MASK(f))

其中f 表示位域,v表示位域的值。
 _F_START(f) 表示取位域中的起始索引,即冒号右边的值。展开就是一个条件操作符。如 _F_START(SYSTEM_CTRL_DPMS) 展开就相当于 
(0?31:30) 显然值为30,正是表示起始索引。 其余就比较好理解了。

_F_END(f) 表示 位域的结束索引位置。即位域冒号的左边的值。 
_F_SIZE(f) 表示 位域的宽度,也就是包含几位。
_F_MASK(f)为位掩码,在_F_START(f) 到 _F_END(f)的位为1,其余为0。
_F_NORMALIZE(v, f) 以整数值的方式得到位域的值。位域右移到0开始的位置。
_F_DENORMALIZE(v, f)将以整数表示的值放到位域中去的表示。将值左移到位域的位置。

上面这些是底层宏定义。
3)位域宏操作:
实际上对位域的操作往往是获取或设定某寄存器某位域的值。这些宏定义如下 

#define FIELD_GET(x, reg, field) /
( /
    _F_NORMALIZE((x), reg ## _ ## field) /
)

#define FIELD_SET(x, reg, field, value) /
( /
    (x & ~_F_MASK(reg ## _ ## field)) /
    | _F_DENORMALIZE(reg ## _ ## field ## _ ## value, reg ## _ ## field) /
)
#define FIELD_CLEAR(reg, field) /
( /
    ~ _F_MASK(reg ## _ ## field) /
)
上述三个宏只返回值,并不修改寄存器。上述宏定义中,field,value都不是完整的宏定义名称,完整的宏定义名称是reg_field,reg_field_value。这就是为什么3个定义要按照这种关系来写的原因。好处是关系更清晰。代码显得短小。

一般的使用过程 是 使用 FIELD_GET获取值,再使用FIELD_SET修改某个位域的值,最后用新值修改寄存器。典型的使用:
void panelPlaneEnable(void)
{
    unsigned long panel_ctrl = 0;
    panel_ctrl = regRead32(PANEL_DISPLAY_CTRL); //
    panel_ctrl = FIELD_SET(panel_ctrl, PANEL_DISPLAY_CTRL, PLANE, ENABLE);//修改
    regWrite32(PANEL_DISPLAY_CTRL, panel_ctrl);//
}

这样做的好处是程序可以只使用寄存器,位域,位域值,和几个宏来进行操作。可读性好,也简化了编程!!

这篇关于驱动中位域操作简析的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

Linux_kernel驱动开发11

一、改回nfs方式挂载根文件系统         在产品将要上线之前,需要制作不同类型格式的根文件系统         在产品研发阶段,我们还是需要使用nfs的方式挂载根文件系统         优点:可以直接在上位机中修改文件系统内容,延长EMMC的寿命         【1】重启上位机nfs服务         sudo service nfs-kernel-server resta

动手学深度学习【数据操作+数据预处理】

import osos.makedirs(os.path.join('.', 'data'), exist_ok=True)data_file = os.path.join('.', 'data', 'house_tiny.csv')with open(data_file, 'w') as f:f.write('NumRooms,Alley,Price\n') # 列名f.write('NA

线程的四种操作

所属专栏:Java学习        1. 线程的开启 start和run的区别: run:描述了线程要执行的任务,也可以称为线程的入口 start:调用系统函数,真正的在系统内核中创建线程(创建PCB,加入到链表中),此处的start会根据不同的系统,分别调用不同的api,创建好之后的线程,再单独去执行run(所以说,start的本质是调用系统api,系统的api

Java IO 操作——个人理解

之前一直Java的IO操作一知半解。今天看到一个便文章觉得很有道理( 原文章),记录一下。 首先,理解Java的IO操作到底操作的什么内容,过程又是怎么样子。          数据来源的操作: 来源有文件,网络数据。使用File类和Sockets等。这里操作的是数据本身,1,0结构。    File file = new File("path");   字

MySQL——表操作

目录 一、创建表 二、查看表 2.1 查看表中某成员的数据 2.2 查看整个表中的表成员 2.3 查看创建表时的句柄 三、修改表 alter 3.1 重命名 rename 3.2 新增一列 add 3.3 更改列属性 modify 3.4 更改列名称 change 3.5 删除某列 上一篇博客介绍了库的操作,接下来来看一下表的相关操作。 一、创建表 create

封装MySQL操作时Where条件语句的组织

在对数据库进行封装的过程中,条件语句应该是相对难以处理的,毕竟条件语句太过于多样性。 条件语句大致分为以下几种: 1、单一条件,比如:where id = 1; 2、多个条件,相互间关系统一。比如:where id > 10 and age > 20 and score < 60; 3、多个条件,相互间关系不统一。比如:where (id > 10 OR age > 20) AND sco

PHP7扩展开发之流操作

前言 啥是流操作?简单来讲就是对一些文件,网络的IO操作。PHP已经把这些IO操作,封装成流操作。这节,我们将使用PHP扩展实现一个目录遍历的功能。PHP示例代码如下: <?phpfunction list_dir($dir) {if (is_dir($dir) === false) {return;} $dh = opendir($dir);if ($dh == false) {ret

浙大数据结构:树的定义与操作

四种遍历 #include<iostream>#include<queue>using namespace std;typedef struct treenode *BinTree;typedef BinTree position;typedef int ElementType;struct treenode{ElementType data;BinTree left;BinTre

浙大数据结构:04-树7 二叉搜索树的操作集

这道题答案都在PPT上,所以先学会再写的话并不难。 1、BinTree Insert( BinTree BST, ElementType X ) 递归实现,小就进左子树,大就进右子树。 为空就新建结点插入。 BinTree Insert( BinTree BST, ElementType X ){if(!BST){BST=(BinTree)malloc(sizeof(struct TNo

hibernate修改数据库已有的对象【简化操作】

陈科肇 直接上代码: /*** 更新新的数据并并未修改旧的数据* @param oldEntity 数据库存在的实体* @param newEntity 更改后的实体* @throws IllegalAccessException * @throws IllegalArgumentException */public void updateNew(T oldEntity,T newEntity