Simple storage staging

- Use one concurrent queue of transactions in InstanceManager
This commit is contained in:
Jozufozu 2023-04-07 00:14:39 -07:00
parent ab8ab8dbd4
commit 945ed9a1e4
3 changed files with 43 additions and 45 deletions

View file

@ -1,8 +1,8 @@
package com.jozufozu.flywheel.impl.instancing.manager; package com.jozufozu.flywheel.impl.instancing.manager;
import java.util.HashSet;
import java.util.List; import java.util.List;
import java.util.Set; import java.util.Queue;
import java.util.concurrent.ConcurrentLinkedQueue;
import java.util.function.Consumer; import java.util.function.Consumer;
import org.joml.FrustumIntersection; import org.joml.FrustumIntersection;
@ -16,10 +16,10 @@ import com.jozufozu.flywheel.impl.instancing.ratelimit.BandedPrimeLimiter;
import com.jozufozu.flywheel.impl.instancing.ratelimit.DistanceUpdateLimiter; import com.jozufozu.flywheel.impl.instancing.ratelimit.DistanceUpdateLimiter;
import com.jozufozu.flywheel.impl.instancing.ratelimit.NonLimiter; import com.jozufozu.flywheel.impl.instancing.ratelimit.NonLimiter;
import com.jozufozu.flywheel.impl.instancing.storage.Storage; import com.jozufozu.flywheel.impl.instancing.storage.Storage;
import com.jozufozu.flywheel.impl.instancing.storage.Transaction;
public abstract class InstanceManager<T> { public abstract class InstanceManager<T> {
private final Set<T> queuedAdditions = new HashSet<>(64); private final Queue<Transaction<T>> queue = new ConcurrentLinkedQueue<>();
private final Set<T> queuedUpdates = new HashSet<>(64);
protected DistanceUpdateLimiter tickLimiter; protected DistanceUpdateLimiter tickLimiter;
protected DistanceUpdateLimiter frameLimiter; protected DistanceUpdateLimiter frameLimiter;
@ -61,9 +61,7 @@ public abstract class InstanceManager<T> {
return; return;
} }
synchronized (queuedAdditions) { queue.add(Transaction.add(obj));
queuedAdditions.add(obj);
}
} }
/** /**
@ -90,9 +88,7 @@ public abstract class InstanceManager<T> {
return; return;
} }
synchronized (queuedUpdates) { queue.add(Transaction.update(obj));
queuedUpdates.add(obj);
}
} }
public void remove(T obj) { public void remove(T obj) {
@ -107,37 +103,11 @@ public abstract class InstanceManager<T> {
getStorage().invalidate(); getStorage().invalidate();
} }
protected void processQueuedAdditions() { protected void processQueue() {
if (queuedAdditions.isEmpty()) { var storage = getStorage();
return; Transaction<T> transaction;
} while ((transaction = queue.poll()) != null) {
transaction.apply(storage);
List<T> queued;
synchronized (queuedAdditions) {
queued = List.copyOf(queuedAdditions);
queuedAdditions.clear();
}
if (!queued.isEmpty()) {
queued.forEach(getStorage()::add);
}
}
protected void processQueuedUpdates() {
if (queuedUpdates.isEmpty()) {
return;
}
List<T> queued;
synchronized (queuedUpdates) {
queued = List.copyOf(queuedUpdates);
queuedUpdates.clear();
}
if (!queued.isEmpty()) {
queued.forEach(getStorage()::update);
} }
} }
@ -152,8 +122,7 @@ public abstract class InstanceManager<T> {
*/ */
public void tick(TaskExecutor executor, double cameraX, double cameraY, double cameraZ) { public void tick(TaskExecutor executor, double cameraX, double cameraY, double cameraZ) {
tickLimiter.tick(); tickLimiter.tick();
processQueuedAdditions(); processQueue();
processQueuedUpdates();
var instances = getStorage().getTickableInstances(); var instances = getStorage().getTickableInstances();
distributeWork(executor, instances, instance -> tickInstance(instance, cameraX, cameraY, cameraZ)); distributeWork(executor, instances, instance -> tickInstance(instance, cameraX, cameraY, cameraZ));
@ -167,8 +136,7 @@ public abstract class InstanceManager<T> {
public void beginFrame(TaskExecutor executor, double cameraX, double cameraY, double cameraZ, FrustumIntersection frustum) { public void beginFrame(TaskExecutor executor, double cameraX, double cameraY, double cameraZ, FrustumIntersection frustum) {
frameLimiter.tick(); frameLimiter.tick();
processQueuedAdditions(); processQueue();
processQueuedUpdates();
var instances = getStorage().getDynamicInstances(); var instances = getStorage().getDynamicInstances();
distributeWork(executor, instances, instance -> updateInstance(instance, cameraX, cameraY, cameraZ, frustum)); distributeWork(executor, instances, instance -> updateInstance(instance, cameraX, cameraY, cameraZ, frustum));

View file

@ -0,0 +1,7 @@
package com.jozufozu.flywheel.impl.instancing.storage;
public enum Action {
ADD,
REMOVE,
UPDATE,
}

View file

@ -0,0 +1,23 @@
package com.jozufozu.flywheel.impl.instancing.storage;
public record Transaction<T>(T obj, Action action) {
public static <T> Transaction<T> add(T obj) {
return new Transaction<>(obj, Action.ADD);
}
public static <T> Transaction<T> remove(T obj) {
return new Transaction<>(obj, Action.REMOVE);
}
public static <T> Transaction<T> update(T obj) {
return new Transaction<>(obj, Action.UPDATE);
}
public void apply(Storage<T> storage) {
switch (action) {
case ADD -> storage.add(obj);
case REMOVE -> storage.remove(obj);
case UPDATE -> storage.update(obj);
}
}
}