`
lmyessential
  • 浏览: 31515 次
  • 性别: Icon_minigender_1
  • 来自: 福州
社区版块
存档分类
最新评论

关于位域

    博客分类:
  • ARM
阅读更多

在s3c2410的时钟和电源管理模块中有一个寄存器MPLLCON为MPLL锁相环的配置寄存器:,以下为其中的一些位定义:
MDIV [19:12]   Main divider control
PDIV  [9:4]       Pre-divider control  
SDIV  [1:0]       Post divider control
这里就涉及到如何从 MPLLCON中获取这三个值的问题,一种简单的办法是利用掩码,比如要获取MDIV的值:
        #define MDIV_MASK    0x00ff0000
        MDIV=MPLLCON & MDIV_MASK
               MDIV=MDIV >>12
在linux内核和vivi bootloader中使用了另外一种通用方法,那就是位域。

请看下面这个获取CPU时钟的函数:

unsigned  long s3c2410_get_cpu_clk(void)
{
    unsigned long val=MPLLCON;
    return(    (   (GET_MDIV(val)+8) *FIN    )/  (   ( GET_PDIV(val)+2 )  *   (  1<<GET_SDIV(val)  )     )    );
                
}
以下为宏定义:
#define GET_MDIV(x)        FExtr(x, fMPLL_MDIV)
#define GET_PDIV(x)        FExtr(x, fMPLL_PDIV)
#define GET_SDIV(x)        FExtr(x, fMPLL_SDIV)
#define fMPLL_MDIV        Fld(8,12)
#define fMPLL_PDIV           Fld(6,4)
#define fMPLL_SDIV            Fld(2,0)
其中的宏定义在一个处理位域的头文件中:
/*
*    FILE        bitfield.h
*
*    Version     1.1
*    Author      Copyright (c) Marc A. Viredaz, 1998
*                DEC Western Research Laboratory, Palo Alto, CA
*    Date        April 1998 (April 1997)
*    System      Advanced RISC Machine (ARM)
*    Language    C or ARM Assembly
*    Purpose     Definition of macros to operate on bit fields.
*/



#ifndef __BITFIELD_H
#define __BITFIELD_H

#ifndef __ASSEMBLY__
#define UData(Data)    ((unsigned long) (Data))
#else
#define UData(Data)    (Data)
#endif


/*
* MACRO: Fld
*
* Purpose
*    The macro "Fld" encodes a bit field, given its size and its shift value
*    with respect to bit 0.
*
* Note
*    A more intuitive way to encode bit fields would have been to use their
*    mask. However, extracting size and shift value information from a bit
*    field's mask is cumbersome and might break the assembler (255-character
*    line-size limit).
*
* Input
*    Size          Size of the bit field, in number of bits.
*    Shft          Shift value of the bit field with respect to bit 0.
*
* Output
*    Fld           Encoded bit field.
*/

#define Fld(Size, Shft)    (((Size) << 16) + (Shft))


/*
* MACROS: FSize, FShft, FMsk, FAlnMsk, F1stBit
*
* Purpose
*    The macros "FSize", "FShft", "FMsk", "FAlnMsk", and "F1stBit" return
*    the size, shift value, mask, aligned mask, and first bit of a
*    bit field.
*
* Input
*    Field         Encoded bit field (using the macro "Fld").
*
* Output
*    FSize         Size of the bit field, in number of bits.
*    FShft         Shift value of the bit field with respect to bit 0.
*    FMsk          Mask for the bit field.
*    FAlnMsk       Mask for the bit field, aligned on bit 0.
*    F1stBit       First bit of the bit field.
*/

#define FSize(Field)    ((Field) >> 16)
#define FShft(Field)    ((Field) & 0x0000FFFF)
#define FMsk(Field)    (((UData (1) << FSize (Field)) - 1) << FShft (Field))
#define FAlnMsk(Field)    ((UData (1) << FSize (Field)) - 1)
#define F1stBit(Field)    (UData (1) << FShft (Field))


/*
* MACRO: FInsrt
*
* Purpose
*    The macro "FInsrt" inserts a value into a bit field by shifting the
*    former appropriately.
*
* Input
*    Value         Bit-field value.
*    Field         Encoded bit field (using the macro "Fld").
*
* Output
*    FInsrt        Bit-field value positioned appropriately.
*/

#define FInsrt(Value, Field) \
                    (UData (Value) << FShft (Field))


/*
* MACRO: FExtr
*
* Purpose
*    The macro "FExtr" extracts the value of a bit field by masking and
*    shifting it appropriately.
*
* Input
*    Data          Data containing the bit-field to be extracted.
*    Field         Encoded bit field (using the macro "Fld").
*
* Output
*    FExtr         Bit-field value.
*/

#define FExtr(Data, Field) \
                    ((UData (Data) >> FShft (Field)) & FAlnMsk (Field))


#endif /* __BITFIELD_H */

 呵呵,位域果然强大,当初看《Programming In C》 时提到了位域,还专门用了一章来讲,现在终于明白为什么了。
以下以通过GET_MDIV(MPLLCON)获取MDIV为例,简要说明一下位域的处理过程:
#define  MPLLCON       0X4C000004
           x= (*(volatile unsigned long *)(MPLLCON));    x 取MPLLCON中的值
          ;此例中假设MPLLCON的值为0x0005C080

        #define GET_MDIV(x)        FExtr(x, fMPLL_MDIV)
      ==>          GET_MDIV(x) = FExtr(x,fMPLL_MDIV)
       #define fMPLL_MDIV        Fld(8,12)
         ==>        FExtr(x,fMPLL_MDIV)=FExtr(x,Fld(8,12) )
    
    #define Fld(Size, Shft)    (((Size) << 16) + (Shft))
      ==>    Fld(8,12)   = 0x0008000C
    
#define FExtr(Data, Field)                    ((UData (Data) >> FShft (Field)) & FAlnMsk (Field))
       ==>   FExtr(x, 0x0008000C)= (0x0005C080>>FShft(0x0008000C))&FAlnMsk(0x0008000C)

     #define FShft(Field)    ((Field) & 0x0000FFFF)
         ==>  Fshft(0x0008000C)=  0x0000000C=12
      #define FAlnMsk(Field)    ((UData (1) << FSize (Field)) - 1)
        ==>         FAlnMsk(0x0008000C)= ( 0x00000001<<FSize(0x0008000C) )  - 1
        #define    FSize(Field)    ((Field) >> 16)        
           ==> FSize(0x0008000C)= 0x00000008
       ==>     FAlnMsk(0x0008000C)=0x000000FF

       ==>FExtr(x, 0x0008000C)=(0x0005C080>>12)&0x000000FF
                                               =0x0000005C
正确的提取出了结果
研究具体实现细节比较累,但是使用起来确是很方便的,包含bitifield.h头文件,然后就可以使用预先定义好的良好的宏了,就像上面所做的那样。
下面是我对要使用这种通用方法的原因的理解:
              通用性,避免重复输入,避免手动移位,计算掩码等操作时因粗心引入错误。使用这种方法只需提供  位域的位数和偏移值,比较方便。

分享到:
评论

相关推荐

    C语言中关于位域的概念和使用

    描述了在C语言中如何对数据的位进行操作,使用FPGA的嵌入式设计应用

    行业分类-电子-关于位域操作电路及位域操作方法的说明分析.rar

    位域操作在电子工程,尤其是数字电路设计中是一项至关重要的技术。位域操作涉及到对二进制数据中的特定比特位进行设置、清除或测试,它广泛应用于嵌入式系统、微处理器、数字信号处理和各种电子设备的控制逻辑中。在...

    DSP 位域文件详解

    关于 DSP 位域文件的详细解释将在下面进行。 DSP 寄存器 DSP 寄存器是 DSP 微处理器的核心组件,负责存储和处理数据。在 DSP 中,寄存器可以分为通用寄存器和特殊寄存器两种。通用寄存器用于存储数据和指令,而...

    数据结构位域

    ### 数据结构位域详解 #### 一、位域概述 位域是C语言提供的一种特殊的数据结构,主要用于节省存储空间。在很多情况下,某些变量的值并不需要占用一个完整的字节,仅需占用几位二进制位就足够了。比如表示一个开关...

    C语言中的位域的使用

    ### C语言中的位域使用详解 #### 一、位域的概念及作用 在C语言编程中,有时我们可能需要处理一些只需要几个比特位就能表示的数据,例如开关状态、某些标志位等。在这种情况下,如果使用普通的整型或者字符型变量...

    c/c++类的位域详细介绍

    "C/C++位域详细介绍" C/C++中的位域(Bit Field)是指在结构体中使用的位域成员,它们可以将整数成员存储到比编译器通常允许的更小的内存空间中。位域的宽度可以通过常量整数表达式来指明。这种技术广泛应用于需要...

    C# 位域方面的解释

    C# 位域方面的解释 C# 位域是一种特殊的枚举类型,用于表示多种复合的状态。通过使用 [Flags] 特性标记,枚举可以被用来表示多个状态的组合。 在 C# 中,枚举可以用两种方式使用:一种是表示唯一的元素序列,例如...

    结构体位域说明

    结构体位域说明 结构体位域说明 结构体位域说明

    C++ 位域

    **C++ 位域概述** 位域是C++语言中的一种特性,允许程序员在结构体或联合体中按位来定义变量,以便高效地处理内存中的二进制数据。位域的概念源于计算机内部数据存储的基本单位是位(bit),通过位域可以更精确地...

    linux c 简单位域操作

    linux c 位域操作 报文 位域操作 报文 位域操作 报文 简单位域操作 位域操作 报文 位域操作 报文 位域操作 报文

    Linux c位域操作完整版 经典版

    Linux c位域操作完整版 经典版 Linux c位域操作完整版 经典版

    c c++位域研究总结!!!

    位域研究总结 位域(Bit-fields)是 C 语言和 C++ 语言中都有的一个概念,但是位域有很多需要注意的问题。其中一个重要的问题是大端和小端字节序的问题。 大端和小端字节序是计算机科学中的一种约定,用于描述多...

    C语言位域的使用方法

    详细描述位域的使用方法. 所谓“位域”是把一个字节中的二进位划分为几个不同的区域, 并说明每个区域的位数。每个域有一个域名,允许在程序中按域名进行操作。 这样就可以把几个不同的对象用一个字节的二进制位域来...

    C语言结构体之位域详解

    文章主要对C语言结构体的位域知识进行详细介绍。

    IAR FOR STM8位域的定义和应用

    在使用IAR Embedded Workbench for STM8进行编程时,理解位域的定义和应用至关重要,因为这有助于优化内存使用和提高程序效率。 位域(Bit Field)是C语言提供的一种特性,允许我们为一个结构体内的变量定义更小的...

    VB.net C# 位域设置读取原代码

    关于.net开发面向对象程序 VB、C#语言时 有关整数位域使用方法的源码,可提供读写任意位bit的数据 文档提供VB源码,C#可参考使用

    int-to-bits.rar_Int.intToIntBits_int-bits_int赋值给位域_整数存入位域

    标题中的“int-to-bits.rar”表明这是一个关于将整数转换为位表示的代码压缩包,而“Int.intToIntBits”可能是一个函数名,用于将整数转换为位域。描述中提到的是如何将整数的值传递给位域,这在处理硬件接口、数据...

    C语言 位域PPT课件.pptx

    【位域】 位域是C语言中结构体(struct)的一种特殊形式,它允许我们定义结构体成员的位长度,使得我们可以精确控制存储空间的使用。位域主要用于处理那些只需要少量存储空间(小于一个字节)的数据,比如标志位...

    C++点滴(初学者)

    - **6.6.1 关于位域** - **6.7 sizeof用于表达式** - **6.8 一个经典问题** #### 七、内联inline函数 **7.1 C语言中的宏** 宏是一种简单的文本替换机制,但它没有类型安全性和函数调用的成本。 **7.2 C++中引入...

    内存对其-结构体、位域结构体

    ### 内存对齐:结构体与位域结构体详解 在计算机科学中,内存对齐(Memory Alignment)是一项至关重要的技术,它涉及到数据在内存中的存储方式,尤其是在结构体和位域结构体的设计与使用中。内存对齐能够显著提高...

Global site tag (gtag.js) - Google Analytics