mirror of
https://github.com/Jozufozu/Flywheel.git
synced 2025-01-22 10:57:55 +01:00
Simple storage staging
- Use one concurrent queue of transactions in InstanceManager
This commit is contained in:
parent
ab8ab8dbd4
commit
945ed9a1e4
3 changed files with 43 additions and 45 deletions
|
@ -1,8 +1,8 @@
|
|||
package com.jozufozu.flywheel.impl.instancing.manager;
|
||||
|
||||
import java.util.HashSet;
|
||||
import java.util.List;
|
||||
import java.util.Set;
|
||||
import java.util.Queue;
|
||||
import java.util.concurrent.ConcurrentLinkedQueue;
|
||||
import java.util.function.Consumer;
|
||||
|
||||
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.NonLimiter;
|
||||
import com.jozufozu.flywheel.impl.instancing.storage.Storage;
|
||||
import com.jozufozu.flywheel.impl.instancing.storage.Transaction;
|
||||
|
||||
public abstract class InstanceManager<T> {
|
||||
private final Set<T> queuedAdditions = new HashSet<>(64);
|
||||
private final Set<T> queuedUpdates = new HashSet<>(64);
|
||||
private final Queue<Transaction<T>> queue = new ConcurrentLinkedQueue<>();
|
||||
|
||||
protected DistanceUpdateLimiter tickLimiter;
|
||||
protected DistanceUpdateLimiter frameLimiter;
|
||||
|
@ -61,9 +61,7 @@ public abstract class InstanceManager<T> {
|
|||
return;
|
||||
}
|
||||
|
||||
synchronized (queuedAdditions) {
|
||||
queuedAdditions.add(obj);
|
||||
}
|
||||
queue.add(Transaction.add(obj));
|
||||
}
|
||||
|
||||
/**
|
||||
|
@ -90,9 +88,7 @@ public abstract class InstanceManager<T> {
|
|||
return;
|
||||
}
|
||||
|
||||
synchronized (queuedUpdates) {
|
||||
queuedUpdates.add(obj);
|
||||
}
|
||||
queue.add(Transaction.update(obj));
|
||||
}
|
||||
|
||||
public void remove(T obj) {
|
||||
|
@ -107,37 +103,11 @@ public abstract class InstanceManager<T> {
|
|||
getStorage().invalidate();
|
||||
}
|
||||
|
||||
protected void processQueuedAdditions() {
|
||||
if (queuedAdditions.isEmpty()) {
|
||||
return;
|
||||
}
|
||||
|
||||
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);
|
||||
protected void processQueue() {
|
||||
var storage = getStorage();
|
||||
Transaction<T> transaction;
|
||||
while ((transaction = queue.poll()) != null) {
|
||||
transaction.apply(storage);
|
||||
}
|
||||
}
|
||||
|
||||
|
@ -152,8 +122,7 @@ public abstract class InstanceManager<T> {
|
|||
*/
|
||||
public void tick(TaskExecutor executor, double cameraX, double cameraY, double cameraZ) {
|
||||
tickLimiter.tick();
|
||||
processQueuedAdditions();
|
||||
processQueuedUpdates();
|
||||
processQueue();
|
||||
|
||||
var instances = getStorage().getTickableInstances();
|
||||
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) {
|
||||
frameLimiter.tick();
|
||||
processQueuedAdditions();
|
||||
processQueuedUpdates();
|
||||
processQueue();
|
||||
|
||||
var instances = getStorage().getDynamicInstances();
|
||||
distributeWork(executor, instances, instance -> updateInstance(instance, cameraX, cameraY, cameraZ, frustum));
|
||||
|
|
|
@ -0,0 +1,7 @@
|
|||
package com.jozufozu.flywheel.impl.instancing.storage;
|
||||
|
||||
public enum Action {
|
||||
ADD,
|
||||
REMOVE,
|
||||
UPDATE,
|
||||
}
|
|
@ -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);
|
||||
}
|
||||
}
|
||||
}
|
Loading…
Reference in a new issue