From 20b3f78b9cadbf5e3c0eb455b0dcb142c81ce1fe Mon Sep 17 00:00:00 2001 From: Jozufozu Date: Tue, 29 Oct 2024 20:01:20 -0700 Subject: [PATCH] A real page turner - Try to shuffle over instances into pages with space - Clear out now-unused logic from ObjectStorage - Some cleanup and more comments in IndirectInstancer --- .../backend/engine/AbstractInstancer.java | 2 +- .../flywheel/backend/engine/DrawManager.java | 2 +- .../engine/indirect/IndirectInstancer.java | 143 ++++++++++++++++-- .../engine/indirect/ObjectStorage.java | 85 ----------- .../engine/instancing/InstancedInstancer.java | 2 +- 5 files changed, 135 insertions(+), 99 deletions(-) diff --git a/common/src/backend/java/dev/engine_room/flywheel/backend/engine/AbstractInstancer.java b/common/src/backend/java/dev/engine_room/flywheel/backend/engine/AbstractInstancer.java index 95bc88e58..605ee0441 100644 --- a/common/src/backend/java/dev/engine_room/flywheel/backend/engine/AbstractInstancer.java +++ b/common/src/backend/java/dev/engine_room/flywheel/backend/engine/AbstractInstancer.java @@ -20,7 +20,7 @@ protected AbstractInstancer(InstancerKey key, Recreate recreate) { public abstract int instanceCount(); - public abstract void removeDeletedInstances(); + public abstract void parallelUpdate(); public abstract void delete(); diff --git a/common/src/backend/java/dev/engine_room/flywheel/backend/engine/DrawManager.java b/common/src/backend/java/dev/engine_room/flywheel/backend/engine/DrawManager.java index f34887f00..898253bb0 100644 --- a/common/src/backend/java/dev/engine_room/flywheel/backend/engine/DrawManager.java +++ b/common/src/backend/java/dev/engine_room/flywheel/backend/engine/DrawManager.java @@ -50,7 +50,7 @@ public AbstractInstancer getInstancer(InstancerKey ke public Plan createFramePlan() { // Go wide on instancers to process deletions in parallel. - return ForEachPlan.of(() -> new ArrayList<>(instancers.values()), AbstractInstancer::removeDeletedInstances); + return ForEachPlan.of(() -> new ArrayList<>(instancers.values()), AbstractInstancer::parallelUpdate); } public void flush(LightStorage lightStorage, EnvironmentStorage environmentStorage) { diff --git a/common/src/backend/java/dev/engine_room/flywheel/backend/engine/indirect/IndirectInstancer.java b/common/src/backend/java/dev/engine_room/flywheel/backend/engine/indirect/IndirectInstancer.java index c09a3e9e9..3bec5e4f7 100644 --- a/common/src/backend/java/dev/engine_room/flywheel/backend/engine/indirect/IndirectInstancer.java +++ b/common/src/backend/java/dev/engine_room/flywheel/backend/engine/indirect/IndirectInstancer.java @@ -6,6 +6,7 @@ import java.util.concurrent.atomic.AtomicInteger; import java.util.concurrent.atomic.AtomicReference; +import org.jetbrains.annotations.NotNull; import org.jetbrains.annotations.Nullable; import org.jetbrains.annotations.UnknownNullability; import org.joml.Vector4fc; @@ -20,13 +21,12 @@ import dev.engine_room.flywheel.lib.math.MoreMath; public class IndirectInstancer extends AbstractInstancer { - private final AtomicReference pages; - private final long instanceStride; private final InstanceWriter writer; private final List associatedDraws = new ArrayList<>(); private final Vector4fc boundingSphere; + private final AtomicReference pages; private final AtomicBitSet changedPages = new AtomicBitSet(); private final AtomicBitSet fullPages = new AtomicBitSet(); private final Class instanceClass; @@ -45,7 +45,19 @@ public IndirectInstancer(InstancerKey key, Recreate recreate) { instanceClass = (Class) type.create(new InstanceHandleImpl(null)) .getClass(); - pages = new AtomicReference<>((InstancePage[]) Array.newInstance(InstancePage.class, 0)); + pages = new AtomicReference<>(pageArray(0)); + } + + @NotNull + @SuppressWarnings("unchecked") + private InstancePage[] pageArray(int length) { + return (InstancePage[]) Array.newInstance(InstancePage.class, length); + } + + @NotNull + @SuppressWarnings("unchecked") + private I[] instanceArray() { + return (I[]) Array.newInstance(instanceClass, ObjectStorage.PAGE_SIZE); } public final class InstancePage implements InstanceHandleImpl.State { @@ -57,19 +69,30 @@ public final class InstancePage implements InstanceHandleImpl.State { */ private final AtomicInteger valid; - InstancePage(Class clazz, int pageNo) { + InstancePage(int pageNo) { this.pageNo = pageNo; - this.instances = (I[]) Array.newInstance(clazz, ObjectStorage.PAGE_SIZE); + this.instances = instanceArray(); this.handles = (InstanceHandleImpl[]) new InstanceHandleImpl[ObjectStorage.PAGE_SIZE]; this.valid = new AtomicInteger(0); } + public int count() { + return Integer.bitCount(valid.get()); + } + + /** + * Attempt to add the given instance/handle to this page. + * + * @param instance The instance to add + * @param handle The instance's handle + * @return true if the instance was added, false if the page is full + */ public boolean add(I instance, InstanceHandleImpl handle) { // Thread safety: we loop until we either win the race and add the given instance, or we // run out of space because other threads trying to add at the same time. while (true) { int currentValue = valid.get(); - if (currentValue == 0xFFFFFFFF) { + if (isFull(currentValue)) { // The page is full, must search elsewhere return false; } @@ -84,10 +107,13 @@ public boolean add(I instance, InstanceHandleImpl handle) { instances[index] = instance; handles[index] = handle; handle.state = this; - handle.index = (pageNo << ObjectStorage.LOG_2_PAGE_SIZE) + index; + // Handle index is unique amongst all pages of this instancer. + handle.index = local2HandleIndex(index); changedPages.set(pageNo); - if (newValue == 0xFFFFFFFF) { + if (isFull(newValue)) { + // The page is now full, mark it so in the bitset. + // This is safe because only one bit position changes at a time. fullPages.set(pageNo); } return true; @@ -95,6 +121,10 @@ public boolean add(I instance, InstanceHandleImpl handle) { } } + private int local2HandleIndex(int index) { + return (pageNo << ObjectStorage.LOG_2_PAGE_SIZE) + index; + } + @Override public InstanceHandleImpl.State setChanged(int index) { changedPages.set(pageNo); @@ -114,6 +144,7 @@ public InstanceHandleImpl.State setDeleted(int index) { if (valid.compareAndSet(currentValue, newValue)) { fullPages.clear(pageNo); + changedPages.set(pageNo); return InstanceHandleImpl.Deleted.instance(); } } @@ -129,6 +160,56 @@ public InstanceHandleImpl.State setVisible(InstanceHandleImpl handle, int return new InstanceHandleImpl.Hidden<>(recreate, instances[localIndex]); } + + public int takeFrom(InstancePage other) { + // Fill the holes in this page with instances from the other page. + + int valid = this.valid.get(); + int otherValid = other.valid.get(); + + // If the other page is empty, or we're full, we're done. + if (otherValid == 0 || valid == 0xFFFFFFFF) { + return valid; + } + + // Something is going to change, so mark stuff ahead of time. + changedPages.set(pageNo); + changedPages.set(other.pageNo); + + for (int i = 0; i < ObjectStorage.PAGE_SIZE; i++) { + int mask = 1 << i; + + if ((otherValid & mask) == 0) { + continue; + } + + int writePos = Integer.numberOfTrailingZeros(~valid); + + instances[writePos] = other.instances[i]; + handles[writePos] = other.handles[i]; + + handles[writePos].state = this; + handles[writePos].index = local2HandleIndex(writePos); + + // Clear out the other page. + otherValid &= ~mask; + other.handles[i] = null; + other.instances[i] = null; + + // Set the bit in this page and find the next write position. + valid |= 1 << writePos; + + // If we're full, we're done. + if (isFull(valid)) { + break; + } + } + + this.valid.set(valid); + other.valid.set(otherValid); + + return valid; + } } public void addDraw(IndirectDraw draw) { @@ -209,8 +290,45 @@ public void uploadInstances(StagingBuffer stagingBuffer, int instanceVbo) { changedPages.clear(); } - public void removeDeletedInstances() { + public void parallelUpdate() { + if (true) { + // FIXME: infinite loop when the page in readpos doesn't have enough to fill the page in writepos + return; + } + + var pages = this.pages.get(); + + // If there are at least 2 pages with space, we can consolidate. + if (fullPages.cardinality() > (pages.length - 2)) { + return; + } + // Note this runs after visuals are updated so we don't really have to take care for thread safety. + + int writePos = 0; + + while (true) { + writePos = fullPages.nextClearBit(writePos); + int readPos = fullPages.nextClearBit(writePos + 1); + + if (writePos >= pages.length || readPos >= pages.length) { + break; + } + + InstancePage writeTo = pages[writePos]; + InstancePage readFrom = pages[readPos]; + + int validNow = writeTo.takeFrom(readFrom); + + if (isFull(validNow)) { + fullPages.set(writePos); + writePos = readPos; + } + } + } + + private static boolean isFull(int valid) { + return valid == 0xFFFFFFFF; } @Override @@ -319,10 +437,10 @@ private void addInner(I instance, InstanceHandleImpl handle) { // Thread safety: segments contains all pages from the currently visible pages, plus extra. // all pages in the currently visible pages are canonical and will not change. // Can't just `new InstancePage[]` because it has a generic parameter. - InstancePage[] newPages = (InstancePage[]) Array.newInstance(InstancePage.class, desiredLength); + InstancePage[] newPages = pageArray(desiredLength); System.arraycopy(pages, 0, newPages, 0, pages.length); - newPages[pages.length] = new InstancePage(instanceClass, pages.length); + newPages[pages.length] = new InstancePage(pages.length); // because we are using a compareAndSet, if this thread "wins the race" and successfully sets this variable, then the new page becomes canonical. if (this.pages.compareAndSet(pages, newPages)) { @@ -356,6 +474,9 @@ public int instanceCount() { * Clear all instances without freeing resources. */ public void clear() { + this.pages.set(pageArray(0)); + changedPages.clear(); + fullPages.clear(); } } diff --git a/common/src/backend/java/dev/engine_room/flywheel/backend/engine/indirect/ObjectStorage.java b/common/src/backend/java/dev/engine_room/flywheel/backend/engine/indirect/ObjectStorage.java index bf12c46a1..3f43b8374 100644 --- a/common/src/backend/java/dev/engine_room/flywheel/backend/engine/indirect/ObjectStorage.java +++ b/common/src/backend/java/dev/engine_room/flywheel/backend/engine/indirect/ObjectStorage.java @@ -98,9 +98,6 @@ public class Mapping { private static final int[] EMPTY_ALLOCATION = new int[0]; private int[] pages = EMPTY_ALLOCATION; - private int modelIndex = -1; - private int objectCount = 0; - public void updatePage(int i, int modelIndex, int i1) { var ptr = ptrForPage(pages[i]); MemoryUtil.memPutInt(ptr, modelIndex); @@ -109,61 +106,6 @@ public void updatePage(int i, int modelIndex, int i1) { ObjectStorage.this.needsUpload = true; } - /** - * Adjust this allocation to the given model index and object count. - * - *

This method triggers eager resizing of the allocation to fit the new object count. - * If the model index is different from the current one, all frame descriptors will be updated. - * - * @param modelIndex The model index the objects in this allocation are associated with. - * @param objectCount The number of objects in this allocation. - */ - public void update(int modelIndex, int objectCount) { - boolean incremental = this.modelIndex == modelIndex; - - if (incremental && objectCount == this.objectCount) { - // Nothing will change. - return; - } - - ObjectStorage.this.needsUpload = true; - - this.modelIndex = modelIndex; - this.objectCount = objectCount; - - var oldLength = pages.length; - var newLength = objectIndex2PageIndex((objectCount + PAGE_MASK)); - - if (oldLength > newLength) { - // Eagerly free the now unnecessary pages. - // shrink will zero out the pageTable entries for the freed pages. - shrink(oldLength, newLength); - - if (incremental) { - // Only update the last page, everything else is unchanged. - updateRange(newLength - 1, newLength); - } - } else if (oldLength < newLength) { - // Allocate new pages to fit the new object count. - grow(newLength, oldLength); - - if (incremental) { - // Update the old last page + all new pages - updateRange(oldLength - 1, newLength); - } - } else { - if (incremental) { - // Only update the last page. - updateRange(oldLength - 1, oldLength); - } - } - - if (!incremental) { - // Update all pages. - updateRange(0, newLength); - } - } - public void updateCount(int newLength) { var oldLength = pages.length; if (oldLength > newLength) { @@ -189,37 +131,10 @@ public void delete() { ObjectStorage.this.free(page); } pages = EMPTY_ALLOCATION; - modelIndex = -1; - objectCount = 0; ObjectStorage.this.needsUpload = true; } - /** - * Calculates the page descriptor for the given page index. - * Runs under the assumption than all pages are full except maybe the last one. - */ - private int calculatePageDescriptor(int pageIndex) { - int countInPage; - if (objectCount % PAGE_SIZE != 0 && pageIndex == pages.length - 1) { - // Last page && it isn't full -> use the remainder. - countInPage = objectCount & PAGE_MASK; - } else if (objectCount > 0) { - // Full page. - countInPage = PAGE_SIZE; - } else { - // Empty page, this shouldn't be reachable because we eagerly free empty pages. - countInPage = 0; - } - return (modelIndex & 0x3FFFFF) | (countInPage << 26); - } - - private void updateRange(int start, int oldLength) { - for (int i = start; i < oldLength; i++) { - MemoryUtil.memPutInt(ptrForPage(pages[i]), calculatePageDescriptor(i)); - } - } - private void grow(int neededPages, int oldLength) { pages = Arrays.copyOf(pages, neededPages); diff --git a/common/src/backend/java/dev/engine_room/flywheel/backend/engine/instancing/InstancedInstancer.java b/common/src/backend/java/dev/engine_room/flywheel/backend/engine/instancing/InstancedInstancer.java index 833e90ddf..c56b281eb 100644 --- a/common/src/backend/java/dev/engine_room/flywheel/backend/engine/instancing/InstancedInstancer.java +++ b/common/src/backend/java/dev/engine_room/flywheel/backend/engine/instancing/InstancedInstancer.java @@ -109,7 +109,7 @@ public boolean needsToGrow(long capacity) { return capacity > vbo.size(); } - public void removeDeletedInstances() { + public void parallelUpdate() { if (deleted.isEmpty()) { return; }