Hopefully speed up BlockOptionalMeta

This commit is contained in:
Logan Darklock 2019-08-31 10:28:01 -07:00
parent 84f45ebb48
commit 6ed86cb3c7
No known key found for this signature in database
GPG Key ID: B8C37CEDE1AC60EA
3 changed files with 79 additions and 1 deletions

View File

@ -17,16 +17,21 @@
package baritone.api.utils; package baritone.api.utils;
import com.google.common.collect.ImmutableSet;
import net.minecraft.block.Block; import net.minecraft.block.Block;
import net.minecraft.block.state.IBlockState; import net.minecraft.block.state.IBlockState;
import net.minecraft.item.ItemStack; import net.minecraft.item.ItemStack;
import net.minecraft.util.ResourceLocation; import net.minecraft.util.ResourceLocation;
import net.minecraft.util.Rotation;
import javax.annotation.Nonnull; import javax.annotation.Nonnull;
import javax.annotation.Nullable; import javax.annotation.Nullable;
import java.util.ArrayList;
import java.util.List;
import java.util.regex.MatchResult; import java.util.regex.MatchResult;
import java.util.regex.Matcher; import java.util.regex.Matcher;
import java.util.regex.Pattern; import java.util.regex.Pattern;
import java.util.stream.Collectors;
import static java.util.Objects.isNull; import static java.util.Objects.isNull;
@ -34,12 +39,25 @@ public final class BlockOptionalMeta {
private final Block block; private final Block block;
private final int meta; private final int meta;
private final boolean noMeta; private final boolean noMeta;
private final ImmutableSet<Integer> stateMetas;
private static final Pattern pattern = Pattern.compile("^(.+?)(?::(\\d+))?$"); private static final Pattern pattern = Pattern.compile("^(.+?)(?::(\\d+))?$");
private static ImmutableSet<Integer> getStateMetas(@Nonnull Block block, @Nullable Integer meta) {
List<IBlockState> blockstates = block.getBlockState().getValidStates();
return ImmutableSet.copyOf(
(ArrayList<Integer>) blockstates.stream()
.filter(blockstate -> meta == null || block.getMetaFromState(blockstate.withRotation(Rotation.NONE)) == meta)
.map(IBlockState::hashCode)
.collect(Collectors.toCollection(ArrayList::new))
);
}
public BlockOptionalMeta(@Nonnull Block block, @Nullable Integer meta) { public BlockOptionalMeta(@Nonnull Block block, @Nullable Integer meta) {
this.block = block; this.block = block;
this.noMeta = isNull(meta); this.noMeta = isNull(meta);
this.meta = noMeta ? 0 : meta; this.meta = noMeta ? 0 : meta;
this.stateMetas = getStateMetas(block, meta);
} }
public BlockOptionalMeta(@Nonnull Block block) { public BlockOptionalMeta(@Nonnull Block block) {
@ -64,6 +82,7 @@ public final class BlockOptionalMeta {
block = Block.REGISTRY.getObject(id); block = Block.REGISTRY.getObject(id);
meta = noMeta ? 0 : Integer.parseInt(matchResult.group(2)); meta = noMeta ? 0 : Integer.parseInt(matchResult.group(2));
stateMetas = getStateMetas(block, noMeta ? null : meta);
} }
public Block getBlock() { public Block getBlock() {
@ -80,7 +99,7 @@ public final class BlockOptionalMeta {
public boolean matches(@Nonnull IBlockState blockstate) { public boolean matches(@Nonnull IBlockState blockstate) {
Block block = blockstate.getBlock(); Block block = blockstate.getBlock();
return block == this.block && (noMeta || block.damageDropped(blockstate) == this.meta); return block == this.block && stateMetas.contains(blockstate.hashCode());
} }
@Override @Override

View File

@ -0,0 +1,58 @@
/*
* This file is part of Baritone.
*
* Baritone is free software: you can redistribute it and/or modify
* it under the terms of the GNU Lesser General Public License as published by
* the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*
* Baritone is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU Lesser General Public License for more details.
*
* You should have received a copy of the GNU Lesser General Public License
* along with Baritone. If not, see <https://www.gnu.org/licenses/>.
*/
package baritone.launch.mixins;
import com.google.common.collect.ImmutableMap;
import net.minecraft.block.properties.IProperty;
import org.spongepowered.asm.mixin.Final;
import org.spongepowered.asm.mixin.Mixin;
import org.spongepowered.asm.mixin.Overwrite;
import org.spongepowered.asm.mixin.Shadow;
import org.spongepowered.asm.mixin.Unique;
import org.spongepowered.asm.mixin.injection.At;
import org.spongepowered.asm.mixin.injection.Inject;
import org.spongepowered.asm.mixin.injection.callback.CallbackInfo;
@Mixin(targets = "net.minecraft.block.state.BlockStateContainer$StateImplementation")
public abstract class MixinStateImplementation {
@Shadow
@Final
private ImmutableMap<IProperty<?>, Comparable<?>> properties;
/**
* Block states are fucking immutable
*/
@Unique
private int hashCode;
@Inject(method = "<init>*", at = @At("RETURN"))
private void onInit(CallbackInfo ci) {
hashCode = properties.hashCode();
}
/**
* Cache this instead of using the fucking map every time
*
* @author LoganDark
*/
@Override
@Overwrite
public int hashCode() {
return hashCode;
}
}

View File

@ -27,6 +27,7 @@
"MixinNetworkManager", "MixinNetworkManager",
"MixinRenderChunk", "MixinRenderChunk",
"MixinRenderList", "MixinRenderList",
"MixinStateImplementation",
"MixinTabCompleter", "MixinTabCompleter",
"MixinVboRenderList", "MixinVboRenderList",
"MixinWorldClient" "MixinWorldClient"