2015-03-23 16:03:14 -05:00
|
|
|
/*
|
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))) */
|
|
|
|
|
2015-06-01 10:28:29 -05: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) {
|
|
|
|
|
2016-11-08 10:25:07 +01:00
|
|
|
/*
|
|
|
|
* Big endian
|
2015-02-26 09:02:31 -06:00
|
|
|
*/
|
|
|
|
|
|
|
|
mask = 0x00000001 << (start + bits - 1);
|
|
|
|
|
|
|
|
if (mask & value) {
|
|
|
|
|
2016-11-08 10:25:07 +01:00
|
|
|
/*
|
|
|
|
* This is a negative value
|
2015-02-26 09:02:31 -06:00
|
|
|
*/
|
|
|
|
|
|
|
|
for (i = start, j = 0; j < bits; i++, j++) {
|
2016-11-08 10:25:07 +01:00
|
|
|
/*
|
|
|
|
* First, turn off all bits associated with
|
2015-02-26 09:02:31 -06:00
|
|
|
* this bit field within the unsigned integer.
|
2016-11-08 10:25:07 +01:00
|
|
|
* 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 {
|
|
|
|
|
2016-11-08 10:25:07 +01:00
|
|
|
/*
|
|
|
|
* This is a positive value
|
2015-02-26 09:02:31 -06:00
|
|
|
*/
|
|
|
|
|
|
|
|
for (i = start, j = 0; j < bits; i++, j++) {
|
2016-11-08 10:25:07 +01:00
|
|
|
/*
|
|
|
|
* First, turn off all bits associated with
|
2015-02-26 09:02:31 -06:00
|
|
|
* this bit field within the unsigned integer.
|
2016-11-08 10:25:07 +01:00
|
|
|
* 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 {
|
|
|
|
|
2016-11-08 10:25:07 +01:00
|
|
|
/*
|
|
|
|
* Little endian
|
2015-02-26 09:02:31 -06:00
|
|
|
*/
|
|
|
|
|
2016-11-08 10:25:07 +01: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);
|
|
|
|
}
|