1
0
mirror of https://github.com/SquidDev-CC/CC-Tweaked synced 2024-12-15 12:40:30 +00:00

Always expose nbt from turtle.getItemDetail

- Document the thread safety of DetailRegistry a little better.

 - Turtles now duplicate their inventory to the "previous
   inventory" (now called inventorySnapshot) immediately, rather than
   when the block is ticked.

   This is slightly more resource intensive, but I don't think it's so
   bad we need to worry.

 - As this snapshot is now always up-to-date, we can read it from the
   computer thread. Given the item is immutable, it's safe to read NBT
   from it.

   _Technically_ this is not safe under the Java memory model, but in
   practice I don't think we'll observe the wrong value.

Closes #1306
This commit is contained in:
Jonathan Coates 2023-01-25 18:37:14 +00:00
parent 6cd32a6368
commit 1554c7b397
No known key found for this signature in database
GPG Key ID: B9E431FF07C98D06
8 changed files with 80 additions and 46 deletions

View File

@ -32,6 +32,8 @@ public interface DetailRegistry<T> {
/** /**
* Compute basic details about an object. This is cheaper than computing all details operation, and so is suitable * Compute basic details about an object. This is cheaper than computing all details operation, and so is suitable
* for when you need to compute the details for a large number of values. * for when you need to compute the details for a large number of values.
* <p>
* This method <em>MAY</em> be thread safe: consult the instance's documentation for details.
* *
* @param object The object to get details for. * @param object The object to get details for.
* @return The basic details. * @return The basic details.
@ -40,6 +42,8 @@ public interface DetailRegistry<T> {
/** /**
* Compute all details about an object, using {@link #getBasicDetails(Object)} and any registered providers. * Compute all details about an object, using {@link #getBasicDetails(Object)} and any registered providers.
* <p>
* This method is <em>NOT</em> thread safe. It should only be called from the computer thread.
* *
* @param object The object to get details for. * @param object The object to get details for.
* @return The computed details. * @return The computed details.

View File

@ -15,12 +15,17 @@ import net.minecraft.world.level.block.Block;
public class VanillaDetailRegistries { public class VanillaDetailRegistries {
/** /**
* Provides details for {@link ItemStack}s. * Provides details for {@link ItemStack}s.
* <p>
* This instance's {@link DetailRegistry#getBasicDetails(Object)} is thread safe (assuming the stack is immutable)
* and may be called from the computer thread.
*/ */
public static final DetailRegistry<ItemStack> ITEM_STACK = ComputerCraftAPIService.get().getItemStackDetailRegistry(); public static final DetailRegistry<ItemStack> ITEM_STACK = ComputerCraftAPIService.get().getItemStackDetailRegistry();
/** /**
* Provides details for {@link BlockReference}, a reference to a {@link Block} in the world. * Provides details for {@link BlockReference}, a reference to a {@link Block} in the world.
* <p>
* This instance's {@link DetailRegistry#getBasicDetails(Object)} is thread safe and may be called from the computer
* thread.
*/ */
public static final DetailRegistry<BlockReference> BLOCK_IN_WORLD = ComputerCraftAPIService.get().getBlockInWorldDetailRegistry(); public static final DetailRegistry<BlockReference> BLOCK_IN_WORLD = ComputerCraftAPIService.get().getBlockInWorldDetailRegistry();
} }

View File

@ -15,6 +15,7 @@ import net.minecraft.nbt.CompoundTag;
import net.minecraft.world.Container; import net.minecraft.world.Container;
import net.minecraft.world.level.Level; import net.minecraft.world.level.Level;
import net.minecraft.world.phys.Vec3; import net.minecraft.world.phys.Vec3;
import org.jetbrains.annotations.ApiStatus;
import javax.annotation.Nullable; import javax.annotation.Nullable;
@ -24,6 +25,7 @@ import javax.annotation.Nullable;
* This should not be implemented by your classes. Do not interact with turtles except via this interface and * This should not be implemented by your classes. Do not interact with turtles except via this interface and
* {@link ITurtleUpgrade}. * {@link ITurtleUpgrade}.
*/ */
@ApiStatus.NonExtendable
public interface ITurtleAccess { public interface ITurtleAccess {
/** /**
* Returns the world in which the turtle resides. * Returns the world in which the turtle resides.

View File

@ -25,14 +25,9 @@ import java.util.*;
* Data providers for items. * Data providers for items.
*/ */
public class ItemDetails { public class ItemDetails {
public static <T extends Map<? super String, Object>> T fillBasicSafe(T data, ItemStack stack) { public static void fillBasic(Map<? super String, Object> data, ItemStack stack) {
data.put("name", DetailHelpers.getId(RegistryWrappers.ITEMS, stack.getItem())); data.put("name", DetailHelpers.getId(RegistryWrappers.ITEMS, stack.getItem()));
data.put("count", stack.getCount()); data.put("count", stack.getCount());
return data;
}
public static void fillBasic(Map<? super String, Object> data, ItemStack stack) {
fillBasicSafe(data, stack);
var hash = NBTUtil.getNBTHash(stack.getTag()); var hash = NBTUtil.getNBTHash(stack.getTag());
if (hash != null) data.put("nbt", hash); if (hash != null) data.put("nbt", hash);
} }

View File

@ -7,16 +7,13 @@ package dan200.computercraft.shared.turtle.apis;
import dan200.computercraft.api.detail.VanillaDetailRegistries; import dan200.computercraft.api.detail.VanillaDetailRegistries;
import dan200.computercraft.api.lua.*; import dan200.computercraft.api.lua.*;
import dan200.computercraft.api.turtle.ITurtleAccess;
import dan200.computercraft.api.turtle.TurtleCommand; import dan200.computercraft.api.turtle.TurtleCommand;
import dan200.computercraft.api.turtle.TurtleCommandResult; import dan200.computercraft.api.turtle.TurtleCommandResult;
import dan200.computercraft.api.turtle.TurtleSide; import dan200.computercraft.api.turtle.TurtleSide;
import dan200.computercraft.core.apis.IAPIEnvironment; import dan200.computercraft.core.apis.IAPIEnvironment;
import dan200.computercraft.core.metrics.Metrics; import dan200.computercraft.core.metrics.Metrics;
import dan200.computercraft.shared.details.ItemDetails;
import dan200.computercraft.shared.turtle.core.*; import dan200.computercraft.shared.turtle.core.*;
import java.util.HashMap;
import java.util.Optional; import java.util.Optional;
/** /**
@ -70,9 +67,9 @@ import java.util.Optional;
*/ */
public class TurtleAPI implements ILuaAPI { public class TurtleAPI implements ILuaAPI {
private final IAPIEnvironment environment; private final IAPIEnvironment environment;
private final ITurtleAccess turtle; private final TurtleAccessInternal turtle;
public TurtleAPI(IAPIEnvironment environment, ITurtleAccess turtle) { public TurtleAPI(IAPIEnvironment environment, TurtleAccessInternal turtle) {
this.environment = environment; this.environment = environment;
this.turtle = turtle; this.turtle = turtle;
} }
@ -760,20 +757,15 @@ public class TurtleAPI implements ILuaAPI {
@LuaFunction @LuaFunction
public final MethodResult getItemDetail(ILuaContext context, Optional<Integer> slot, Optional<Boolean> detailed) throws LuaException { public final MethodResult getItemDetail(ILuaContext context, Optional<Integer> slot, Optional<Boolean> detailed) throws LuaException {
int actualSlot = checkSlot(slot).orElse(turtle.getSelectedSlot()); int actualSlot = checkSlot(slot).orElse(turtle.getSelectedSlot());
return detailed.orElse(false) if (detailed.orElse(false)) {
? context.executeMainThreadTask(() -> getItemDetail(actualSlot, true)) return context.executeMainThreadTask(() -> {
: MethodResult.of(getItemDetail(actualSlot, false)); var stack = turtle.getInventory().getItem(actualSlot);
} return new Object[]{ stack.isEmpty() ? null : VanillaDetailRegistries.ITEM_STACK.getDetails(stack) };
});
private Object[] getItemDetail(int slot, boolean detailed) { } else {
var stack = turtle.getInventory().getItem(slot); var stack = turtle.getItemSnapshot(actualSlot);
if (stack.isEmpty()) return new Object[]{ null }; return MethodResult.of(stack.isEmpty() ? null : VanillaDetailRegistries.ITEM_STACK.getBasicDetails(stack));
}
var table = detailed
? VanillaDetailRegistries.ITEM_STACK.getDetails(stack)
: ItemDetails.fillBasicSafe(new HashMap<>(), stack);
return new Object[]{ table };
} }

View File

@ -56,7 +56,7 @@ public class TurtleBlockEntity extends AbstractComputerBlockEntity implements Ba
} }
private final NonNullList<ItemStack> inventory = NonNullList.withSize(INVENTORY_SIZE, ItemStack.EMPTY); private final NonNullList<ItemStack> inventory = NonNullList.withSize(INVENTORY_SIZE, ItemStack.EMPTY);
private final NonNullList<ItemStack> previousInventory = NonNullList.withSize(INVENTORY_SIZE, ItemStack.EMPTY); private final NonNullList<ItemStack> inventorySnapshot = NonNullList.withSize(INVENTORY_SIZE, ItemStack.EMPTY);
private boolean inventoryChanged = false; private boolean inventoryChanged = false;
private TurtleBrain brain = new TurtleBrain(this); private TurtleBrain brain = new TurtleBrain(this);
private MoveState moveState = MoveState.NOT_MOVED; private MoveState moveState = MoveState.NOT_MOVED;
@ -78,7 +78,7 @@ public class TurtleBlockEntity extends AbstractComputerBlockEntity implements Ba
getFamily(), Config.turtleTermWidth, getFamily(), Config.turtleTermWidth,
Config.turtleTermHeight Config.turtleTermHeight
); );
computer.addAPI(new TurtleAPI(computer.getAPIEnvironment(), getAccess())); computer.addAPI(new TurtleAPI(computer.getAPIEnvironment(), brain));
brain.setupComputer(computer); brain.setupComputer(computer);
return computer; return computer;
} }
@ -141,11 +141,7 @@ public class TurtleBlockEntity extends AbstractComputerBlockEntity implements Ba
if (inventoryChanged) { if (inventoryChanged) {
var computer = getServerComputer(); var computer = getServerComputer();
if (computer != null) computer.queueEvent("turtle_inventory"); if (computer != null) computer.queueEvent("turtle_inventory");
inventoryChanged = false; inventoryChanged = false;
for (var n = 0; n < getContainerSize(); n++) {
previousInventory.set(n, getItem(n).copy());
}
} }
} }
@ -178,13 +174,13 @@ public class TurtleBlockEntity extends AbstractComputerBlockEntity implements Ba
// Read inventory // Read inventory
var nbttaglist = nbt.getList("Items", Tag.TAG_COMPOUND); var nbttaglist = nbt.getList("Items", Tag.TAG_COMPOUND);
inventory.clear(); inventory.clear();
previousInventory.clear(); inventorySnapshot.clear();
for (var i = 0; i < nbttaglist.size(); i++) { for (var i = 0; i < nbttaglist.size(); i++) {
var tag = nbttaglist.getCompound(i); var tag = nbttaglist.getCompound(i);
var slot = tag.getByte("Slot") & 0xff; var slot = tag.getByte("Slot") & 0xff;
if (slot < getContainerSize()) { if (slot < getContainerSize()) {
inventory.set(slot, ItemStack.of(tag)); inventory.set(slot, ItemStack.of(tag));
previousInventory.set(slot, inventory.get(slot).copy()); inventorySnapshot.set(slot, inventory.get(slot).copy());
} }
} }
@ -273,7 +269,7 @@ public class TurtleBlockEntity extends AbstractComputerBlockEntity implements Ba
void setOwningPlayer(GameProfile player) { void setOwningPlayer(GameProfile player) {
brain.setOwningPlayer(player); brain.setOwningPlayer(player);
setChanged(); onTileEntityChange();
} }
// IInventory // IInventory
@ -283,16 +279,20 @@ public class TurtleBlockEntity extends AbstractComputerBlockEntity implements Ba
return inventory; return inventory;
} }
public ItemStack getItemSnapshot(int slot) {
return slot >= 0 && slot < inventorySnapshot.size() ? inventorySnapshot.get(slot) : ItemStack.EMPTY;
}
@Override @Override
public void setChanged() { public void setChanged() {
super.setChanged(); super.setChanged();
if (!inventoryChanged) {
for (var n = 0; n < getContainerSize(); n++) { for (var slot = 0; slot < getContainerSize(); slot++) {
if (!ItemStack.matches(getItem(n), previousInventory.get(n))) { var item = getItem(slot);
inventoryChanged = true; if (ItemStack.matches(item, inventorySnapshot.get(slot))) continue;
break;
} inventoryChanged = true;
} inventorySnapshot.set(slot, item.copy());
} }
} }
@ -340,7 +340,7 @@ public class TurtleBlockEntity extends AbstractComputerBlockEntity implements Ba
public void transferStateFrom(TurtleBlockEntity copy) { public void transferStateFrom(TurtleBlockEntity copy) {
super.transferStateFrom(copy); super.transferStateFrom(copy);
Collections.copy(inventory, copy.inventory); Collections.copy(inventory, copy.inventory);
Collections.copy(previousInventory, copy.previousInventory); Collections.copy(inventorySnapshot, copy.inventorySnapshot);
inventoryChanged = copy.inventoryChanged; inventoryChanged = copy.inventoryChanged;
brain = copy.brain; brain = copy.brain;
brain.setOwner(this); brain.setOwner(this);

View File

@ -0,0 +1,27 @@
/*
* This file is part of ComputerCraft - http://www.computercraft.info
* Copyright Daniel Ratcliffe, 2011-2022. Do not distribute without permission.
* Send enquiries to dratcliffe@gmail.com
*/
package dan200.computercraft.shared.turtle.core;
import dan200.computercraft.api.turtle.ITurtleAccess;
import net.minecraft.world.item.ItemStack;
/**
* An internal version of {@link ITurtleAccess}.
* <p>
* This exposes additional functionality we don't want in the public API, but where we don't want access to the full
* {@link TurtleBrain} interface.
*/
public interface TurtleAccessInternal extends ITurtleAccess {
/**
* Get an immutable snapshot of an item in the inventory. This is a thread-safe version of
* {@code getInventory().getItem()}.
*
* @param slot The slot
* @return The current item. This should NOT be modified.
* @see net.minecraft.world.Container#getItem(int)
*/
ItemStack getItemSnapshot(int slot);
}

View File

@ -10,7 +10,10 @@ import com.mojang.authlib.GameProfile;
import dan200.computercraft.api.lua.ILuaCallback; import dan200.computercraft.api.lua.ILuaCallback;
import dan200.computercraft.api.lua.MethodResult; import dan200.computercraft.api.lua.MethodResult;
import dan200.computercraft.api.peripheral.IPeripheral; import dan200.computercraft.api.peripheral.IPeripheral;
import dan200.computercraft.api.turtle.*; import dan200.computercraft.api.turtle.ITurtleUpgrade;
import dan200.computercraft.api.turtle.TurtleAnimation;
import dan200.computercraft.api.turtle.TurtleCommand;
import dan200.computercraft.api.turtle.TurtleSide;
import dan200.computercraft.core.computer.ComputerSide; import dan200.computercraft.core.computer.ComputerSide;
import dan200.computercraft.core.util.Colour; import dan200.computercraft.core.util.Colour;
import dan200.computercraft.impl.TurtleUpgrades; import dan200.computercraft.impl.TurtleUpgrades;
@ -34,6 +37,7 @@ import net.minecraft.world.Container;
import net.minecraft.world.entity.Entity; import net.minecraft.world.entity.Entity;
import net.minecraft.world.entity.MoverType; import net.minecraft.world.entity.MoverType;
import net.minecraft.world.item.DyeColor; import net.minecraft.world.item.DyeColor;
import net.minecraft.world.item.ItemStack;
import net.minecraft.world.level.Level; import net.minecraft.world.level.Level;
import net.minecraft.world.level.material.PushReaction; import net.minecraft.world.level.material.PushReaction;
import net.minecraft.world.phys.AABB; import net.minecraft.world.phys.AABB;
@ -47,7 +51,7 @@ import java.util.function.Predicate;
import static dan200.computercraft.shared.common.IColouredItem.NBT_COLOUR; import static dan200.computercraft.shared.common.IColouredItem.NBT_COLOUR;
import static dan200.computercraft.shared.util.WaterloggableHelpers.WATERLOGGED; import static dan200.computercraft.shared.util.WaterloggableHelpers.WATERLOGGED;
public class TurtleBrain implements ITurtleAccess { public class TurtleBrain implements TurtleAccessInternal {
public static final String NBT_RIGHT_UPGRADE = "RightUpgrade"; public static final String NBT_RIGHT_UPGRADE = "RightUpgrade";
public static final String NBT_RIGHT_UPGRADE_DATA = "RightUpgradeNbt"; public static final String NBT_RIGHT_UPGRADE_DATA = "RightUpgradeNbt";
public static final String NBT_LEFT_UPGRADE = "LeftUpgrade"; public static final String NBT_LEFT_UPGRADE = "LeftUpgrade";
@ -456,7 +460,7 @@ public class TurtleBrain implements ITurtleAccess {
return overlay; return overlay;
} }
public void setOverlay(ResourceLocation overlay) { public void setOverlay(@Nullable ResourceLocation overlay) {
if (!Objects.equal(this.overlay, overlay)) { if (!Objects.equal(this.overlay, overlay)) {
this.overlay = overlay; this.overlay = overlay;
BlockEntityHelpers.updateBlock(owner); BlockEntityHelpers.updateBlock(owner);
@ -768,6 +772,11 @@ public class TurtleBrain implements ITurtleAccess {
return previous + (next - previous) * f; return previous + (next - previous) * f;
} }
@Override
public ItemStack getItemSnapshot(int slot) {
return owner.getItemSnapshot(slot);
}
private static final class CommandCallback implements ILuaCallback { private static final class CommandCallback implements ILuaCallback {
final MethodResult pull = MethodResult.pullEvent("turtle_response", this); final MethodResult pull = MethodResult.pullEvent("turtle_response", this);
private final int command; private final int command;