diff --git a/common/src/api/java/dev/engine_room/flywheel/api/backend/Backend.java b/common/src/api/java/dev/engine_room/flywheel/api/backend/Backend.java
index 6cc6228f9..c6f5251a6 100644
--- a/common/src/api/java/dev/engine_room/flywheel/api/backend/Backend.java
+++ b/common/src/api/java/dev/engine_room/flywheel/api/backend/Backend.java
@@ -13,6 +13,13 @@ public interface Backend {
*/
Engine createEngine(LevelAccessor level);
+ /**
+ * The version of flywheel this backend was written against.
+ *
+ * @return A backend version.
+ */
+ BackendVersion version();
+
/**
* The priority of this backend.
*
The backend with the highest priority upon first launch will be chosen as the default backend.
diff --git a/common/src/api/java/dev/engine_room/flywheel/api/backend/BackendVersion.java b/common/src/api/java/dev/engine_room/flywheel/api/backend/BackendVersion.java
new file mode 100644
index 000000000..87f798174
--- /dev/null
+++ b/common/src/api/java/dev/engine_room/flywheel/api/backend/BackendVersion.java
@@ -0,0 +1,11 @@
+package dev.engine_room.flywheel.api.backend;
+
+public record BackendVersion(int major, int minor) implements Comparable {
+ @Override
+ public int compareTo(BackendVersion o) {
+ if (major != o.major) {
+ return Integer.compare(major, o.major);
+ }
+ return Integer.compare(minor, o.minor);
+ }
+}
diff --git a/common/src/backend/java/dev/engine_room/flywheel/backend/Backends.java b/common/src/backend/java/dev/engine_room/flywheel/backend/Backends.java
index c6de7eb47..da5a71372 100644
--- a/common/src/backend/java/dev/engine_room/flywheel/backend/Backends.java
+++ b/common/src/backend/java/dev/engine_room/flywheel/backend/Backends.java
@@ -2,6 +2,7 @@
import dev.engine_room.flywheel.api.Flywheel;
import dev.engine_room.flywheel.api.backend.Backend;
+import dev.engine_room.flywheel.api.backend.BackendVersion;
import dev.engine_room.flywheel.backend.compile.IndirectPrograms;
import dev.engine_room.flywheel.backend.compile.InstancingPrograms;
import dev.engine_room.flywheel.backend.engine.EngineImpl;
@@ -12,11 +13,14 @@
import dev.engine_room.flywheel.lib.util.ShadersModHelper;
public final class Backends {
+ private static final BackendVersion VERSION = new BackendVersion(1, 0);
+
/**
* Use GPU instancing to render everything.
*/
public static final Backend INSTANCING = SimpleBackend.builder()
.engineFactory(level -> new EngineImpl(level, new InstancedDrawManager(InstancingPrograms.get()), 256))
+ .version(VERSION)
.priority(500)
.supported(() -> GlCompat.SUPPORTS_INSTANCING && InstancingPrograms.allLoaded() && !ShadersModHelper.isShaderPackInUse())
.register(Flywheel.rl("instancing"));
@@ -26,6 +30,7 @@ public final class Backends {
*/
public static final Backend INDIRECT = SimpleBackend.builder()
.engineFactory(level -> new EngineImpl(level, new IndirectDrawManager(IndirectPrograms.get()), 256))
+ .version(VERSION)
.priority(1000)
.supported(() -> GlCompat.SUPPORTS_INDIRECT && IndirectPrograms.allLoaded() && !ShadersModHelper.isShaderPackInUse())
.register(Flywheel.rl("indirect"));
diff --git a/common/src/lib/java/dev/engine_room/flywheel/lib/backend/SimpleBackend.java b/common/src/lib/java/dev/engine_room/flywheel/lib/backend/SimpleBackend.java
index bc09227c1..0ac537c3b 100644
--- a/common/src/lib/java/dev/engine_room/flywheel/lib/backend/SimpleBackend.java
+++ b/common/src/lib/java/dev/engine_room/flywheel/lib/backend/SimpleBackend.java
@@ -4,7 +4,10 @@
import java.util.function.BooleanSupplier;
import java.util.function.Function;
+import org.jetbrains.annotations.Nullable;
+
import dev.engine_room.flywheel.api.backend.Backend;
+import dev.engine_room.flywheel.api.backend.BackendVersion;
import dev.engine_room.flywheel.api.backend.Engine;
import net.minecraft.resources.ResourceLocation;
import net.minecraft.world.level.LevelAccessor;
@@ -13,11 +16,13 @@ public final class SimpleBackend implements Backend {
private final Function engineFactory;
private final int priority;
private final BooleanSupplier isSupported;
+ private final BackendVersion version;
- public SimpleBackend(int priority, Function engineFactory, BooleanSupplier isSupported) {
+ public SimpleBackend(int priority, Function engineFactory, BooleanSupplier isSupported, BackendVersion version) {
this.priority = priority;
this.engineFactory = engineFactory;
this.isSupported = isSupported;
+ this.version = version;
}
public static Builder builder() {
@@ -29,6 +34,11 @@ public Engine createEngine(LevelAccessor level) {
return engineFactory.apply(level);
}
+ @Override
+ public BackendVersion version() {
+ return version;
+ }
+
@Override
public int priority() {
return priority;
@@ -40,9 +50,13 @@ public boolean isSupported() {
}
public static final class Builder {
- private Function engineFactory;
private int priority = 0;
+ @Nullable
+ private Function engineFactory;
+ @Nullable
private BooleanSupplier isSupported;
+ @Nullable
+ private BackendVersion version;
public Builder engineFactory(Function engineFactory) {
this.engineFactory = engineFactory;
@@ -59,11 +73,17 @@ public Builder supported(BooleanSupplier isSupported) {
return this;
}
+ public Builder version(BackendVersion version) {
+ this.version = version;
+ return this;
+ }
+
public Backend register(ResourceLocation id) {
Objects.requireNonNull(engineFactory);
Objects.requireNonNull(isSupported);
+ Objects.requireNonNull(version);
- return Backend.REGISTRY.registerAndGet(id, new SimpleBackend(priority, engineFactory, isSupported));
+ return Backend.REGISTRY.registerAndGet(id, new SimpleBackend(priority, engineFactory, isSupported, version));
}
}
}
diff --git a/common/src/main/java/dev/engine_room/flywheel/impl/BackendManagerImpl.java b/common/src/main/java/dev/engine_room/flywheel/impl/BackendManagerImpl.java
index 75133adeb..6697857cf 100644
--- a/common/src/main/java/dev/engine_room/flywheel/impl/BackendManagerImpl.java
+++ b/common/src/main/java/dev/engine_room/flywheel/impl/BackendManagerImpl.java
@@ -1,9 +1,11 @@
package dev.engine_room.flywheel.impl;
import java.util.ArrayList;
+import java.util.List;
import dev.engine_room.flywheel.api.Flywheel;
import dev.engine_room.flywheel.api.backend.Backend;
+import dev.engine_room.flywheel.api.backend.BackendVersion;
import dev.engine_room.flywheel.impl.visualization.VisualizationManagerImpl;
import dev.engine_room.flywheel.lib.backend.SimpleBackend;
import net.minecraft.client.multiplayer.ClientLevel;
@@ -15,6 +17,7 @@ public final class BackendManagerImpl {
throw new UnsupportedOperationException("Cannot create engine when backend is off.");
})
.supported(() -> true)
+ .version(new BackendVersion(0, 0))
.register(Flywheel.rl("off"));
public static final Backend DEFAULT_BACKEND = findDefaultBackend();
@@ -33,7 +36,7 @@ public static boolean isBackendOn() {
}
// Don't store this statically because backends can theoretically change their priorities at runtime.
- private static ArrayList backendsByPriority() {
+ private static List backendsByPriority() {
var backends = new ArrayList<>(Backend.REGISTRY.getAll());
// Sort with keys backwards so that the highest priority is first.
@@ -53,27 +56,46 @@ private static Backend findDefaultBackend() {
}
private static void chooseBackend() {
+ var requirements = FlwImplXplat.INSTANCE.modRequirements();
+
var preferred = FlwConfig.INSTANCE.backend();
if (preferred.isSupported()) {
- backend = preferred;
- return;
+ if (meetsModRequirements(requirements, preferred)) {
+ backend = preferred;
+ return;
+ } else {
+ var minVersion = requirements.minimumBackendVersion();
+ FlwImpl.LOGGER.warn("Preferred backend '{}' does not meet minimum version requirement {}.{}", Backend.REGISTRY.getIdOrThrow(preferred), minVersion.major(), minVersion.minor());
+ }
}
+ backend = fallback(preferred, requirements);
+
+ FlwImpl.LOGGER.warn("Flywheel backend fell back from '{}' to '{}'", Backend.REGISTRY.getIdOrThrow(preferred), Backend.REGISTRY.getIdOrThrow(backend));
+ }
+
+ private static Backend fallback(Backend preferred, ModRequirements requirements) {
var backendsByPriority = backendsByPriority();
var startIndex = backendsByPriority.indexOf(preferred) + 1;
// For safety in case we don't find anything
- backend = OFF_BACKEND;
+ var out = OFF_BACKEND;
for (int i = startIndex; i < backendsByPriority.size(); i++) {
var candidate = backendsByPriority.get(i);
if (candidate.isSupported()) {
- backend = candidate;
- break;
+ if (meetsModRequirements(requirements, candidate)) {
+ out = candidate;
+ break;
+ }
}
}
+ return out;
+ }
- FlwImpl.LOGGER.warn("Flywheel backend fell back from '{}' to '{}'", Backend.REGISTRY.getIdOrThrow(preferred), Backend.REGISTRY.getIdOrThrow(backend));
+ private static boolean meetsModRequirements(ModRequirements requirements, Backend candidate) {
+ return candidate.version()
+ .compareTo(requirements.minimumBackendVersion()) >= 0;
}
public static String getBackendString() {
diff --git a/common/src/main/java/dev/engine_room/flywheel/impl/FlwImplXplat.java b/common/src/main/java/dev/engine_room/flywheel/impl/FlwImplXplat.java
index c8fb9fec1..380c91b06 100644
--- a/common/src/main/java/dev/engine_room/flywheel/impl/FlwImplXplat.java
+++ b/common/src/main/java/dev/engine_room/flywheel/impl/FlwImplXplat.java
@@ -17,4 +17,6 @@ public interface FlwImplXplat {
boolean useSodium0_6Compat();
boolean useIrisCompat();
+
+ ModRequirements modRequirements();
}
diff --git a/common/src/main/java/dev/engine_room/flywheel/impl/ModRequirements.java b/common/src/main/java/dev/engine_room/flywheel/impl/ModRequirements.java
new file mode 100644
index 000000000..199788316
--- /dev/null
+++ b/common/src/main/java/dev/engine_room/flywheel/impl/ModRequirements.java
@@ -0,0 +1,10 @@
+package dev.engine_room.flywheel.impl;
+
+import java.util.List;
+
+import dev.engine_room.flywheel.api.backend.BackendVersion;
+
+public record ModRequirements(BackendVersion minimumBackendVersion, List entries) {
+ public record Entry(String modId, BackendVersion version) {
+ }
+}
diff --git a/fabric/src/main/java/dev/engine_room/flywheel/impl/FlwImplXplatImpl.java b/fabric/src/main/java/dev/engine_room/flywheel/impl/FlwImplXplatImpl.java
index d1c346906..775440cf9 100644
--- a/fabric/src/main/java/dev/engine_room/flywheel/impl/FlwImplXplatImpl.java
+++ b/fabric/src/main/java/dev/engine_room/flywheel/impl/FlwImplXplatImpl.java
@@ -1,9 +1,17 @@
package dev.engine_room.flywheel.impl;
+import java.util.ArrayList;
+import java.util.List;
+
+import org.jetbrains.annotations.Nullable;
+
+import dev.engine_room.flywheel.api.backend.BackendVersion;
import dev.engine_room.flywheel.api.event.ReloadLevelRendererCallback;
import dev.engine_room.flywheel.impl.compat.CompatMod;
import dev.engine_room.flywheel.impl.compat.FabricSodiumCompat;
import net.fabricmc.loader.api.FabricLoader;
+import net.fabricmc.loader.api.ModContainer;
+import net.fabricmc.loader.api.metadata.CustomValue;
import net.minecraft.client.multiplayer.ClientLevel;
public class FlwImplXplatImpl implements FlwImplXplat {
@@ -36,4 +44,54 @@ public boolean useSodium0_6Compat() {
public boolean useIrisCompat() {
return CompatMod.IRIS.isLoaded;
}
+
+ @Override
+ public ModRequirements modRequirements() {
+ List entries = new ArrayList<>();
+
+ for (ModContainer mod : FabricLoader.getInstance()
+ .getAllMods()) {
+ var metadata = mod.getMetadata();
+
+ var custom = metadata.getCustomValue("flywheel");
+
+ if (custom != null && custom.getType() == CustomValue.CvType.OBJECT) {
+ var object = custom.getAsObject();
+
+ var major = getCustomValueAsInt(object.get("backend_major_version"));
+ var minor = getCustomValueAsInt(object.get("backend_minor_version"));
+
+ // Major version is required
+ if (major != null) {
+ // Minor version defaults to 0
+ var version = new BackendVersion(major, minor == null ? 0 : minor);
+
+ entries.add(new ModRequirements.Entry(metadata.getId(), version));
+ } else {
+ FlwImpl.LOGGER.warn("Mod {} has invalid required backend version", metadata.getId());
+ }
+ }
+ }
+
+ if (!entries.isEmpty()) {
+ var minVersion = entries.stream()
+ .map(ModRequirements.Entry::version)
+ .min(BackendVersion::compareTo)
+ .get();
+
+ return new ModRequirements(minVersion, entries);
+ } else {
+ return new ModRequirements(new BackendVersion(0, 0), List.of());
+ }
+ }
+
+ @Nullable
+ private static Integer getCustomValueAsInt(@Nullable CustomValue major) {
+ if (major != null && major.getType() == CustomValue.CvType.NUMBER) {
+ return major.getAsNumber()
+ .intValue();
+ }
+
+ return null;
+ }
}
diff --git a/fabric/src/main/resources/fabric.mod.json b/fabric/src/main/resources/fabric.mod.json
index d90f772c9..24caeb709 100644
--- a/fabric/src/main/resources/fabric.mod.json
+++ b/fabric/src/main/resources/fabric.mod.json
@@ -35,5 +35,11 @@
"breaks": {
"sodium": ["<0.5.0", "~0.6.0- <0.6.0-beta.2"],
"embeddium": "*"
+ },
+ "custom" : {
+ "flywheel" : {
+ "backend_major_version" : 1,
+ "backend_minor_version" : 0
+ }
}
}
diff --git a/forge/src/main/java/dev/engine_room/flywheel/impl/FlwImplXplatImpl.java b/forge/src/main/java/dev/engine_room/flywheel/impl/FlwImplXplatImpl.java
index 78e99c9e0..bd28b6d7e 100644
--- a/forge/src/main/java/dev/engine_room/flywheel/impl/FlwImplXplatImpl.java
+++ b/forge/src/main/java/dev/engine_room/flywheel/impl/FlwImplXplatImpl.java
@@ -1,10 +1,18 @@
package dev.engine_room.flywheel.impl;
+import java.util.ArrayList;
+import java.util.List;
+
+import com.electronwill.nightconfig.core.Config;
+
+import dev.engine_room.flywheel.api.backend.BackendVersion;
import dev.engine_room.flywheel.api.event.ReloadLevelRendererEvent;
import dev.engine_room.flywheel.impl.compat.CompatMod;
import net.minecraft.client.multiplayer.ClientLevel;
import net.minecraftforge.common.MinecraftForge;
+import net.minecraftforge.fml.ModList;
import net.minecraftforge.fml.loading.LoadingModList;
+import net.minecraftforge.forgespi.language.IModInfo;
public class FlwImplXplatImpl implements FlwImplXplat {
@Override
@@ -36,4 +44,46 @@ public boolean useSodium0_6Compat() {
public boolean useIrisCompat() {
return CompatMod.IRIS.isLoaded || CompatMod.OCULUS.isLoaded;
}
+
+ @Override
+ public ModRequirements modRequirements() {
+ List entries = new ArrayList<>();
+
+ ModList.get()
+ .forEachModFile(file -> {
+ var info = file.getModFileInfo();
+
+ for (IModInfo mod : info.getMods()) {
+ var modProperties = mod.getModProperties()
+ .get("flywheel");
+
+ // There's no well-defined API for custom properties like in fabric.
+ // It just returns an object, but internally it's represented with nightconfig.
+ if (modProperties instanceof Config config) {
+ // Minor version defaults to 0, major is required
+ int major = config.getIntOrElse("backend_major_version", Integer.MIN_VALUE);
+ int minor = config.getIntOrElse("backend_minor_version", 0);
+
+ if (major != Integer.MIN_VALUE) {
+ entries.add(new ModRequirements.Entry(mod.getModId(), new BackendVersion(major, minor)));
+ } else {
+ FlwImpl.LOGGER.warn("Mod {} has invalid required backend version", mod.getModId());
+ }
+ } else {
+ FlwImpl.LOGGER.warn("Mod {} has invalid flywheel properties", mod.getModId());
+ }
+ }
+ });
+
+ if (!entries.isEmpty()) {
+ var minVersion = entries.stream()
+ .map(ModRequirements.Entry::version)
+ .min(BackendVersion::compareTo)
+ .get();
+
+ return new ModRequirements(minVersion, entries);
+ } else {
+ return new ModRequirements(new BackendVersion(0, 0), List.of());
+ }
+ }
}
diff --git a/forge/src/main/resources/META-INF/mods.toml b/forge/src/main/resources/META-INF/mods.toml
index be991ac88..991e1095a 100644
--- a/forge/src/main/resources/META-INF/mods.toml
+++ b/forge/src/main/resources/META-INF/mods.toml
@@ -14,6 +14,10 @@ authors = "Jozufozu, PepperCode1"
displayURL = "${mod_homepage}"
displayTest = "IGNORE_ALL_VERSION"
+[modproperties.${ mod_id }.flywheel]
+backend_major_version = 1
+backend_minor_version = 0
+
[[dependencies.${mod_id}]]
modId = "minecraft"
mandatory = true