blob: ad06c74c75625492966576cebe65d0a91d7da644 [file] [log] [blame]
// Copyright 2014 The Bazel Authors. All rights reserved.
//
// 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 com.google.devtools.build.lib.syntax;
import com.google.common.collect.Interner;
import com.google.devtools.build.lib.concurrent.BlazeInterners;
import com.google.devtools.build.lib.concurrent.ThreadSafety.Immutable;
import com.google.devtools.build.lib.events.Location;
import com.google.devtools.build.lib.skyframe.serialization.autocodec.AutoCodec;
import com.google.devtools.build.lib.util.Pair;
import java.util.Arrays;
import java.util.Objects;
/**
* A table to keep track of line numbers in source files. The client creates a LineNumberTable for
* their buffer using {@link #create}. The client can then ask for the line and column given a
* position using ({@link #getLineAndColumn(int)}).
*/
@AutoCodec
@Immutable
final class LineNumberTable {
private static final Interner<LineNumberTable> LINE_NUMBER_TABLE_INTERNER =
BlazeInterners.newWeakInterner();
/** A mapping from line number (line >= 1) to character offset into the file. */
private final int[] linestart;
private final String file;
private final int bufferLength;
private LineNumberTable(char[] buffer, String file) {
this(computeLinestart(buffer), file, buffer.length);
}
private LineNumberTable(int[] linestart, String file, int bufferLength) {
this.linestart = linestart;
this.file = file;
this.bufferLength = bufferLength;
}
@AutoCodec.Instantiator
static LineNumberTable createForSerialization(int[] linestart, String file, int bufferLength) {
return LINE_NUMBER_TABLE_INTERNER.intern(new LineNumberTable(linestart, file, bufferLength));
}
static LineNumberTable create(char[] buffer, String file) {
return new LineNumberTable(buffer, file);
}
private int getLineAt(int offset) {
if (offset < 0) {
throw new IllegalStateException("Illegal position: " + offset);
}
int lowBoundary = 1;
int highBoundary = linestart.length - 1;
while (true) {
if ((highBoundary - lowBoundary) <= 1) {
if (linestart[highBoundary] > offset) {
return lowBoundary;
} else {
return highBoundary;
}
}
int medium = lowBoundary + ((highBoundary - lowBoundary) >> 1);
if (linestart[medium] > offset) {
highBoundary = medium;
} else {
lowBoundary = medium;
}
}
}
Location.LineAndColumn getLineAndColumn(int offset) {
int line = getLineAt(offset);
int column = offset - linestart[line] + 1;
return new Location.LineAndColumn(line, column);
}
String getFile() {
return file;
}
Pair<Integer, Integer> getOffsetsForLine(int line) {
if (line <= 0 || line >= linestart.length) {
throw new IllegalArgumentException("Illegal line: " + line);
}
return Pair.of(
linestart[line], line < linestart.length - 1 ? linestart[line + 1] : bufferLength);
}
@Override
public int hashCode() {
return Objects.hash(Arrays.hashCode(linestart), file, bufferLength);
}
@Override
public boolean equals(Object other) {
if (!(other instanceof LineNumberTable)) {
return false;
}
LineNumberTable that = (LineNumberTable) other;
return this.bufferLength == that.bufferLength
&& Arrays.equals(this.linestart, that.linestart)
&& this.file.equals(that.file);
}
private static int[] computeLinestart(char[] buffer) {
// Compute the size.
int size = 2;
for (int i = 0; i < buffer.length; i++) {
if (buffer[i] == '\n') {
size++;
}
}
int[] linestart = new int[size];
int index = 0;
linestart[index++] = 0; // The 0th line does not exist - so we fill something in
// to make sure the start pos for the 1st line ends up at
// linestart[1]. Using 0 is useful for tables that are
// completely empty.
linestart[index++] = 0; // The first line ("line 1") starts at offset 0.
// Scan the buffer and record the offset of each line start. Doing this
// once upfront is faster than checking each char as it is pulled from
// the buffer.
for (int i = 0; i < buffer.length; i++) {
if (buffer[i] == '\n') {
linestart[index++] = i + 1;
}
}
return linestart;
}
}