Skip to content

Commit

Permalink
AttributeContainer.Large now handles more than 127 attributes.
Browse files Browse the repository at this point in the history
The previous implementation failed with more than 127 attributes because the call to Arrays.binarySearch treats the byte values as signed, so values of 128 and above wrapped around and became negative, breaking binarySearch's precondition that the array be sorted.

Also adds a small test.

PiperOrigin-RevId: 345302645
  • Loading branch information
katre authored and meisterT committed Dec 15, 2020
1 parent 75a1158 commit b2023c5
Show file tree
Hide file tree
Showing 2 changed files with 51 additions and 27 deletions.
Original file line number Diff line number Diff line change
Expand Up @@ -16,7 +16,6 @@
import com.google.common.annotations.VisibleForTesting;
import com.google.common.base.Preconditions;
import com.google.errorprone.annotations.CheckReturnValue;
import java.util.Arrays;
import java.util.BitSet;
import javax.annotation.Nullable;

Expand Down Expand Up @@ -154,14 +153,33 @@ private static int nonNullCount(Object[] attrValues) {
return numSet;
}

/** Returns index into state array for attrIndex, or -1 if not found */
private static int getStateIndex(byte[] state, int start, int attrIndex, int mask) {
// Binary search, treating values as unsigned bytes.
int lo = start;
int hi = state.length - 1;
while (hi >= lo) {
int mid = (lo + hi) / 2;
int midAttrIndex = state[mid] & mask;
if (midAttrIndex == attrIndex) {
return mid;
} else if (midAttrIndex < attrIndex) {
lo = mid + 1;
} else {
hi = mid - 1;
}
}
return -1;
}

/**
* A frozen implementation of AttributeContainer which supports RuleClasses with up to 126
* attributes.
*/
@VisibleForTesting
static final class Small extends Frozen {

private int maxAttrCount;
private final int maxAttrCount;

// Conceptually an AttributeContainer is an unordered set of triples
// (attribute, value, explicit).
Expand Down Expand Up @@ -219,25 +237,6 @@ private Small(Object[] attrValues, BitSet explicitAttrs) {
}
}

/** Returns index into state array for attrIndex, or -1 if not found */
private int getStateIndex(int attrIndex) {
// Binary search on the bottom 7 bits.
int lo = 0;
int hi = state.length - 1;
while (hi >= lo) {
int mid = (lo + hi) / 2;
int midAttrIndex = state[mid] & 0x7f;
if (midAttrIndex == attrIndex) {
return mid;
} else if (midAttrIndex < attrIndex) {
lo = mid + 1;
} else {
hi = mid - 1;
}
}
return -1;
}

/**
* Returns true iff the value of the specified attribute is explicitly set in the BUILD file. In
* addition, this method return false if the rule has no attribute with the given name.
Expand All @@ -247,7 +246,7 @@ boolean isAttributeValueExplicitlySpecified(int attrIndex) {
if (attrIndex < 0) {
return false;
}
int stateIndex = getStateIndex(attrIndex);
int stateIndex = getStateIndex(state, 0, attrIndex, 0x7f);
return stateIndex >= 0 && (state[stateIndex] & 0x80) != 0;
}

Expand All @@ -258,7 +257,7 @@ Object getAttributeValue(int attrIndex) {
throw new IndexOutOfBoundsException(
"Maximum valid attrIndex is " + (maxAttrCount - 1) + ". Given " + attrIndex);
}
int stateIndex = getStateIndex(attrIndex);
int stateIndex = getStateIndex(state, 0, attrIndex, 0x7f);
return stateIndex < 0 ? null : values[stateIndex];
}
}
Expand Down Expand Up @@ -308,16 +307,16 @@ private static int prefixSize(int attrCount) {
/** Set the specified bit in the byte array. Assumes bitIndex is a valid index. */
private static void setBit(byte[] bits, int bitIndex) {
int idx = (bitIndex + 1);
byte explicitByte = bits[idx >> 3];
int explicitByte = bits[idx >> 3];
byte mask = (byte) (1 << (idx & 0x07));
bits[idx >> 3] = (byte) (explicitByte | mask);
}

/** Get the specified bit in the byte array. Assumes bitIndex is a valid index. */
private static boolean getBit(byte[] bits, int bitIndex) {
int idx = (bitIndex + 1);
byte explicitByte = bits[idx >> 3];
byte mask = (byte) (1 << (idx & 0x07));
int explicitByte = bits[idx >> 3];
int mask = (byte) (1 << (idx & 0x07));
return (explicitByte & mask) != 0;
}

Expand Down Expand Up @@ -374,7 +373,7 @@ Object getAttributeValue(int attrIndex) {
"Maximum valid attrIndex is " + (maxAttrCount - 1) + ". Given " + attrIndex);
}
int p = prefixSize(maxAttrCount);
int stateIndex = Arrays.binarySearch(state, p, state.length, (byte) attrIndex);
int stateIndex = getStateIndex(state, p, attrIndex, 0xff);
return stateIndex < 0 ? null : values[stateIndex - p];
}
}
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -158,4 +158,29 @@ public void testFreezeWorks_smallImplementation() {
public void testFreezeWorks_largeImplementation() {
checkFreezeWorks((short) 150, AttributeContainer.Large.class);
}

private void testContainerSize(int size) {
AttributeContainer container = new Mutable(size);
for (int i = 0; i < size; i++) {
container.setAttributeValue(i, "value " + i, i % 2 == 0);
}
AttributeContainer frozen = container.freeze();
// Check values.
for (int i = 0; i < size; i++) {
assertThat(frozen.getAttributeValue(i)).isEqualTo("value " + i);
assertThat(frozen.isAttributeValueExplicitlySpecified(i)).isEqualTo(i % 2 == 0);
}
}

@Test
public void testSmallContainer() {
// At 127 attributes, we shift from AttributeContainer.Small to AttributeContainer.Large.
testContainerSize(126);
}

@Test
public void testLargeContainer() {
// AttributeContainer.Large can handle at max 254 attributes.
testContainerSize(254);
}
}

0 comments on commit b2023c5

Please sign in to comment.