How do I extract specific 'n' bits of a 32-bit unsigned integer in C?

前端 未结 8 2311
庸人自扰
庸人自扰 2020-12-02 08:02

Could anyone tell me as to how to extract \'n\' specific bits from a 32-bit unsigned integer in C.

For example, say I want the first 17 bits of the 32-bit value;

相关标签:
8条回答
  • 2020-12-02 08:27

    If you want n bits specific then you could first create a bitmask and then AND it with your number to take the desired bits.

    Simple function to create mask from bit a to bit b.

    unsigned createMask(unsigned a, unsigned b)
    {
       unsigned r = 0;
       for (unsigned i=a; i<=b; i++)
           r |= 1 << i;
    
       return r;
    }
    

    You should check that a<=b.

    If you want bits 12 to 16 call the function and then simply & (logical AND) r with your number N

    r = createMask(12,16);
    unsigned result = r & N;
    

    If you want you can shift the result. Hope this helps

    0 讨论(0)
  • 2020-12-02 08:30

    Modulus works to get bottom bits (only), although I think value & 0x1ffff expresses "take the bottom 17 bits" more directly than value % 131072, and so is easier to understand as doing that.

    The top 17 bits of a 32-bit unsigned value would be value & 0xffff8000 (if you want them still in their positions at the top), or value >> 15 if you want the top 17 bits of the value in the bottom 17 bits of the result.

    0 讨论(0)
  • 2020-12-02 08:38

    Bitwise AND your integer with the mask having exactly those bits set that you want to extract. Then shift the result right to reposition the extracted bits if desired.

    unsigned int lowest_17_bits = myuint32 & 0x1FFFF;
    unsigned int highest_17_bits = (myuint32 & (0x1FFFF << (32 - 17))) >> (32 - 17);
    

    Edit: The latter repositions the highest 17 bits as the lowest 17; this can be useful if you need to extract an integer from “within” a larger one. You can omit the right shift (>>) if this is not desired.

    0 讨论(0)
  • 2020-12-02 08:39
    #define GENERAL__GET_BITS_FROM_U8(source,lsb,msb) \
        ((uint8_t)((source) & \
            ((uint8_t)(((uint8_t)(0xFF >> ((uint8_t)(7-((uint8_t)(msb) & 7))))) & \
                 ((uint8_t)(0xFF << ((uint8_t)(lsb) & 7)))))))
    
    #define GENERAL__GET_BITS_FROM_U16(source,lsb,msb) \
        ((uint16_t)((source) & \
            ((uint16_t)(((uint16_t)(0xFFFF >> ((uint8_t)(15-((uint8_t)(msb) & 15))))) & \
                ((uint16_t)(0xFFFF << ((uint8_t)(lsb) & 15)))))))
    
    #define GENERAL__GET_BITS_FROM_U32(source,lsb,msb) \
        ((uint32_t)((source) & \
            ((uint32_t)(((uint32_t)(0xFFFFFFFF >> ((uint8_t)(31-((uint8_t)(msb) & 31))))) & \
                ((uint32_t)(0xFFFFFFFF << ((uint8_t)(lsb) & 31)))))))
    
    0 讨论(0)
  • 2020-12-02 08:40

    There is a single BEXTR (Bit field extract (with register)) x86 instruction on Intel and AMD CPUs and UBFX on ARM. There are intrinsic functions such as _bextr_u32() (link requires sign-in) that allow to invoke this instruction explicitly.

    They implement (source >> offset) & ((1 << n) - 1) C code: get n continuous bits from source starting at the offset bit. Here's a complete function definition that handles edge cases:

    #include <limits.h>
    
    unsigned getbits(unsigned value, unsigned offset, unsigned n)
    {
      const unsigned max_n = CHAR_BIT * sizeof(unsigned);
      if (offset >= max_n)
        return 0; /* value is padded with infinite zeros on the left */
      value >>= offset; /* drop offset bits */
      if (n >= max_n)
        return value; /* all  bits requested */
      const unsigned mask = (1u << n) - 1; /* n '1's */
      return value & mask;
    }
    

    For example, to get 3 bits from 2273 (0b100011100001) starting at 5-th bit, call getbits(2273, 5, 3)—it extracts 7 (0b111).

    For example, say I want the first 17 bits of the 32-bit value; what is it that I should do?

    unsigned first_bits = value & ((1u << 17) - 1); // & 0x1ffff
    

    Assuming CHAR_BIT * sizeof(unsigned) is 32 on your system.

    I presume I am supposed to use the modulus operator and I tried it and was able to get the last 8 bits and last 16 bits

    unsigned last8bitsvalue  = value & ((1u <<  8) - 1); // & 0xff
    unsigned last16bitsvalue = value & ((1u << 16) - 1); // & 0xffff
    

    If the offset is always zero as in all your examples in the question then you don't need the more general getbits(). There is a special cpu instruction BLSMSK that helps to compute the mask ((1 << n) - 1).

    0 讨论(0)
  • 2020-12-02 08:42

    Instead of thinking of it as 'extracting', I like to think of it as 'isolating'. Once the desired bits are isolated, you can do what you will with them.

    To isolate any set of bits, apply an AND mask.

    If you want the last X bits of a value, there is a simple trick that can be used.

    unsigned  mask;
    mask = (1 << X) - 1;
    lastXbits = value & mask;
    

    If you want to isolate a run of X bits in the middle of 'value' starting at 'startBit' ...

    unsigned  mask;
    mask = ((1 << X) - 1) << startBit;
    isolatedXbits = value & mask;
    

    Hope this helps.

    0 讨论(0)
提交回复
热议问题