/*
Copyright 2014 Immutables Authors and Contributors
Licensed under the Apache License, Version 2.0 (the "License");
you may not use this file except in compliance with the License.
You may obtain a copy of the License at
http://www.apache.org/licenses/LICENSE-2.0
Unless required by applicable law or agreed to in writing, software
distributed under the License is distributed on an "AS IS" BASIS,
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
See the License for the specific language governing permissions and
limitations under the License.
*/
package org.immutables.value.processor.meta;
import javax.annotation.Nullable;
import com.google.common.base.Function;
import com.google.common.collect.ImmutableList;
import com.google.common.collect.ImmutableMap;
import com.google.common.collect.ImmutableSortedMap;
import com.google.common.collect.Maps;
import com.google.common.collect.Multimaps;
import com.google.common.primitives.Longs;
import java.util.Collection;
import java.util.IdentityHashMap;
import java.util.Set;
import static com.google.common.base.Preconditions.*;
/**
* Structure to calculate bit packing
*/
public final class LongBits implements Function<Iterable<? extends Object>, LongBits.LongPositions> {
private static final int BITS_IN_LONG = Longs.BYTES * Byte.SIZE;
@Override
public LongPositions apply(Iterable<? extends Object> input) {
return forIterable(input, BITS_IN_LONG);
}
public LongPositions forIterable(Iterable<? extends Object> input, int bitPerLong) {
return new LongPositions(input, bitPerLong);
}
public static final class LongPositions implements Function<Object, BitPosition> {
private final IdentityHashMap<Object, BitPosition> positions = Maps.newIdentityHashMap();
private final ImmutableList<Object> elements;
private final ImmutableMap<Integer, LongSet> longPositions;
LongPositions(Iterable<? extends Object> elements, final int bitPerLong) {
this.elements = ImmutableList.copyOf(elements);
checkArgument(bitPerLong <= BITS_IN_LONG, bitPerLong);
for (int i = 0; i < this.elements.size(); i++) {
positions.put(
this.elements.get(i),
new BitPosition(
i / bitPerLong,
i % bitPerLong));
}
this.longPositions = ImmutableSortedMap.copyOf(
Maps.transformEntries(
Multimaps.index(positions.values(), ToLongIndex.FUNCTION).asMap(),
new Maps.EntryTransformer<Integer, Collection<BitPosition>, LongSet>() {
@Override
public LongSet transformEntry(Integer key, Collection<BitPosition> position) {
return new LongSet(key, position);
}
}));
}
public Set<Integer> longsIndeces() {
return longPositions.keySet();
}
public Collection<LongSet> longs() {
return longPositions.values();
}
@Nullable
@Override
public BitPosition apply(Object input) {
return positions.get(input);
}
}
public static final class LongSet {
public final int index;
public final long occupation;
public final Iterable<BitPosition> positions;
LongSet(int index, Iterable<BitPosition> positions) {
this.index = index;
this.positions = ImmutableList.copyOf(positions);
this.occupation = computeOccupation();
}
private long computeOccupation() {
long occupation = 0;
for (BitPosition position : this.positions) {
occupation |= position.mask;
}
return occupation;
}
}
enum ToLongIndex implements Function<BitPosition, Integer> {
FUNCTION;
@Override
public Integer apply(BitPosition input) {
return input.index;
}
}
public static final class BitPosition {
public final int index;
public final int bit;
public final long mask;
BitPosition(int index, int bit) {
this.index = index;
this.bit = bit;
this.mask = 1L << bit;
}
}
}