124 lines
3.4 KiB
C
Raw Normal View History

/*
2015-02-26 09:02:31 -06:00
PURPOSE: (Extract a bitfield. And return bitfield value.)
REFERENCE: ((meow) (woof))
PROGRAMMERS: (((Jane Falgout) (Lincom)) ((Alex Lin) (NASA) (8/00))) */
#include "trick/bitfield_proto.h"
2015-02-26 09:02:31 -06:00
int extract_bitfield_any(int value, /* In: Value to extract bits from */
int size, /* In: Declared size of bitfield */
int start, /* In: Starting bit */
int bits)
{ /* In: Number of bits in bitfield */
int i = 0, j = 0;
unsigned int mask;
unsigned int bf;
int sbf;
union {
long l;
char c[sizeof(long)];
} un;
un.l = 1; /* Used for Little/Big Endian Detection */
bf = 0x00000000;
sbf = 0x00000000;
if (un.c[sizeof(long) - 1] == 1) {
/*
* Big endian
2015-02-26 09:02:31 -06:00
*/
mask = 0x00000001 << (start + bits - 1);
if (mask & value) {
/*
* This is a negative value
2015-02-26 09:02:31 -06:00
*/
for (i = start, j = 0; j < bits; i++, j++) {
/*
* First, turn off all bits associated with
2015-02-26 09:02:31 -06:00
* this bit field within the unsigned integer.
* Then assign the input value to the
2015-02-26 09:02:31 -06:00
* appropriate bits in the unsigned integer.
*/
mask = 0x00000001 << i;
if ((value & mask) != mask) {
sbf |= mask;
}
}
sbf = -((sbf >> start) + 1);
} else {
/*
* This is a positive value
2015-02-26 09:02:31 -06:00
*/
for (i = start, j = 0; j < bits; i++, j++) {
/*
* First, turn off all bits associated with
2015-02-26 09:02:31 -06:00
* this bit field within the unsigned integer.
* Then assign the input value to the appropriate
2015-02-26 09:02:31 -06:00
* bits in the unsigned integer.
*/
mask = 0x00000001 << i;
if ((value & mask) == mask) {
bf |= mask;
}
}
sbf = bf >> start;
}
} else {
/*
* Little endian
2015-02-26 09:02:31 -06:00
*/
/*
2015-02-26 09:02:31 -06:00
* Shift the bit field contents of the current value
* to the lsb of the underlying int.
*/
if (size == sizeof(short)) {
value = value << 16;
} else if (size == sizeof(char)) {
value = value << 24;
}
if ((start + bits) < 32) {
bf = (value) >> (32 - start - bits);
} else {
bf = value;
}
/* Now, AND the shifted bits with a mask of the appropriate size to obtain the bitfield value. */
bf &= (0xFFFFFFFF >> (32 - bits));
/* If this bitfield is a signed bitfield and the sign bit is set, take the two's complement of the value. */
mask = 0x00000001 << (bits - 1);
if (mask & bf) {
/* This bitfield is signed and has the sign bit set. Set sbf to the reverse bits of the bitfield contents.
Then right shift and add 1. */
sbf = 0x00000000;
for (i = 0; i < bits; i++) {
mask = 0x00000001 << i;
if ((bf & mask) != mask) {
sbf |= mask;
}
}
sbf = -(sbf + 1);
} else {
sbf = bf;
}
}
return (sbf);
}