|  | // 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; | 
|  | } | 
|  | } |