Skip to content

Commit

Permalink
[Skylark] Implement specialized RangeListView iterators.
Browse files Browse the repository at this point in the history
  • Loading branch information
ttsugriy committed May 24, 2018
1 parent acc3e60 commit e869eec
Show file tree
Hide file tree
Showing 2 changed files with 85 additions and 1 deletion.
Original file line number Diff line number Diff line change
Expand Up @@ -18,6 +18,7 @@
import com.google.common.collect.ImmutableList;
import com.google.common.collect.Iterables;
import com.google.common.collect.Lists;
import com.google.common.collect.UnmodifiableIterator;
import com.google.devtools.build.lib.events.Location;
import com.google.devtools.build.lib.skylarkinterface.Param;
import com.google.devtools.build.lib.skylarkinterface.SkylarkCallable;
Expand All @@ -28,7 +29,9 @@
import java.util.AbstractList;
import java.util.ArrayList;
import java.util.Collections;
import java.util.Iterator;
import java.util.List;
import java.util.NoSuchElementException;
import java.util.RandomAccess;
import javax.annotation.Nullable;

Expand Down Expand Up @@ -226,6 +229,65 @@ public static final class RangeList extends SkylarkList<Integer> implements

/** Provides access to range elements based on their index. */
private static class RangeListView extends AbstractList<Integer> {

/** Iterator for increasing sequences. */
private static class IncreasingIterator extends UnmodifiableIterator<Integer> {
private final int stop;
private final int step;

private int cursor;

private IncreasingIterator(int start, int stop, int step) {
this.cursor = start;
this.stop = stop;
this.step = step;
}

@Override
public boolean hasNext() {
return cursor < stop;
}

@Override
public Integer next() {
if (!hasNext()) {
throw new NoSuchElementException();
}
int current = cursor;
cursor += step;
return current;
}
}

/** Iterator for decreasing sequences. */
private static class DecreasingIterator extends UnmodifiableIterator<Integer> {
private final int stop;
private final int step;

private int cursor;

private DecreasingIterator(int start, int stop, int step) {
this.cursor = start;
this.stop = stop;
this.step = step;
}

@Override
public boolean hasNext() {
return cursor > stop;
}

@Override
public Integer next() {
if (!hasNext()) {
throw new NoSuchElementException();
}
int current = cursor;
cursor += step;
return current;
}
}

private static int computeSize(int start, int stop, int step) {
final int length = Math.abs(stop - start);
final int absolute_step = Math.abs(step);
Expand Down Expand Up @@ -258,6 +320,19 @@ public Integer get(int index) {
public int size() {
return size;
}

/**
* Returns an iterator optimized for traversing range elements, since it's the most frequent
* operation for which ranges are used.
*/
@Override
public Iterator<Integer> iterator() {
if (step > 0) {
return new IncreasingIterator(start, stop, step);
} else {
return new DecreasingIterator(start, stop, step);
}
}
}

private final AbstractList<Integer> contents;
Expand All @@ -279,6 +354,7 @@ public ImmutableList<Integer> getImmutableList() {
@Override
public SkylarkList<Integer> getSlice(Object start, Object end, Object step, Location loc,
Mutability mutability) throws EvalException {
// TODO: use lazy slice implementation
List<Integer> sliceIndices = EvalUtils.getSliceIndices(start, end, step, this.size(), loc);
ImmutableList.Builder<Integer> builder = ImmutableList.builderWithExpectedSize(sliceIndices.size());
for (int pos : sliceIndices) {
Expand Down Expand Up @@ -317,6 +393,7 @@ public Tuple<Integer> toTuple() {
}

public static RangeList of(int start, int stop, int step) {
Preconditions.checkArgument(step != 0);
return new RangeList(start, stop, step);
}
}
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -486,7 +486,14 @@ public void testRange() throws Exception {
.testStatement("str(range(5, 0, -1))", "[5, 4, 3, 2, 1]")
.testStatement("str(range(5, 0, -10))", "[5]")
.testStatement("str(range(0, -3, -2))", "[0, -2]")
.testStatement("str(range(3)[1:2])", "[1]")
.testStatement("str(range(5)[1:])", "[1, 2, 3, 4]")
.testStatement("len(range(5)[1:])", 4)
.testStatement("str(range(5)[:2])", "[0, 1]")
.testStatement("str(range(10)[1:9:2])", "[1, 3, 5, 7]")
.testStatement("str(range(10)[1:10:2])", "[1, 3, 5, 7, 9]")
.testStatement("str(range(10)[1:11:2])", "[1, 3, 5, 7, 9]")
.testStatement("str(range(0, 10, 2)[::2])", "[0, 4, 8]")
.testStatement("str(range(0, 10, 2)[::-2])", "[8, 4, 0]")
.testIfErrorContains("step cannot be 0", "range(2, 3, 0)");
}

Expand Down

0 comments on commit e869eec

Please sign in to comment.