mirror of
https://github.com/Jozufozu/Flywheel.git
synced 2025-01-12 23:36:09 +01:00
Always the first thing to fall apart
- Fix crumbling on indirect
This commit is contained in:
parent
eae1d0ef94
commit
68453d8349
4 changed files with 30 additions and 10 deletions
|
@ -8,6 +8,8 @@ import java.util.Queue;
|
|||
import java.util.concurrent.ConcurrentHashMap;
|
||||
import java.util.concurrent.ConcurrentLinkedQueue;
|
||||
|
||||
import org.jetbrains.annotations.Nullable;
|
||||
|
||||
import com.mojang.datafixers.util.Pair;
|
||||
|
||||
import dev.engine_room.flywheel.api.RenderContext;
|
||||
|
@ -106,7 +108,13 @@ public abstract class DrawManager<N extends AbstractInstancer<?>> {
|
|||
return false;
|
||||
}
|
||||
|
||||
protected static <I extends AbstractInstancer<?>> Map<GroupKey<?>, Int2ObjectMap<List<Pair<I, InstanceHandleImpl<?>>>>> doCrumblingSort(Class<I> clazz, List<Engine.CrumblingBlock> crumblingBlocks) {
|
||||
@FunctionalInterface
|
||||
protected interface State2Instancer<I extends AbstractInstancer<?>> {
|
||||
// I tried using a plain Function<State<?>, I> here, but it exploded with type errors.
|
||||
@Nullable I apply(InstanceHandleImpl.State<?> state);
|
||||
}
|
||||
|
||||
protected static <I extends AbstractInstancer<?>> Map<GroupKey<?>, Int2ObjectMap<List<Pair<I, InstanceHandleImpl<?>>>>> doCrumblingSort(List<Engine.CrumblingBlock> crumblingBlocks, State2Instancer<I> cast) {
|
||||
Map<GroupKey<?>, Int2ObjectMap<List<Pair<I, InstanceHandleImpl<?>>>>> byType = new HashMap<>();
|
||||
for (Engine.CrumblingBlock block : crumblingBlocks) {
|
||||
int progress = block.progress();
|
||||
|
@ -123,16 +131,12 @@ public abstract class DrawManager<N extends AbstractInstancer<?>> {
|
|||
continue;
|
||||
}
|
||||
|
||||
InstanceHandleImpl.State<?> abstractInstancer = impl.state;
|
||||
// AbstractInstancer directly implement HandleState, so this check is valid.
|
||||
if (!clazz.isInstance(abstractInstancer)) {
|
||||
// This rejects instances that were created by a different engine,
|
||||
// and also instances that are hidden or deleted.
|
||||
var instancer = cast.apply(impl.state);
|
||||
|
||||
if (instancer == null) {
|
||||
continue;
|
||||
}
|
||||
|
||||
var instancer = clazz.cast(abstractInstancer);
|
||||
|
||||
byType.computeIfAbsent(new GroupKey<>(instancer.type, instancer.environment), $ -> new Int2ObjectArrayMap<>())
|
||||
.computeIfAbsent(progress, $ -> new ArrayList<>())
|
||||
.add(Pair.of(instancer, impl));
|
||||
|
|
|
@ -189,7 +189,7 @@ public class IndirectDrawManager extends DrawManager<IndirectInstancer<?>> {
|
|||
}
|
||||
|
||||
public void renderCrumbling(List<Engine.CrumblingBlock> crumblingBlocks) {
|
||||
var byType = doCrumblingSort(IndirectInstancer.class, crumblingBlocks);
|
||||
var byType = doCrumblingSort(crumblingBlocks, IndirectInstancer::fromState);
|
||||
|
||||
if (byType.isEmpty()) {
|
||||
return;
|
||||
|
|
|
@ -74,6 +74,14 @@ public class IndirectInstancer<I extends Instance> extends AbstractInstancer<I>
|
|||
return new InstanceHandleImpl[ObjectStorage.PAGE_SIZE];
|
||||
}
|
||||
|
||||
@Nullable
|
||||
public static IndirectInstancer<?> fromState(InstanceHandleImpl.State<?> handle) {
|
||||
if (handle instanceof InstancePage<?> instancer) {
|
||||
return instancer.parent;
|
||||
}
|
||||
return null;
|
||||
}
|
||||
|
||||
private static final class InstancePage<I extends Instance> implements InstanceHandleImpl.State<I> {
|
||||
private final IndirectInstancer<I> parent;
|
||||
private final int pageNo;
|
||||
|
|
|
@ -147,7 +147,15 @@ public class InstancedDrawManager extends DrawManager<InstancedInstancer<?>> {
|
|||
@Override
|
||||
public void renderCrumbling(List<Engine.CrumblingBlock> crumblingBlocks) {
|
||||
// Sort draw calls into buckets, so we don't have to do as many shader binds.
|
||||
var byType = doCrumblingSort(InstancedInstancer.class, crumblingBlocks);
|
||||
var byType = doCrumblingSort(crumblingBlocks, handle -> {
|
||||
// AbstractInstancer directly implement HandleState, so this check is valid.
|
||||
if (handle instanceof InstancedInstancer<?> instancer) {
|
||||
return instancer;
|
||||
}
|
||||
// This rejects instances that were created by a different engine,
|
||||
// and also instances that are hidden or deleted.
|
||||
return null;
|
||||
});
|
||||
|
||||
if (byType.isEmpty()) {
|
||||
return;
|
||||
|
|
Loading…
Reference in a new issue