Skip to content

Commit

Permalink
A real page turner
Browse files Browse the repository at this point in the history
- Try to shuffle over instances into pages with space
- Clear out now-unused logic from ObjectStorage
- Some cleanup and more comments in IndirectInstancer
  • Loading branch information
Jozufozu committed Oct 30, 2024
1 parent a7e7090 commit 20b3f78
Show file tree
Hide file tree
Showing 5 changed files with 135 additions and 99 deletions.
Original file line number Diff line number Diff line change
Expand Up @@ -20,7 +20,7 @@ protected AbstractInstancer(InstancerKey<I> key, Recreate<I> recreate) {

public abstract int instanceCount();

public abstract void removeDeletedInstances();
public abstract void parallelUpdate();

public abstract void delete();

Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -50,7 +50,7 @@ public <I extends Instance> AbstractInstancer<I> getInstancer(InstancerKey<I> ke

public Plan<RenderContext> 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) {
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -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;
Expand All @@ -20,13 +21,12 @@
import dev.engine_room.flywheel.lib.math.MoreMath;

public class IndirectInstancer<I extends Instance> extends AbstractInstancer<I> {
private final AtomicReference<InstancePage[]> pages;

private final long instanceStride;
private final InstanceWriter<I> writer;
private final List<IndirectDraw> associatedDraws = new ArrayList<>();
private final Vector4fc boundingSphere;

private final AtomicReference<InstancePage[]> pages;
private final AtomicBitSet changedPages = new AtomicBitSet();
private final AtomicBitSet fullPages = new AtomicBitSet();
private final Class<I> instanceClass;
Expand All @@ -45,7 +45,19 @@ public IndirectInstancer(InstancerKey<I> key, Recreate<I> recreate) {

instanceClass = (Class<I>) type.create(new InstanceHandleImpl<I>(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<I> {
Expand All @@ -57,19 +69,30 @@ public final class InstancePage implements InstanceHandleImpl.State<I> {
*/
private final AtomicInteger valid;

InstancePage(Class<I> clazz, int pageNo) {
InstancePage(int pageNo) {
this.pageNo = pageNo;
this.instances = (I[]) Array.newInstance(clazz, ObjectStorage.PAGE_SIZE);
this.instances = instanceArray();
this.handles = (InstanceHandleImpl<I>[]) 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<I> 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;
}
Expand All @@ -84,17 +107,24 @@ public boolean add(I instance, InstanceHandleImpl<I> 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;
}
}
}

private int local2HandleIndex(int index) {
return (pageNo << ObjectStorage.LOG_2_PAGE_SIZE) + index;
}

@Override
public InstanceHandleImpl.State<I> setChanged(int index) {
changedPages.set(pageNo);
Expand All @@ -114,6 +144,7 @@ public InstanceHandleImpl.State<I> setDeleted(int index) {

if (valid.compareAndSet(currentValue, newValue)) {
fullPages.clear(pageNo);
changedPages.set(pageNo);
return InstanceHandleImpl.Deleted.instance();
}
}
Expand All @@ -129,6 +160,56 @@ public InstanceHandleImpl.State<I> setVisible(InstanceHandleImpl<I> 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) {
Expand Down Expand Up @@ -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
Expand Down Expand Up @@ -319,10 +437,10 @@ private void addInner(I instance, InstanceHandleImpl<I> 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)) {
Expand Down Expand Up @@ -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();

}
}
Original file line number Diff line number Diff line change
Expand Up @@ -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);
Expand All @@ -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.
*
* <p>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) {
Expand All @@ -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);

Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -109,7 +109,7 @@ public boolean needsToGrow(long capacity) {
return capacity > vbo.size();
}

public void removeDeletedInstances() {
public void parallelUpdate() {
if (deleted.isEmpty()) {
return;
}
Expand Down

0 comments on commit 20b3f78

Please sign in to comment.