|  | // 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.LineAndColumn; | 
|  | import com.google.devtools.build.lib.skyframe.serialization.autocodec.AutoCodec; | 
|  | import com.google.devtools.build.lib.util.Pair; | 
|  | import com.google.devtools.build.lib.vfs.PathFragment; | 
|  | 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 | 
|  | 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 PathFragment path; | 
|  | private final int bufferLength; | 
|  |  | 
|  | private LineNumberTable(char[] buffer, PathFragment path) { | 
|  | this(computeLinestart(buffer), path, buffer.length); | 
|  | } | 
|  |  | 
|  | private LineNumberTable(int[] linestart, PathFragment path, int bufferLength) { | 
|  | this.linestart = linestart; | 
|  | this.path = path; | 
|  | this.bufferLength = bufferLength; | 
|  | } | 
|  |  | 
|  | @AutoCodec.Instantiator | 
|  | static LineNumberTable createForSerialization( | 
|  | int[] linestart, PathFragment path, int bufferLength) { | 
|  | return LINE_NUMBER_TABLE_INTERNER.intern(new LineNumberTable(linestart, path, bufferLength)); | 
|  | } | 
|  |  | 
|  | static LineNumberTable create(char[] buffer, PathFragment path) { | 
|  | return new LineNumberTable(buffer, path); | 
|  | } | 
|  |  | 
|  | 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; | 
|  | } | 
|  | } | 
|  | } | 
|  |  | 
|  | LineAndColumn getLineAndColumn(int offset) { | 
|  | int line = getLineAt(offset); | 
|  | int column = offset - linestart[line] + 1; | 
|  | return new LineAndColumn(line, column); | 
|  | } | 
|  |  | 
|  | PathFragment getPath(int offset) { | 
|  | return path; | 
|  | } | 
|  |  | 
|  | 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), path, 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) | 
|  | && Objects.equals(this.path, that.path); | 
|  | } | 
|  |  | 
|  | 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; | 
|  | } | 
|  | } |