blob: 074d6e80e8426d2411e47f20e06c7c35d91634b4 [file] [log] [blame]
// Copyright 2023 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.packages;
import com.google.common.base.Joiner;
import com.google.common.base.Preconditions;
import com.google.common.collect.ImmutableCollection;
import com.google.common.collect.ImmutableList;
import com.google.devtools.build.lib.skyframe.serialization.VisibleForSerialization;
import java.util.ArrayList;
import java.util.Collections;
import java.util.List;
import javax.annotation.Nullable;
import net.starlark.java.eval.EvalException;
import net.starlark.java.eval.Starlark;
import net.starlark.java.eval.StarlarkList;
import net.starlark.java.syntax.Location;
import net.starlark.java.syntax.TokenKind;
/**
* A struct-like Info (provider instance) for providers defined in Starlark that have a schema.
*
* <p>Maintainer's note: This class is memory-optimized in a way that can cause profiling
* instability in some pathological cases. See {@link StarlarkProvider#optimizeField} for more
* information.
*/
public class StarlarkInfoWithSchema extends StarlarkInfo {
private final StarlarkProvider provider;
// For each field in provider.getFields the table contains on corresponding position either null,
// a legal Starlark value, or an optimized value (see StarlarkProvider#optimizeField).
private final Object[] table;
// `table` elements should already be optimized by caller, see StarlarkProvider#optimizeField
@VisibleForSerialization // private
StarlarkInfoWithSchema(StarlarkProvider provider, Object[] table, @Nullable Location loc) {
super(loc);
this.provider = provider;
this.table = table;
}
@Override
public Provider getProvider() {
return provider;
}
@VisibleForSerialization // private
Object[] getTable() {
return table;
}
/**
* Constructs a StarlarkInfo from an array of alternating key/value pairs as provided by
* Starlark.fastcall. Checks that each key is provided at most once, and is defined by the schema,
* which must be sorted. This function exists solely for the StarlarkProvider constructor.
*/
static StarlarkInfoWithSchema createFromNamedArgs(
StarlarkProvider provider, Object[] table, Location loc) throws EvalException {
ImmutableList<String> fields = provider.getFields();
Object[] valueTable = new Object[fields.size()];
List<String> unexpected = null;
for (int i = 0; i < table.length; i += 2) {
int pos = Collections.binarySearch(fields, (String) table[i]);
if (pos >= 0) {
if (valueTable[pos] != null) {
throw Starlark.errorf(
"got multiple values for parameter %s in call to instantiate provider %s",
table[i], provider.getPrintableName());
}
valueTable[pos] = provider.optimizeField(pos, table[i + 1]);
} else {
if (unexpected == null) {
unexpected = new ArrayList<>();
}
unexpected.add((String) table[i]);
}
}
if (unexpected != null) {
throw Starlark.errorf(
"got unexpected field%s '%s' in call to instantiate provider %s",
unexpected.size() > 1 ? "s" : "",
Joiner.on("', '").join(unexpected),
provider.getPrintableName());
}
return new StarlarkInfoWithSchema(provider, valueTable, loc);
}
@Override
public ImmutableCollection<String> getFieldNames() {
ImmutableList.Builder<String> fieldNames = new ImmutableList.Builder<>();
ImmutableList<String> fields = provider.getFields();
for (int i = 0; i < fields.size(); i++) {
if (table[i] != null) {
fieldNames.add(fields.get(i));
}
}
return fieldNames.build();
}
@Override
public boolean isImmutable() {
// If the provider is not yet exported, the hash code of the object is subject to change.
if (!provider.isExported()) {
return false;
}
for (int i = 0; i < table.length; i++) {
if (table[i] != null
&& !(provider.isOptimised(i, table[i]) // optimised fields might not be Starlark values
|| Starlark.isImmutable(table[i]))) {
return false;
}
}
return true;
}
@Nullable
@Override
public Object getValue(String name) {
ImmutableList<String> fields = provider.getFields();
int i = Collections.binarySearch(fields, name);
return i >= 0 ? provider.retrieveOptimizedField(i, table[i]) : null;
}
@Nullable
@Override
public StarlarkInfoWithSchema binaryOp(TokenKind op, Object that, boolean thisLeft)
throws EvalException {
if (op == TokenKind.PLUS && that instanceof StarlarkInfo) {
final Provider thatProvider = ((StarlarkInfo) that).getProvider();
if (!provider.equals(thatProvider)) {
throw Starlark.errorf(
"Cannot use '+' operator on instances of different providers (%s and %s)",
provider.getPrintableName(), thatProvider.getPrintableName());
}
Preconditions.checkArgument(that instanceof StarlarkInfoWithSchema);
return thisLeft
? plus(this, (StarlarkInfoWithSchema) that) //
: plus((StarlarkInfoWithSchema) that, this);
}
return null;
}
private static StarlarkInfoWithSchema plus(StarlarkInfoWithSchema x, StarlarkInfoWithSchema y)
throws EvalException {
int n = x.table.length;
Object[] ztable = new Object[n];
for (int i = 0; i < n; i++) {
if (x.table[i] != null && y.table[i] != null) {
ImmutableList<String> schema = x.provider.getFields();
throw Starlark.errorf("cannot add struct instances with common field '%s'", schema.get(i));
}
ztable[i] = x.table[i] != null ? x.table[i] : y.table[i];
}
return new StarlarkInfoWithSchema(x.provider, ztable, Location.BUILTIN);
}
@Override
public StarlarkInfoWithSchema unsafeOptimizeMemoryLayout() {
int n = table.length;
for (int i = 0; i < n; i++) {
if (table[i] instanceof StarlarkList<?>) {
// On duplicated lists, ImmutableStarlarkLists are duplicated, but not underlying Object
// arrays
table[i] = ((StarlarkList<?>) table[i]).unsafeOptimizeMemoryLayout();
} else if (table[i] instanceof StarlarkInfo) {
table[i] = ((StarlarkInfo) table[i]).unsafeOptimizeMemoryLayout();
}
}
return this;
}
}