mirror of
https://github.com/Jozufozu/Flywheel.git
synced 2025-01-07 12:56:31 +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;
|
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));
|
||||||
|
|
|
@ -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