Compare commits
11 Commits
v0.3.0-bet
...
v0.1.0-bet
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
4597f04a6d | ||
|
|
20764890a7 | ||
|
|
f719abdb64 | ||
|
|
0dce47bba6 | ||
|
|
47584ceee1 | ||
|
|
2fc858a683 | ||
|
|
68ea18bc51 | ||
|
|
9944946b29 | ||
|
|
ec1120ed10 | ||
|
|
8040c7a264 | ||
|
|
a9e36795e5 |
2
.github/workflows/build.yml
vendored
2
.github/workflows/build.yml
vendored
@@ -31,6 +31,8 @@ jobs:
|
|||||||
key: ${{ runner.os }}-project-local-gradle-caches-${{ hashFiles('**/libs.versions.toml', '**/*.gradle*', '**/gradle-wrapper.properties') }}
|
key: ${{ runner.os }}-project-local-gradle-caches-${{ hashFiles('**/libs.versions.toml', '**/*.gradle*', '**/gradle-wrapper.properties') }}
|
||||||
restore-keys: |
|
restore-keys: |
|
||||||
${{ runner.os }}-project-local-gradle-caches-
|
${{ runner.os }}-project-local-gradle-caches-
|
||||||
|
- name: "setup concurrentutil"
|
||||||
|
run: ./installConcurrentUtil.sh
|
||||||
- name: "execute gradle build"
|
- name: "execute gradle build"
|
||||||
run: ./gradlew build
|
run: ./gradlew build
|
||||||
- name: Determine Snapshot Status
|
- name: Determine Snapshot Status
|
||||||
|
|||||||
3
.gitignore
vendored
3
.gitignore
vendored
@@ -116,6 +116,3 @@ run/
|
|||||||
|
|
||||||
# Avoid ignoring Gradle wrapper jar file (.jar files are usually ignored)
|
# Avoid ignoring Gradle wrapper jar file (.jar files are usually ignored)
|
||||||
!gradle-wrapper.jar
|
!gradle-wrapper.jar
|
||||||
|
|
||||||
ConcurrentUtil/
|
|
||||||
YamlConfig/
|
|
||||||
|
|||||||
3
.gitmodules
vendored
3
.gitmodules
vendored
@@ -0,0 +1,3 @@
|
|||||||
|
[submodule "ConcurrentUtil"]
|
||||||
|
path = ConcurrentUtil
|
||||||
|
url = https://github.com/Spottedleaf/ConcurrentUtil.git
|
||||||
|
|||||||
1
ConcurrentUtil
Submodule
1
ConcurrentUtil
Submodule
Submodule ConcurrentUtil added at 08d3ca3241
@@ -12,13 +12,13 @@ Fabric/NeoForge mod for optimising performance of the integrated (singleplayer/L
|
|||||||
Moonrise aims to optimise the game *without changing Vanilla behavior*. If you find that there are changes to Vanilla behavior,
|
Moonrise aims to optimise the game *without changing Vanilla behavior*. If you find that there are changes to Vanilla behavior,
|
||||||
please open an issue.
|
please open an issue.
|
||||||
|
|
||||||
Moonrise is an official port of several important [Paper](https://github.com/PaperMC/Paper/)
|
Moonrise ports several important [Paper](https://github.com/PaperMC/Paper/)
|
||||||
patches. Listed below are notable patches:
|
patches. Listed below are notable patches:
|
||||||
- [Starlight](https://github.com/PaperMC/Starlight/)
|
|
||||||
- Chunk system rewrite
|
- Chunk system rewrite
|
||||||
- Collision optimisations
|
- Collision optimisations
|
||||||
- Entity tracker optimisations
|
- Entity tracker optimisations
|
||||||
- Random ticking optimisations
|
- Random ticking optimisations
|
||||||
|
- [Starlight](https://github.com/PaperMC/Starlight/)
|
||||||
|
|
||||||
## Known Compatibility Issues
|
## Known Compatibility Issues
|
||||||
| Mod | Status |
|
| Mod | Status |
|
||||||
|
|||||||
124
build.gradle
124
build.gradle
@@ -1,53 +1,40 @@
|
|||||||
import me.modmuss50.mpp.ReleaseType
|
import me.modmuss50.mpp.ReleaseType
|
||||||
|
|
||||||
plugins {
|
plugins {
|
||||||
id("java-library")
|
id("xyz.jpenilla.quiet-architectury-loom")
|
||||||
id("net.neoforged.moddev")
|
id("me.modmuss50.mod-publish-plugin") version "0.7.2" apply false
|
||||||
id("me.modmuss50.mod-publish-plugin") version "0.8.4" apply false
|
|
||||||
}
|
}
|
||||||
|
|
||||||
extensions.create("runConfigCommon", RunConfigCommon.class)
|
/*
|
||||||
|
* Gets the version name from the latest Git tag
|
||||||
def getGitCommit = providers.exec {
|
*/
|
||||||
|
// https://stackoverflow.com/questions/28498688/gradle-script-to-autoversion-and-include-the-commit-hash-in-android
|
||||||
|
def getGitCommit = { ->
|
||||||
|
def stdout = new ByteArrayOutputStream()
|
||||||
|
exec {
|
||||||
commandLine 'git', 'rev-parse', '--short', 'HEAD'
|
commandLine 'git', 'rev-parse', '--short', 'HEAD'
|
||||||
}.standardOutput.getAsText().map { it.trim() }
|
standardOutput = stdout
|
||||||
|
}
|
||||||
def aw2at = Aw2AtTask.configureDefault(
|
return stdout.toString().trim()
|
||||||
getProject(),
|
|
||||||
layout.projectDirectory.file("src/main/resources/moonrise.accesswidener").getAsFile(),
|
|
||||||
sourceSets.main
|
|
||||||
)
|
|
||||||
|
|
||||||
neoForge {
|
|
||||||
neoFormVersion = neoform_version
|
|
||||||
validateAccessTransformers = true
|
|
||||||
accessTransformers.files.setFrom(aw2at.flatMap { t -> t.getOutputFile() })
|
|
||||||
}
|
}
|
||||||
|
|
||||||
runConfigCommon {
|
|
||||||
systemProperties.put "mixin.debug", "true"
|
|
||||||
systemProperties.put "Moonrise.MaxViewDistance", "128"
|
|
||||||
}
|
|
||||||
|
|
||||||
dependencies {
|
dependencies {
|
||||||
compileOnly "net.fabricmc:sponge-mixin:0.15.4+mixin.0.8.7"
|
modImplementation "net.fabricmc:fabric-loader:${project.loader_version}"
|
||||||
compileOnly "io.github.llamalad7:mixinextras-common:0.4.1"
|
|
||||||
// work around minecraft (MDG) forcing ASM 9.3 which is incompatible with the above deps...
|
|
||||||
components.withModule("net.neoforged:minecraft-dependencies", RemoveAsmConstraint.class)
|
|
||||||
|
|
||||||
api("ca.spottedleaf:concurrentutil:${rootProject.concurrentutil_version}") { setTransitive(false) }
|
api("ca.spottedleaf:concurrentutil:${rootProject.concurrentutil_version}")
|
||||||
api("ca.spottedleaf:yamlconfig:${rootProject.yamlconfig_version}") { setTransitive(false) }
|
|
||||||
api("org.yaml:snakeyaml:${rootProject.snakeyaml_version}")
|
api("org.yaml:snakeyaml:${rootProject.snakeyaml_version}")
|
||||||
|
|
||||||
// todo: does cloth publish a platform-agnostic jar in mojang mappings?
|
modImplementation "me.shedaniel.cloth:cloth-config:${rootProject.cloth_version}"
|
||||||
compileOnly "me.shedaniel.cloth:cloth-config-neoforge:${rootProject.cloth_version}"
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
File awFile = file("src/main/resources/moonrise.accesswidener")
|
||||||
|
|
||||||
allprojects {
|
allprojects {
|
||||||
group = rootProject.maven_group
|
group = rootProject.maven_group
|
||||||
version = rootProject.mod_version + "+" + getGitCommit.get()
|
version = rootProject.mod_version + "+" + getGitCommit()
|
||||||
|
|
||||||
plugins.apply("java-library")
|
plugins.apply("xyz.jpenilla.quiet-architectury-loom")
|
||||||
|
|
||||||
java {
|
java {
|
||||||
withSourcesJar()
|
withSourcesJar()
|
||||||
@@ -57,30 +44,25 @@ allprojects {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
dependencies {
|
|
||||||
testImplementation "org.junit.jupiter:junit-jupiter:${rootProject.junit_version}"
|
|
||||||
testRuntimeOnly 'org.junit.platform:junit-platform-launcher'
|
|
||||||
}
|
|
||||||
|
|
||||||
tasks.test {
|
|
||||||
useJUnitPlatform()
|
|
||||||
}
|
|
||||||
|
|
||||||
repositories {
|
repositories {
|
||||||
maven {
|
mavenLocal {
|
||||||
url = "https://repo.papermc.io/repository/maven-public/"
|
|
||||||
mavenContent {
|
mavenContent {
|
||||||
includeGroup("ca.spottedleaf")
|
includeModule("ca.spottedleaf", "concurrentutil")
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
maven {
|
maven {
|
||||||
url = "https://api.modrinth.com/maven"
|
url "https://api.modrinth.com/maven"
|
||||||
mavenContent {
|
mavenContent {
|
||||||
includeGroup("maven.modrinth")
|
includeGroup("maven.modrinth")
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
maven { url = "https://maven.shedaniel.me/" }
|
maven { url "https://maven.shedaniel.me/" }
|
||||||
maven { url = "https://maven.terraformersmc.com/releases/" }
|
maven { url "https://maven.terraformersmc.com/releases/" }
|
||||||
|
}
|
||||||
|
|
||||||
|
dependencies {
|
||||||
|
minecraft "com.mojang:minecraft:${project.minecraft_version}"
|
||||||
|
mappings loom.officialMojangMappings()
|
||||||
}
|
}
|
||||||
|
|
||||||
// make build reproducible
|
// make build reproducible
|
||||||
@@ -98,22 +80,31 @@ allprojects {
|
|||||||
rename { "${it}_${rootProject.base.archivesName.get()}"}
|
rename { "${it}_${rootProject.base.archivesName.get()}"}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
loom {
|
||||||
|
accessWidenerPath = awFile
|
||||||
|
mixin {
|
||||||
|
useLegacyMixinAp = false
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
subprojects {
|
subprojects {
|
||||||
plugins.apply("me.modmuss50.mod-publish-plugin")
|
loom.mods {
|
||||||
plugins.apply 'java-library'
|
main {
|
||||||
plugins.apply 'com.gradleup.shadow'
|
sourceSet("main")
|
||||||
|
sourceSet("main", project.rootProject)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
loom.runs.all {
|
||||||
|
ideConfigGenerated true
|
||||||
|
// property "mixin.debug", "true"
|
||||||
|
}
|
||||||
|
|
||||||
configurations.create("libs")
|
plugins.apply("me.modmuss50.mod-publish-plugin")
|
||||||
configurations.shadow {
|
|
||||||
extendsFrom(configurations.libs)
|
|
||||||
}
|
|
||||||
configurations.implementation {
|
|
||||||
extendsFrom(configurations.libs)
|
|
||||||
}
|
|
||||||
|
|
||||||
publishMods {
|
publishMods {
|
||||||
|
file = remapJar.archiveFile
|
||||||
if (project.version.contains("-beta.")) {
|
if (project.version.contains("-beta.")) {
|
||||||
type = ReleaseType.BETA
|
type = ReleaseType.BETA
|
||||||
} else {
|
} else {
|
||||||
@@ -137,7 +128,20 @@ subprojects {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// Setup a run with lithium for compatibility testing
|
// Setup a run with lithium for compatibility testing
|
||||||
|
sourceSets.create("lithium")
|
||||||
configurations.create("lithium")
|
configurations.create("lithium")
|
||||||
|
loom {
|
||||||
|
createRemapConfigurations(sourceSets.lithium)
|
||||||
|
runs {
|
||||||
|
register("lithiumClient") {
|
||||||
|
client()
|
||||||
|
property "mixin.debug", "true"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
tasks.named("runLithiumClient", net.fabricmc.loom.task.RunGameTask.class) {
|
||||||
|
getClasspath().from(configurations.modRuntimeClasspathLithiumMapped)
|
||||||
|
}
|
||||||
dependencies {
|
dependencies {
|
||||||
String coordinates = "maven.modrinth:lithium:"
|
String coordinates = "maven.modrinth:lithium:"
|
||||||
if (getProject().name == "Moonrise-NeoForge") {
|
if (getProject().name == "Moonrise-NeoForge") {
|
||||||
@@ -145,6 +149,10 @@ subprojects {
|
|||||||
} else {
|
} else {
|
||||||
coordinates += rootProject.fabric_lithium_version
|
coordinates += rootProject.fabric_lithium_version
|
||||||
}
|
}
|
||||||
lithium coordinates
|
modLithiumRuntimeOnly coordinates
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
loom.runs.all {
|
||||||
|
ideConfigGenerated false
|
||||||
|
}
|
||||||
|
|||||||
@@ -1,11 +0,0 @@
|
|||||||
repositories {
|
|
||||||
gradlePluginPortal()
|
|
||||||
mavenCentral()
|
|
||||||
maven("https://maven.fabricmc.net/")
|
|
||||||
maven("https://maven.architectury.dev/")
|
|
||||||
}
|
|
||||||
|
|
||||||
dependencies {
|
|
||||||
implementation("net.fabricmc:access-widener:2.1.0")
|
|
||||||
implementation("dev.architectury:at:1.0.1")
|
|
||||||
}
|
|
||||||
@@ -1,152 +0,0 @@
|
|||||||
import dev.architectury.at.AccessChange;
|
|
||||||
import dev.architectury.at.AccessTransform;
|
|
||||||
import dev.architectury.at.AccessTransformSet;
|
|
||||||
import dev.architectury.at.ModifierChange;
|
|
||||||
import dev.architectury.at.io.AccessTransformFormat;
|
|
||||||
import dev.architectury.at.io.AccessTransformFormats;
|
|
||||||
import java.io.BufferedReader;
|
|
||||||
import java.io.BufferedWriter;
|
|
||||||
import java.io.File;
|
|
||||||
import java.io.IOException;
|
|
||||||
import java.io.StringWriter;
|
|
||||||
import java.nio.file.Files;
|
|
||||||
import java.nio.file.Path;
|
|
||||||
import java.util.Arrays;
|
|
||||||
import java.util.List;
|
|
||||||
import javax.inject.Inject;
|
|
||||||
import net.fabricmc.accesswidener.AccessWidenerReader;
|
|
||||||
import net.fabricmc.accesswidener.AccessWidenerVisitor;
|
|
||||||
import org.cadixdev.bombe.type.signature.MethodSignature;
|
|
||||||
import org.gradle.api.DefaultTask;
|
|
||||||
import org.gradle.api.Project;
|
|
||||||
import org.gradle.api.file.ProjectLayout;
|
|
||||||
import org.gradle.api.file.RegularFileProperty;
|
|
||||||
import org.gradle.api.tasks.CacheableTask;
|
|
||||||
import org.gradle.api.tasks.InputFile;
|
|
||||||
import org.gradle.api.tasks.OutputFile;
|
|
||||||
import org.gradle.api.tasks.PathSensitive;
|
|
||||||
import org.gradle.api.tasks.PathSensitivity;
|
|
||||||
import org.gradle.api.tasks.SourceSet;
|
|
||||||
import org.gradle.api.tasks.TaskAction;
|
|
||||||
import org.gradle.api.tasks.TaskProvider;
|
|
||||||
|
|
||||||
@CacheableTask
|
|
||||||
public abstract class Aw2AtTask extends DefaultTask {
|
|
||||||
|
|
||||||
@InputFile
|
|
||||||
@PathSensitive(PathSensitivity.NONE)
|
|
||||||
public abstract RegularFileProperty getInputFile();
|
|
||||||
|
|
||||||
@OutputFile
|
|
||||||
public abstract RegularFileProperty getOutputFile();
|
|
||||||
|
|
||||||
@Inject
|
|
||||||
public abstract ProjectLayout getLayout();
|
|
||||||
|
|
||||||
public static TaskProvider<Aw2AtTask> configureDefault(
|
|
||||||
final Project project,
|
|
||||||
final File awFile,
|
|
||||||
final SourceSet sourceSet
|
|
||||||
) {
|
|
||||||
final TaskProvider<Aw2AtTask> aw2at = project.getTasks().register("aw2at", Aw2AtTask.class, task -> {
|
|
||||||
task.getOutputFile().set(project.getLayout().getBuildDirectory().file("aw2at/files/accesstransformer.cfg"));
|
|
||||||
task.getInputFile().set(awFile);
|
|
||||||
});
|
|
||||||
|
|
||||||
final TaskProvider<CopyTask> copyTask = project.getTasks().register("copyAt", CopyTask.class, copy -> {
|
|
||||||
copy.getInputFile().set(aw2at.flatMap(Aw2AtTask::getOutputFile));
|
|
||||||
copy.getOutputDirectory().set(project.getLayout().getBuildDirectory().dir("aw2at/dir"));
|
|
||||||
copy.getDestination().set("META-INF/accesstransformer.cfg");
|
|
||||||
});
|
|
||||||
|
|
||||||
sourceSet.resources(resources -> {
|
|
||||||
resources.srcDir(copyTask.flatMap(CopyTask::getOutputDirectory));
|
|
||||||
});
|
|
||||||
|
|
||||||
return aw2at;
|
|
||||||
}
|
|
||||||
|
|
||||||
@TaskAction
|
|
||||||
public void run() {
|
|
||||||
try (final BufferedReader reader = Files.newBufferedReader(this.getInputFile().get().getAsFile().toPath())) {
|
|
||||||
final AccessTransformSet accessTransformSet = toAccessTransformSet(reader);
|
|
||||||
Files.deleteIfExists(this.getOutputFile().get().getAsFile().toPath());
|
|
||||||
Files.createDirectories(this.getOutputFile().get().getAsFile().toPath().getParent());
|
|
||||||
writeLF(AccessTransformFormats.FML, this.getOutputFile().get().getAsFile().toPath(), accessTransformSet);
|
|
||||||
} catch (final IOException e) {
|
|
||||||
throw new RuntimeException(e);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
private static void writeLF(final AccessTransformFormat format, final Path path, final AccessTransformSet at) throws IOException {
|
|
||||||
final StringWriter stringWriter = new StringWriter();
|
|
||||||
final BufferedWriter writer = new BufferedWriter(stringWriter);
|
|
||||||
format.write(writer, at);
|
|
||||||
writer.close();
|
|
||||||
final List<String> lines = Arrays.stream(stringWriter.toString()
|
|
||||||
// unify line endings
|
|
||||||
.replace("\r\n", "\n")
|
|
||||||
.split("\n"))
|
|
||||||
// skip blank lines
|
|
||||||
.filter(it -> !it.isBlank())
|
|
||||||
// sort
|
|
||||||
.sorted()
|
|
||||||
.toList();
|
|
||||||
Files.writeString(path, String.join("\n", lines));
|
|
||||||
}
|
|
||||||
|
|
||||||
// Below methods are heavily based on architectury-loom Aw2At class (MIT licensed)
|
|
||||||
/*
|
|
||||||
MIT License
|
|
||||||
|
|
||||||
Copyright (c) 2016 FabricMC
|
|
||||||
|
|
||||||
Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
||||||
of this software and associated documentation files (the "Software"), to deal
|
|
||||||
in the Software without restriction, including without limitation the rights
|
|
||||||
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
||||||
copies of the Software, and to permit persons to whom the Software is
|
|
||||||
furnished to do so, subject to the following conditions:
|
|
||||||
|
|
||||||
The above copyright notice and this permission notice shall be included in all
|
|
||||||
copies or substantial portions of the Software.
|
|
||||||
|
|
||||||
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
||||||
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
||||||
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
||||||
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
||||||
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
||||||
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
|
||||||
SOFTWARE.
|
|
||||||
*/
|
|
||||||
|
|
||||||
public static AccessTransformSet toAccessTransformSet(final BufferedReader reader) throws IOException {
|
|
||||||
AccessTransformSet atSet = AccessTransformSet.create();
|
|
||||||
|
|
||||||
new AccessWidenerReader(new AccessWidenerVisitor() {
|
|
||||||
@Override
|
|
||||||
public void visitClass(final String name, final AccessWidenerReader.AccessType access, final boolean transitive) {
|
|
||||||
atSet.getOrCreateClass(name).merge(toAt(access));
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public void visitMethod(final String owner, final String name, final String descriptor, final AccessWidenerReader.AccessType access, final boolean transitive) {
|
|
||||||
atSet.getOrCreateClass(owner).mergeMethod(MethodSignature.of(name, descriptor), toAt(access));
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public void visitField(final String owner, final String name, final String descriptor, final AccessWidenerReader.AccessType access, final boolean transitive) {
|
|
||||||
atSet.getOrCreateClass(owner).mergeField(name, toAt(access));
|
|
||||||
}
|
|
||||||
}).read(reader);
|
|
||||||
|
|
||||||
return atSet;
|
|
||||||
}
|
|
||||||
|
|
||||||
public static AccessTransform toAt(final AccessWidenerReader.AccessType access) {
|
|
||||||
return switch (access) {
|
|
||||||
case ACCESSIBLE -> AccessTransform.of(AccessChange.PUBLIC);
|
|
||||||
case EXTENDABLE, MUTABLE -> AccessTransform.of(AccessChange.PUBLIC, ModifierChange.REMOVE);
|
|
||||||
};
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,44 +0,0 @@
|
|||||||
import java.io.IOException;
|
|
||||||
import java.nio.file.Files;
|
|
||||||
import java.nio.file.Path;
|
|
||||||
import java.util.Comparator;
|
|
||||||
import java.util.stream.Stream;
|
|
||||||
import org.gradle.api.DefaultTask;
|
|
||||||
import org.gradle.api.file.DirectoryProperty;
|
|
||||||
import org.gradle.api.file.RegularFileProperty;
|
|
||||||
import org.gradle.api.provider.Property;
|
|
||||||
import org.gradle.api.tasks.Input;
|
|
||||||
import org.gradle.api.tasks.InputFile;
|
|
||||||
import org.gradle.api.tasks.OutputDirectory;
|
|
||||||
import org.gradle.api.tasks.TaskAction;
|
|
||||||
|
|
||||||
public abstract class CopyTask extends DefaultTask {
|
|
||||||
|
|
||||||
@InputFile
|
|
||||||
public abstract RegularFileProperty getInputFile();
|
|
||||||
|
|
||||||
@Input
|
|
||||||
public abstract Property<String> getDestination();
|
|
||||||
|
|
||||||
@OutputDirectory
|
|
||||||
public abstract DirectoryProperty getOutputDirectory();
|
|
||||||
|
|
||||||
@TaskAction
|
|
||||||
public void run() {
|
|
||||||
final Path outputDirPath = this.getOutputDirectory().get().getAsFile().toPath();
|
|
||||||
try {
|
|
||||||
try (final Stream<Path> walk = Files.walk(outputDirPath)) {
|
|
||||||
for (final Path path : walk.sorted(Comparator.reverseOrder()).toList()) {
|
|
||||||
Files.delete(path);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
final Path destFile = outputDirPath.resolve(this.getDestination().get());
|
|
||||||
Files.createDirectories(destFile.getParent());
|
|
||||||
|
|
||||||
Files.copy(this.getInputFile().get().getAsFile().toPath(), destFile);
|
|
||||||
} catch (final IOException e) {
|
|
||||||
throw new RuntimeException(e);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,24 +0,0 @@
|
|||||||
import java.util.Objects;
|
|
||||||
import org.gradle.api.artifacts.CacheableRule;
|
|
||||||
import org.gradle.api.artifacts.ComponentMetadataContext;
|
|
||||||
import org.gradle.api.artifacts.ComponentMetadataRule;
|
|
||||||
|
|
||||||
@CacheableRule
|
|
||||||
public abstract class RemoveAsmConstraint implements ComponentMetadataRule {
|
|
||||||
@Override
|
|
||||||
public void execute(final ComponentMetadataContext ctx) {
|
|
||||||
ctx.getDetails().allVariants(variants -> {
|
|
||||||
variants.withDependencies(deps -> {
|
|
||||||
deps.forEach(dep -> {
|
|
||||||
if (Objects.equals(dep.getGroup(), "org.ow2.asm")) {
|
|
||||||
if (dep.getVersionConstraint().getStrictVersion() != null) {
|
|
||||||
dep.version(v -> {
|
|
||||||
v.require(v.getStrictVersion());
|
|
||||||
});
|
|
||||||
}
|
|
||||||
}
|
|
||||||
});
|
|
||||||
});
|
|
||||||
});
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,5 +0,0 @@
|
|||||||
import org.gradle.api.provider.MapProperty;
|
|
||||||
|
|
||||||
public abstract class RunConfigCommon {
|
|
||||||
public abstract MapProperty<String, String> getSystemProperties();
|
|
||||||
}
|
|
||||||
@@ -1,60 +1,52 @@
|
|||||||
import java.util.stream.Collectors
|
|
||||||
import net.fabricmc.loom.util.gradle.SourceSetHelper
|
|
||||||
|
|
||||||
plugins {
|
plugins {
|
||||||
id("quiet-fabric-loom")
|
id("xyz.jpenilla.quiet-architectury-loom")
|
||||||
id 'maven-publish'
|
id 'maven-publish'
|
||||||
|
id 'com.gradleup.shadow'
|
||||||
|
}
|
||||||
|
|
||||||
|
configurations.create("libs")
|
||||||
|
configurations.shadow {
|
||||||
|
extendsFrom(configurations.libs)
|
||||||
|
}
|
||||||
|
configurations.implementation {
|
||||||
|
extendsFrom(configurations.libs)
|
||||||
}
|
}
|
||||||
|
|
||||||
dependencies {
|
dependencies {
|
||||||
minecraft "com.mojang:minecraft:${project.minecraft_version}"
|
add('shadow', project([path: ":", configuration: "namedElements"]))
|
||||||
mappings loom.officialMojangMappings()
|
|
||||||
modImplementation "net.fabricmc:fabric-loader:${project.loader_version}"
|
|
||||||
testImplementation "net.fabricmc:fabric-loader-junit:${project.loader_version}"
|
|
||||||
|
|
||||||
runtimeOnly(project(":").sourceSets.main.output)
|
runtimeOnly(project(":").sourceSets.main.output)
|
||||||
shadow(project(":"))
|
modImplementation "net.fabricmc:fabric-loader:${project.loader_version}"
|
||||||
compileOnly(project(":"))
|
|
||||||
|
|
||||||
libs("ca.spottedleaf:concurrentutil:${rootProject.concurrentutil_version}") { setTransitive(false) }
|
libs("ca.spottedleaf:concurrentutil:${rootProject.concurrentutil_version}")
|
||||||
libs("ca.spottedleaf:yamlconfig:${rootProject.yamlconfig_version}") { setTransitive(false) }
|
|
||||||
libs("org.yaml:snakeyaml:${rootProject.snakeyaml_version}")
|
libs("org.yaml:snakeyaml:${rootProject.snakeyaml_version}")
|
||||||
|
|
||||||
modImplementation "me.shedaniel.cloth:cloth-config-fabric:${rootProject.cloth_version}"
|
modImplementation "me.shedaniel.cloth:cloth-config-fabric:${rootProject.cloth_version}"
|
||||||
include "me.shedaniel.cloth:cloth-config-fabric:${rootProject.cloth_version}"
|
include "me.shedaniel.cloth:cloth-config-fabric:${rootProject.cloth_version}"
|
||||||
modImplementation "com.terraformersmc:modmenu:${rootProject.modmenu_version}"
|
modImplementation "com.terraformersmc:modmenu:11.0.1"
|
||||||
|
|
||||||
modImplementation platform(fabricApiLibs.bom)
|
|
||||||
modImplementation fabricApiLibs.command.api.v2
|
modImplementation fabricApiLibs.command.api.v2
|
||||||
modImplementation fabricApiLibs.lifecycle.events.v1
|
modImplementation fabricApiLibs.lifecycle.events.v1
|
||||||
include fabricApiLibs.command.api.v2
|
include fabricApiLibs.command.api.v2
|
||||||
include fabricApiLibs.base
|
include fabricApiLibs.base
|
||||||
}
|
}
|
||||||
|
|
||||||
tasks.processResources {
|
processResources {
|
||||||
def properties = [
|
inputs.property "version", project.version
|
||||||
"version": project.version,
|
|
||||||
"minecraft_version": minecraft_version,
|
|
||||||
"loader_version": loader_version,
|
|
||||||
"mod_version": mod_version
|
|
||||||
]
|
|
||||||
inputs.properties(properties)
|
|
||||||
filesMatching("fabric.mod.json") {
|
filesMatching("fabric.mod.json") {
|
||||||
expand properties
|
expand "version": project.version, "minecraft_version": minecraft_version, "loader_version": loader_version, "mod_version": mod_version
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
tasks.shadowJar {
|
shadowJar {
|
||||||
archiveClassifier = "dev-all"
|
archiveClassifier = "dev-all"
|
||||||
destinationDirectory = layout.buildDirectory.dir("libs")
|
destinationDirectory = layout.buildDirectory.dir("libs")
|
||||||
configurations = [project.configurations.shadow]
|
configurations = [project.configurations.shadow]
|
||||||
relocate 'ca.spottedleaf.concurrentutil', 'ca.spottedleaf.moonrise.libs.ca.spottedleaf.concurrentutil'
|
relocate 'ca.spottedleaf.concurrentutil', 'ca.spottedleaf.moonrise.libs.ca.spottedleaf.concurrentutil'
|
||||||
relocate 'ca.spottedleaf.yamlconfig', 'ca.spottedleaf.moonrise.libs.ca.spottedleaf.yamlconfig'
|
|
||||||
relocate 'org.yaml.snakeyaml', 'ca.spottedleaf.moonrise.libs.org.yaml.snakeyaml'
|
relocate 'org.yaml.snakeyaml', 'ca.spottedleaf.moonrise.libs.org.yaml.snakeyaml'
|
||||||
}
|
}
|
||||||
|
|
||||||
publishMods {
|
publishMods {
|
||||||
file = remapJar.archiveFile
|
|
||||||
modLoaders = ["fabric"]
|
modLoaders = ["fabric"]
|
||||||
|
|
||||||
modrinth {
|
modrinth {
|
||||||
@@ -72,52 +64,3 @@ publishMods {
|
|||||||
)
|
)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
loom {
|
|
||||||
accessWidenerPath.set(getRootProject().file("src/main/resources/moonrise.accesswidener"))
|
|
||||||
mixin {
|
|
||||||
useLegacyMixinAp = false
|
|
||||||
}
|
|
||||||
runs.configureEach {
|
|
||||||
ideConfigGenerated true
|
|
||||||
}
|
|
||||||
mods {
|
|
||||||
main {
|
|
||||||
sourceSet("main")
|
|
||||||
sourceSet("main", project.rootProject)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
tasks.test {
|
|
||||||
def classPathGroups = SourceSetHelper.getClasspath(loom.mods.main, getProject()).stream()
|
|
||||||
.map(File.&getAbsolutePath)
|
|
||||||
.collect(Collectors.joining(File.pathSeparator))
|
|
||||||
|
|
||||||
systemProperty("fabric.classPathGroups", classPathGroups)
|
|
||||||
}
|
|
||||||
|
|
||||||
afterEvaluate {
|
|
||||||
loom.runs.configureEach { cfg ->
|
|
||||||
runConfigCommon.systemProperties.get().each {
|
|
||||||
cfg.property it.key, it.value
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Setup a run with lithium for compatibility testing
|
|
||||||
sourceSets.create("lithium")
|
|
||||||
loom {
|
|
||||||
createRemapConfigurations(sourceSets.lithium)
|
|
||||||
runs {
|
|
||||||
register("lithiumClient") {
|
|
||||||
client()
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
configurations.modLithiumRuntimeOnly {
|
|
||||||
extendsFrom configurations.lithium
|
|
||||||
}
|
|
||||||
tasks.named("runLithiumClient", net.fabricmc.loom.task.RunGameTask.class) {
|
|
||||||
getClasspath().from(configurations.modRuntimeClasspathLithiumMapped)
|
|
||||||
}
|
|
||||||
|
|||||||
@@ -1,45 +1,50 @@
|
|||||||
package ca.spottedleaf.moonrise.fabric;
|
package ca.spottedleaf.moonrise.fabric;
|
||||||
|
|
||||||
import ca.spottedleaf.moonrise.common.util.BaseChunkSystemHooks;
|
|
||||||
import ca.spottedleaf.moonrise.common.PlatformHooks;
|
import ca.spottedleaf.moonrise.common.PlatformHooks;
|
||||||
import ca.spottedleaf.moonrise.common.util.ConfigHolder;
|
import ca.spottedleaf.moonrise.common.util.ConfigHolder;
|
||||||
import ca.spottedleaf.moonrise.patches.chunk_system.ticket.ChunkSystemTicketType;
|
import ca.spottedleaf.moonrise.patches.chunk_system.scheduling.NewChunkHolder;
|
||||||
import com.mojang.datafixers.DSL;
|
import net.fabricmc.fabric.api.event.Event;
|
||||||
import com.mojang.datafixers.DataFixer;
|
import net.fabricmc.fabric.api.event.EventFactory;
|
||||||
import com.mojang.serialization.Dynamic;
|
|
||||||
import it.unimi.dsi.fastutil.longs.LongArrays;
|
|
||||||
import net.fabricmc.fabric.api.event.lifecycle.v1.ServerChunkEvents;
|
import net.fabricmc.fabric.api.event.lifecycle.v1.ServerChunkEvents;
|
||||||
import net.fabricmc.loader.api.FabricLoader;
|
import net.fabricmc.loader.api.FabricLoader;
|
||||||
import net.minecraft.core.BlockPos;
|
import net.minecraft.core.BlockPos;
|
||||||
import net.minecraft.nbt.CompoundTag;
|
import net.minecraft.nbt.CompoundTag;
|
||||||
import net.minecraft.nbt.NbtOps;
|
|
||||||
import net.minecraft.server.level.ChunkHolder;
|
|
||||||
import net.minecraft.server.level.GenerationChunkHolder;
|
import net.minecraft.server.level.GenerationChunkHolder;
|
||||||
import net.minecraft.server.level.ServerLevel;
|
import net.minecraft.server.level.ServerLevel;
|
||||||
import net.minecraft.server.level.ServerPlayer;
|
import net.minecraft.server.level.ServerPlayer;
|
||||||
import net.minecraft.server.level.TicketType;
|
|
||||||
import net.minecraft.world.entity.Entity;
|
import net.minecraft.world.entity.Entity;
|
||||||
import net.minecraft.world.entity.boss.EnderDragonPart;
|
|
||||||
import net.minecraft.world.level.BlockGetter;
|
import net.minecraft.world.level.BlockGetter;
|
||||||
import net.minecraft.world.level.ChunkPos;
|
import net.minecraft.world.level.ChunkPos;
|
||||||
|
import net.minecraft.world.level.Explosion;
|
||||||
import net.minecraft.world.level.Level;
|
import net.minecraft.world.level.Level;
|
||||||
import net.minecraft.world.level.block.state.BlockState;
|
import net.minecraft.world.level.block.state.BlockState;
|
||||||
import net.minecraft.world.level.chunk.ChunkAccess;
|
import net.minecraft.world.level.chunk.ChunkAccess;
|
||||||
import net.minecraft.world.level.chunk.ImposterProtoChunk;
|
import net.minecraft.world.level.chunk.ImposterProtoChunk;
|
||||||
import net.minecraft.world.level.chunk.LevelChunk;
|
import net.minecraft.world.level.chunk.LevelChunk;
|
||||||
import net.minecraft.world.level.chunk.ProtoChunk;
|
import net.minecraft.world.level.chunk.ProtoChunk;
|
||||||
import net.minecraft.world.level.chunk.status.ChunkStatusTasks;
|
|
||||||
import net.minecraft.world.level.chunk.storage.SerializableChunkData;
|
|
||||||
import net.minecraft.world.level.entity.EntityTypeTest;
|
import net.minecraft.world.level.entity.EntityTypeTest;
|
||||||
import net.minecraft.world.phys.AABB;
|
import net.minecraft.world.phys.AABB;
|
||||||
import java.util.Collection;
|
import net.minecraft.world.phys.Vec3;
|
||||||
import java.util.List;
|
import java.util.List;
|
||||||
import java.util.function.Predicate;
|
import java.util.function.Predicate;
|
||||||
|
|
||||||
public final class FabricHooks extends BaseChunkSystemHooks implements PlatformHooks {
|
public final class FabricHooks implements PlatformHooks {
|
||||||
|
|
||||||
private static final boolean HAS_FABRIC_LIFECYCLE_EVENTS = FabricLoader.getInstance().isModLoaded("fabric-lifecycle-events-v1");
|
private static final boolean HAS_FABRIC_LIFECYCLE_EVENTS = FabricLoader.getInstance().isModLoaded("fabric-lifecycle-events-v1");
|
||||||
|
|
||||||
|
public interface OnExplosionDetonate {
|
||||||
|
void onExplosion(final Level world, final Explosion explosion, final List<Entity> possiblyAffecting, final double diameter);
|
||||||
|
}
|
||||||
|
|
||||||
|
public static final Event<OnExplosionDetonate> ON_EXPLOSION_DETONATE = EventFactory.createArrayBacked(
|
||||||
|
OnExplosionDetonate.class,
|
||||||
|
listeners -> (final Level world, final Explosion explosion, final List<Entity> possiblyAffecting, final double diameter) -> {
|
||||||
|
for (int i = 0; i < listeners.length; i++) {
|
||||||
|
listeners[i].onExplosion(world, explosion, possiblyAffecting, diameter);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
);
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public String getBrand() {
|
public String getBrand() {
|
||||||
return "Moonrise";
|
return "Moonrise";
|
||||||
@@ -57,6 +62,16 @@ public final class FabricHooks extends BaseChunkSystemHooks implements PlatformH
|
|||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void onExplosion(final Level world, final Explosion explosion, final List<Entity> possiblyAffecting, final double diameter) {
|
||||||
|
ON_EXPLOSION_DETONATE.invoker().onExplosion(world, explosion, possiblyAffecting, diameter);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public Vec3 modifyExplosionKnockback(final Level world, final Explosion explosion, final Entity entity, final Vec3 original) {
|
||||||
|
return original;
|
||||||
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public boolean hasCurrentlyLoadingChunk() {
|
public boolean hasCurrentlyLoadingChunk() {
|
||||||
return false;
|
return false;
|
||||||
@@ -88,7 +103,7 @@ public final class FabricHooks extends BaseChunkSystemHooks implements PlatformH
|
|||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void onChunkHolderTicketChange(final ServerLevel world, final ChunkHolder holder, final int oldLevel, final int newLevel) {
|
public void onChunkHolderTicketChange(final ServerLevel world, final NewChunkHolder holder, final int oldLevel, final int newLevel) {
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -100,7 +115,7 @@ public final class FabricHooks extends BaseChunkSystemHooks implements PlatformH
|
|||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void chunkSyncSave(final ServerLevel world, final ChunkAccess chunk, final SerializableChunkData data) {
|
public void chunkSyncSave(final ServerLevel world, final ChunkAccess chunk, final CompoundTag data) {
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -117,43 +132,13 @@ public final class FabricHooks extends BaseChunkSystemHooks implements PlatformH
|
|||||||
@Override
|
@Override
|
||||||
public void addToGetEntities(final Level world, final Entity entity, final AABB boundingBox, final Predicate<? super Entity> predicate,
|
public void addToGetEntities(final Level world, final Entity entity, final AABB boundingBox, final Predicate<? super Entity> predicate,
|
||||||
final List<Entity> into) {
|
final List<Entity> into) {
|
||||||
final Collection<EnderDragonPart> parts = world.dragonParts();
|
|
||||||
if (parts.isEmpty()) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
for (final EnderDragonPart part : parts) {
|
|
||||||
if (part != entity && part.getBoundingBox().intersects(boundingBox) && (predicate == null || predicate.test(part))) {
|
|
||||||
into.add(part);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public <T extends Entity> void addToGetEntities(final Level world, final EntityTypeTest<Entity, T> entityTypeTest, final AABB boundingBox,
|
public <T extends Entity> void addToGetEntities(final Level world, final EntityTypeTest<Entity, T> entityTypeTest, final AABB boundingBox,
|
||||||
final Predicate<? super T> predicate, final List<? super T> into, final int maxCount) {
|
final Predicate<? super T> predicate, final List<? super T> into, final int maxCount) {
|
||||||
if (into.size() >= maxCount) {
|
|
||||||
// fix neoforge issue: do not add if list is already full
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
final Collection<EnderDragonPart> parts = world.dragonParts();
|
|
||||||
if (parts.isEmpty()) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
for (final EnderDragonPart part : parts) {
|
|
||||||
if (!part.getBoundingBox().intersects(boundingBox)) {
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
final T casted = (T)entityTypeTest.tryCast(part);
|
|
||||||
if (casted != null && (predicate == null || predicate.test(casted))) {
|
|
||||||
into.add(casted);
|
|
||||||
if (into.size() >= maxCount) {
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
@@ -202,12 +187,12 @@ public final class FabricHooks extends BaseChunkSystemHooks implements PlatformH
|
|||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public long configAutoSaveInterval(final ServerLevel world) {
|
public long configAutoSaveInterval() {
|
||||||
return ConfigHolder.getConfig().chunkSaving.autoSaveInterval.getTimeTicks();
|
return ConfigHolder.getConfig().chunkSaving.autoSaveInterval.getTimeTicks();
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public int configMaxAutoSavePerTick(final ServerLevel world) {
|
public int configMaxAutoSavePerTick() {
|
||||||
return ConfigHolder.getConfig().chunkSaving.maxAutoSaveChunksPerTick;
|
return ConfigHolder.getConfig().chunkSaving.maxAutoSaveChunksPerTick;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -215,52 +200,4 @@ public final class FabricHooks extends BaseChunkSystemHooks implements PlatformH
|
|||||||
public boolean configFixMC159283() {
|
public boolean configFixMC159283() {
|
||||||
return ConfigHolder.getConfig().bugFixes.fixMC159283;
|
return ConfigHolder.getConfig().bugFixes.fixMC159283;
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
|
||||||
public boolean forceNoSave(final ChunkAccess chunk) {
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public CompoundTag convertNBT(final DSL.TypeReference type, final DataFixer dataFixer, final CompoundTag nbt,
|
|
||||||
final int fromVersion, final int toVersion) {
|
|
||||||
return (CompoundTag)dataFixer.update(
|
|
||||||
type, new Dynamic<>(NbtOps.INSTANCE, nbt), fromVersion, toVersion
|
|
||||||
).getValue();
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public boolean hasMainChunkLoadHook() {
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public void mainChunkLoad(final ChunkAccess chunk, final SerializableChunkData chunkData) {
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public List<Entity> modifySavedEntities(final ServerLevel world, final int chunkX, final int chunkZ, final List<Entity> entities) {
|
|
||||||
return entities;
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public void unloadEntity(final Entity entity) {
|
|
||||||
entity.setRemoved(Entity.RemovalReason.UNLOADED_TO_CHUNK);
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public void postLoadProtoChunk(final ServerLevel world, final ProtoChunk chunk) {
|
|
||||||
ChunkStatusTasks.postLoadProtoChunk(world, chunk.getEntities());
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public int modifyEntityTrackingRange(final Entity entity, final int currentRange) {
|
|
||||||
return currentRange;
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public long[] getCounterTypesUncached(final TicketType type) {
|
|
||||||
return type == TicketType.FORCED ? new long[] { ChunkSystemTicketType.COUNTER_TYPE_FORCED } : LongArrays.EMPTY_ARRAY;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -0,0 +1,33 @@
|
|||||||
|
package ca.spottedleaf.moonrise.fabric.mixin.chunk_system;
|
||||||
|
|
||||||
|
import ca.spottedleaf.moonrise.patches.chunk_system.level.chunk.ChunkSystemDistanceManager;
|
||||||
|
import net.minecraft.server.level.ChunkLevel;
|
||||||
|
import net.minecraft.server.level.DistanceManager;
|
||||||
|
import net.minecraft.server.level.FullChunkStatus;
|
||||||
|
import net.minecraft.server.level.TicketType;
|
||||||
|
import net.minecraft.world.level.ChunkPos;
|
||||||
|
import org.spongepowered.asm.mixin.Mixin;
|
||||||
|
import org.spongepowered.asm.mixin.Overwrite;
|
||||||
|
|
||||||
|
@Mixin(DistanceManager.class)
|
||||||
|
abstract class FabricDistanceManagerMixin implements ChunkSystemDistanceManager {
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @reason Route to new chunk system
|
||||||
|
* @author Spottedleaf
|
||||||
|
*/
|
||||||
|
@Overwrite
|
||||||
|
public <T> void addRegionTicket(final TicketType<T> type, final ChunkPos pos, final int radius, final T identifier) {
|
||||||
|
this.moonrise$getChunkHolderManager().addTicketAtLevel(type, pos, ChunkLevel.byStatus(FullChunkStatus.FULL) - radius, identifier);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @reason Route to new chunk system
|
||||||
|
* @author Spottedleaf
|
||||||
|
*/
|
||||||
|
@Overwrite
|
||||||
|
public <T> void removeRegionTicket(final TicketType<T> type, final ChunkPos pos, final int radius, final T identifier) {
|
||||||
|
this.moonrise$getChunkHolderManager().removeTicketAtLevel(type, pos, ChunkLevel.byStatus(FullChunkStatus.FULL) - radius, identifier);
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
@@ -36,12 +36,13 @@
|
|||||||
"accessWidener": "moonrise.accesswidener",
|
"accessWidener": "moonrise.accesswidener",
|
||||||
"depends": {
|
"depends": {
|
||||||
"fabricloader": ">=${loader_version}",
|
"fabricloader": ">=${loader_version}",
|
||||||
"minecraft": ">1.21.4 <1.21.6",
|
"minecraft": ">=1.21 <=1.21.1",
|
||||||
"fabric-command-api-v2": "*"
|
"fabric-command-api-v2": "*"
|
||||||
},
|
},
|
||||||
"custom": {
|
"custom": {
|
||||||
"lithium:options": {
|
"lithium:options": {
|
||||||
"mixin.ai.poi": false,
|
"mixin.ai.poi": false,
|
||||||
|
"mixin.alloc.chunk_ticking": false,
|
||||||
"mixin.alloc.deep_passengers": false,
|
"mixin.alloc.deep_passengers": false,
|
||||||
"mixin.alloc.entity_tracker": false,
|
"mixin.alloc.entity_tracker": false,
|
||||||
"mixin.block.flatten_states": false,
|
"mixin.block.flatten_states": false,
|
||||||
@@ -65,6 +66,7 @@
|
|||||||
"mixin.world.block_entity_ticking": false,
|
"mixin.world.block_entity_ticking": false,
|
||||||
"mixin.world.chunk_access": false,
|
"mixin.world.chunk_access": false,
|
||||||
"mixin.world.explosions.block_raycast": false,
|
"mixin.world.explosions.block_raycast": false,
|
||||||
|
"mixin.world.explosions.cache_exposure": false,
|
||||||
"mixin.world.temperature_cache": false,
|
"mixin.world.temperature_cache": false,
|
||||||
"mixin.world.tick_scheduler": false
|
"mixin.world.tick_scheduler": false
|
||||||
},
|
},
|
||||||
|
|||||||
@@ -2,6 +2,7 @@
|
|||||||
"parent": "moonrise.mixins.json",
|
"parent": "moonrise.mixins.json",
|
||||||
"package": "ca.spottedleaf.moonrise.fabric.mixin",
|
"package": "ca.spottedleaf.moonrise.fabric.mixin",
|
||||||
"mixins": [
|
"mixins": [
|
||||||
|
"chunk_system.FabricDistanceManagerMixin",
|
||||||
"chunk_system.FabricMinecraftServerMixin",
|
"chunk_system.FabricMinecraftServerMixin",
|
||||||
"chunk_system.FabricServerLevelMixin",
|
"chunk_system.FabricServerLevelMixin",
|
||||||
"collisions.EntityMixin"
|
"collisions.EntityMixin"
|
||||||
|
|||||||
@@ -1,20 +0,0 @@
|
|||||||
package ca.spottedleaf.moonrise.fabric;
|
|
||||||
|
|
||||||
import net.minecraft.SharedConstants;
|
|
||||||
import net.minecraft.server.Bootstrap;
|
|
||||||
import org.junit.jupiter.api.BeforeAll;
|
|
||||||
import org.junit.jupiter.api.Test;
|
|
||||||
import org.spongepowered.asm.mixin.MixinEnvironment;
|
|
||||||
|
|
||||||
class MixinAuditTest {
|
|
||||||
@BeforeAll
|
|
||||||
static void beforeAll() {
|
|
||||||
SharedConstants.tryDetectVersion();
|
|
||||||
Bootstrap.bootStrap();
|
|
||||||
}
|
|
||||||
|
|
||||||
@Test
|
|
||||||
void auditMixins() {
|
|
||||||
MixinEnvironment.getCurrentEnvironment().audit();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,26 +1,19 @@
|
|||||||
# Done to increase the memory available to gradle.
|
# Done to increase the memory available to gradle.
|
||||||
org.gradle.jvmargs=-Xmx2G
|
org.gradle.jvmargs=-Xmx2G
|
||||||
org.gradle.parallel=true
|
org.gradle.daemon=false
|
||||||
org.gradle.caching=true
|
|
||||||
org.gradle.configuration-cache=true
|
|
||||||
# Fabric Properties
|
# Fabric Properties
|
||||||
# check these on https://modmuss50.me/fabric.html
|
# check these on https://modmuss50.me/fabric.html
|
||||||
minecraft_version=1.21.5
|
minecraft_version=1.21.1
|
||||||
loader_version=0.16.14
|
supported_minecraft_versions=1.21,1.21.1
|
||||||
supported_minecraft_versions=1.21.5
|
loader_version=0.16.5
|
||||||
neoforge_version=21.5.65-beta
|
neoforge_version=21.1.79
|
||||||
neoform_version=1.21.5-20250325.162830
|
snakeyaml_version=2.2
|
||||||
fabric_api_version=0.123.0+1.21.5
|
concurrentutil_version=0.0.2-SNAPSHOT
|
||||||
snakeyaml_version=2.3
|
cloth_version=15.0.128
|
||||||
concurrentutil_version=0.0.3
|
|
||||||
yamlconfig_version=1.0.2
|
|
||||||
cloth_version=18.0.145
|
|
||||||
modmenu_version=14.0.0-rc.2
|
|
||||||
junit_version=5.11.3
|
|
||||||
# version ids from modrinth
|
# version ids from modrinth
|
||||||
fabric_lithium_version=nhc57Td2
|
fabric_lithium_version=frXUdgvL
|
||||||
neo_lithium_version=P5VT33Jo
|
neo_lithium_version=KhdehJ6l
|
||||||
# Mod Properties
|
# Mod Properties
|
||||||
mod_version=0.3.0-beta.1
|
mod_version=0.1.0-beta.9
|
||||||
maven_group=ca.spottedleaf.moonrise
|
maven_group=ca.spottedleaf.moonrise
|
||||||
archives_base_name=moonrise
|
archives_base_name=moonrise
|
||||||
|
|||||||
BIN
gradle/wrapper/gradle-wrapper.jar
vendored
BIN
gradle/wrapper/gradle-wrapper.jar
vendored
Binary file not shown.
2
gradle/wrapper/gradle-wrapper.properties
vendored
2
gradle/wrapper/gradle-wrapper.properties
vendored
@@ -1,6 +1,6 @@
|
|||||||
distributionBase=GRADLE_USER_HOME
|
distributionBase=GRADLE_USER_HOME
|
||||||
distributionPath=wrapper/dists
|
distributionPath=wrapper/dists
|
||||||
distributionUrl=https\://services.gradle.org/distributions/gradle-8.12-all.zip
|
distributionUrl=https\://services.gradle.org/distributions/gradle-8.9-all.zip
|
||||||
networkTimeout=10000
|
networkTimeout=10000
|
||||||
validateDistributionUrl=true
|
validateDistributionUrl=true
|
||||||
zipStoreBase=GRADLE_USER_HOME
|
zipStoreBase=GRADLE_USER_HOME
|
||||||
|
|||||||
3
gradlew
vendored
3
gradlew
vendored
@@ -86,7 +86,8 @@ done
|
|||||||
# shellcheck disable=SC2034
|
# shellcheck disable=SC2034
|
||||||
APP_BASE_NAME=${0##*/}
|
APP_BASE_NAME=${0##*/}
|
||||||
# Discard cd standard output in case $CDPATH is set (https://github.com/gradle/gradle/issues/25036)
|
# Discard cd standard output in case $CDPATH is set (https://github.com/gradle/gradle/issues/25036)
|
||||||
APP_HOME=$( cd -P "${APP_HOME:-./}" > /dev/null && printf '%s\n' "$PWD" ) || exit
|
APP_HOME=$( cd -P "${APP_HOME:-./}" > /dev/null && printf '%s
|
||||||
|
' "$PWD" ) || exit
|
||||||
|
|
||||||
# Use the maximum available, or set MAX_FD != -1 to use that value.
|
# Use the maximum available, or set MAX_FD != -1 to use that value.
|
||||||
MAX_FD=maximum
|
MAX_FD=maximum
|
||||||
|
|||||||
6
installConcurrentUtil.sh
Executable file
6
installConcurrentUtil.sh
Executable file
@@ -0,0 +1,6 @@
|
|||||||
|
#!/bin/bash
|
||||||
|
set -eou pipefail
|
||||||
|
|
||||||
|
git submodule update --init --recursive
|
||||||
|
cd ConcurrentUtil
|
||||||
|
mvn install
|
||||||
@@ -1,10 +1,9 @@
|
|||||||
import java.nio.file.Files
|
import net.fabricmc.loom.util.aw2at.Aw2At
|
||||||
import java.nio.file.StandardCopyOption
|
|
||||||
import net.neoforged.moddevgradle.internal.RunGameTask
|
|
||||||
|
|
||||||
plugins {
|
plugins {
|
||||||
id("net.neoforged.moddev")
|
id("xyz.jpenilla.quiet-architectury-loom")
|
||||||
id 'maven-publish'
|
id 'maven-publish'
|
||||||
|
id 'com.gradleup.shadow'
|
||||||
}
|
}
|
||||||
|
|
||||||
repositories {
|
repositories {
|
||||||
@@ -14,88 +13,41 @@ repositories {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
def aw2at = Aw2AtTask.configureDefault(
|
configurations.implementation {
|
||||||
getProject(),
|
extendsFrom(configurations.shadow)
|
||||||
rootProject.layout.projectDirectory.file("src/main/resources/moonrise.accesswidener").getAsFile(),
|
|
||||||
sourceSets.main
|
|
||||||
)
|
|
||||||
|
|
||||||
neoForge {
|
|
||||||
version = rootProject.neoforge_version
|
|
||||||
validateAccessTransformers = true
|
|
||||||
accessTransformers.files.setFrom(aw2at.flatMap { t -> t.getOutputFile() })
|
|
||||||
mods {
|
|
||||||
moonrise {
|
|
||||||
sourceSet sourceSets.main
|
|
||||||
sourceSet rootProject.sourceSets.main
|
|
||||||
}
|
|
||||||
}
|
|
||||||
runs {
|
|
||||||
client {
|
|
||||||
client()
|
|
||||||
}
|
|
||||||
server {
|
|
||||||
server()
|
|
||||||
}
|
|
||||||
}
|
|
||||||
unitTest {
|
|
||||||
enable()
|
|
||||||
testedMod = mods.moonrise
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
dependencies {
|
dependencies {
|
||||||
runtimeOnly(project(":").sourceSets.main.output)
|
add('shadow', project([path: ":", configuration: "namedElements"]))
|
||||||
shadow(project(":"))
|
neoForge "net.neoforged:neoforge:${rootProject.neoforge_version}"
|
||||||
compileOnly(project(":"))
|
|
||||||
|
|
||||||
libs("ca.spottedleaf:concurrentutil:${rootProject.concurrentutil_version}") { setTransitive(false) }
|
shadow("ca.spottedleaf:concurrentutil:${rootProject.concurrentutil_version}")
|
||||||
libs("ca.spottedleaf:yamlconfig:${rootProject.yamlconfig_version}") { setTransitive(false) }
|
shadow("org.yaml:snakeyaml:${rootProject.snakeyaml_version}")
|
||||||
additionalRuntimeClasspath libs("org.yaml:snakeyaml:${rootProject.snakeyaml_version}")
|
forgeExtra("org.yaml:snakeyaml:${rootProject.snakeyaml_version}")
|
||||||
|
|
||||||
implementation "me.shedaniel.cloth:cloth-config-neoforge:${rootProject.cloth_version}"
|
modImplementation "me.shedaniel.cloth:cloth-config-neoforge:${rootProject.cloth_version}"
|
||||||
jarJar "me.shedaniel.cloth:cloth-config-neoforge:${rootProject.cloth_version}"
|
include "me.shedaniel.cloth:cloth-config-neoforge:${rootProject.cloth_version}"
|
||||||
}
|
}
|
||||||
|
|
||||||
tasks.processResources {
|
processResources {
|
||||||
def properties = [
|
inputs.property "version", project.version
|
||||||
"version": project.version,
|
|
||||||
"minecraft_version": minecraft_version,
|
|
||||||
"mod_version": mod_version
|
|
||||||
]
|
|
||||||
inputs.properties(properties)
|
|
||||||
filesMatching("META-INF/neoforge.mods.toml") {
|
filesMatching("META-INF/neoforge.mods.toml") {
|
||||||
expand properties
|
expand "version": project.version, "minecraft_version": minecraft_version, "loader_version": loader_version, "mod_version": mod_version
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
tasks.jar {
|
shadowJar {
|
||||||
archiveClassifier = "dev"
|
|
||||||
}
|
|
||||||
|
|
||||||
tasks.shadowJar {
|
|
||||||
archiveClassifier = "dev-all"
|
archiveClassifier = "dev-all"
|
||||||
destinationDirectory = layout.buildDirectory.dir("libs")
|
destinationDirectory = layout.buildDirectory.dir("libs")
|
||||||
configurations = [project.configurations.shadow]
|
configurations = [project.configurations.shadow]
|
||||||
relocate 'ca.spottedleaf.concurrentutil', 'ca.spottedleaf.moonrise.libs.ca.spottedleaf.concurrentutil'
|
relocate 'ca.spottedleaf.concurrentutil', 'ca.spottedleaf.moonrise.libs.ca.spottedleaf.concurrentutil'
|
||||||
relocate 'ca.spottedleaf.yamlconfig', 'ca.spottedleaf.moonrise.libs.ca.spottedleaf.yamlconfig'
|
|
||||||
relocate 'org.yaml.snakeyaml', 'ca.spottedleaf.moonrise.libs.org.yaml.snakeyaml'
|
relocate 'org.yaml.snakeyaml', 'ca.spottedleaf.moonrise.libs.org.yaml.snakeyaml'
|
||||||
}
|
}
|
||||||
|
|
||||||
tasks.register("productionJar", Zip.class) {
|
Aw2At.setup(getProject(), tasks.remapJar)
|
||||||
archiveClassifier = ""
|
|
||||||
archiveExtension = "jar"
|
|
||||||
destinationDirectory = layout.buildDirectory.dir("libs")
|
|
||||||
from(tasks.jarJar)
|
|
||||||
from(zipTree(tasks.shadowJar.archiveFile))
|
|
||||||
}
|
|
||||||
|
|
||||||
tasks.assemble {
|
|
||||||
dependsOn tasks.productionJar
|
|
||||||
}
|
|
||||||
|
|
||||||
publishMods {
|
publishMods {
|
||||||
file = productionJar.archiveFile
|
|
||||||
modLoaders = ["neoforge"]
|
modLoaders = ["neoforge"]
|
||||||
|
|
||||||
modrinth {
|
modrinth {
|
||||||
@@ -113,42 +65,3 @@ publishMods {
|
|||||||
)
|
)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
afterEvaluate {
|
|
||||||
neoForge.runs.configureEach { cfg ->
|
|
||||||
runConfigCommon.systemProperties.get().each {
|
|
||||||
cfg.systemProperties.put it.key, it.value
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Setup a run with lithium for compatibility testing
|
|
||||||
neoForge {
|
|
||||||
runs {
|
|
||||||
lithiumClient {
|
|
||||||
client()
|
|
||||||
disableIdeRun()
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
tasks.withType(RunGameTask).configureEach {
|
|
||||||
if (name == "runLithiumClient") {
|
|
||||||
return
|
|
||||||
}
|
|
||||||
def out = gameDirectory.get().getAsFile().toPath().resolve("mods/lithium-tmp.jar")
|
|
||||||
doFirst {
|
|
||||||
Files.deleteIfExists(out)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
def lithium = configurations.lithium
|
|
||||||
tasks.runLithiumClient {
|
|
||||||
def out = gameDirectory.get().getAsFile().toPath().resolve("mods/lithium-tmp.jar")
|
|
||||||
doFirst {
|
|
||||||
for (File file in lithium) {
|
|
||||||
Files.copy(file.toPath(), out, StandardCopyOption.REPLACE_EXISTING)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
doLast {
|
|
||||||
Files.deleteIfExists(out)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|||||||
1
neoforge/gradle.properties
Normal file
1
neoforge/gradle.properties
Normal file
@@ -0,0 +1 @@
|
|||||||
|
loom.platform=neoforge
|
||||||
@@ -1,36 +1,28 @@
|
|||||||
package ca.spottedleaf.moonrise.neoforge;
|
package ca.spottedleaf.moonrise.neoforge;
|
||||||
|
|
||||||
import ca.spottedleaf.moonrise.common.util.BaseChunkSystemHooks;
|
|
||||||
import ca.spottedleaf.moonrise.common.PlatformHooks;
|
import ca.spottedleaf.moonrise.common.PlatformHooks;
|
||||||
import ca.spottedleaf.moonrise.common.util.ConfigHolder;
|
import ca.spottedleaf.moonrise.common.util.ConfigHolder;
|
||||||
import ca.spottedleaf.moonrise.common.util.CoordinateUtils;
|
import ca.spottedleaf.moonrise.common.util.CoordinateUtils;
|
||||||
import ca.spottedleaf.moonrise.patches.chunk_system.ticket.ChunkSystemTicketType;
|
import ca.spottedleaf.moonrise.patches.chunk_system.scheduling.NewChunkHolder;
|
||||||
import com.mojang.datafixers.DSL;
|
|
||||||
import com.mojang.datafixers.DataFixer;
|
|
||||||
import com.mojang.serialization.Dynamic;
|
|
||||||
import it.unimi.dsi.fastutil.longs.LongArrayList;
|
|
||||||
import net.minecraft.core.BlockPos;
|
import net.minecraft.core.BlockPos;
|
||||||
import net.minecraft.nbt.CompoundTag;
|
import net.minecraft.nbt.CompoundTag;
|
||||||
import net.minecraft.nbt.NbtOps;
|
|
||||||
import net.minecraft.server.level.ChunkHolder;
|
|
||||||
import net.minecraft.server.level.GenerationChunkHolder;
|
import net.minecraft.server.level.GenerationChunkHolder;
|
||||||
import net.minecraft.server.level.ServerLevel;
|
import net.minecraft.server.level.ServerLevel;
|
||||||
import net.minecraft.server.level.ServerPlayer;
|
import net.minecraft.server.level.ServerPlayer;
|
||||||
import net.minecraft.server.level.TicketType;
|
|
||||||
import net.minecraft.world.entity.Entity;
|
import net.minecraft.world.entity.Entity;
|
||||||
import net.minecraft.world.level.BlockGetter;
|
import net.minecraft.world.level.BlockGetter;
|
||||||
import net.minecraft.world.level.ChunkPos;
|
import net.minecraft.world.level.ChunkPos;
|
||||||
import net.minecraft.world.level.EmptyBlockGetter;
|
import net.minecraft.world.level.EmptyBlockGetter;
|
||||||
|
import net.minecraft.world.level.Explosion;
|
||||||
import net.minecraft.world.level.Level;
|
import net.minecraft.world.level.Level;
|
||||||
import net.minecraft.world.level.block.state.BlockState;
|
import net.minecraft.world.level.block.state.BlockState;
|
||||||
import net.minecraft.world.level.chunk.ChunkAccess;
|
import net.minecraft.world.level.chunk.ChunkAccess;
|
||||||
import net.minecraft.world.level.chunk.ImposterProtoChunk;
|
import net.minecraft.world.level.chunk.ImposterProtoChunk;
|
||||||
import net.minecraft.world.level.chunk.LevelChunk;
|
import net.minecraft.world.level.chunk.LevelChunk;
|
||||||
import net.minecraft.world.level.chunk.ProtoChunk;
|
import net.minecraft.world.level.chunk.ProtoChunk;
|
||||||
import net.minecraft.world.level.chunk.status.ChunkStatusTasks;
|
|
||||||
import net.minecraft.world.level.chunk.storage.SerializableChunkData;
|
|
||||||
import net.minecraft.world.level.entity.EntityTypeTest;
|
import net.minecraft.world.level.entity.EntityTypeTest;
|
||||||
import net.minecraft.world.phys.AABB;
|
import net.minecraft.world.phys.AABB;
|
||||||
|
import net.minecraft.world.phys.Vec3;
|
||||||
import net.neoforged.neoforge.common.CommonHooks;
|
import net.neoforged.neoforge.common.CommonHooks;
|
||||||
import net.neoforged.neoforge.common.NeoForge;
|
import net.neoforged.neoforge.common.NeoForge;
|
||||||
import net.neoforged.neoforge.entity.PartEntity;
|
import net.neoforged.neoforge.entity.PartEntity;
|
||||||
@@ -38,11 +30,10 @@ import net.neoforged.neoforge.event.EventHooks;
|
|||||||
import net.neoforged.neoforge.event.entity.EntityJoinLevelEvent;
|
import net.neoforged.neoforge.event.entity.EntityJoinLevelEvent;
|
||||||
import net.neoforged.neoforge.event.level.ChunkDataEvent;
|
import net.neoforged.neoforge.event.level.ChunkDataEvent;
|
||||||
import net.neoforged.neoforge.event.level.ChunkEvent;
|
import net.neoforged.neoforge.event.level.ChunkEvent;
|
||||||
import java.util.Collection;
|
|
||||||
import java.util.List;
|
import java.util.List;
|
||||||
import java.util.function.Predicate;
|
import java.util.function.Predicate;
|
||||||
|
|
||||||
public final class NeoForgeHooks extends BaseChunkSystemHooks implements PlatformHooks {
|
public final class NeoForgeHooks implements PlatformHooks {
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public String getBrand() {
|
public String getBrand() {
|
||||||
@@ -61,6 +52,16 @@ public final class NeoForgeHooks extends BaseChunkSystemHooks implements Platfor
|
|||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void onExplosion(final Level world, final Explosion explosion, final List<Entity> possiblyAffecting, final double diameter) {
|
||||||
|
EventHooks.onExplosionDetonate(world, explosion, possiblyAffecting, diameter);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public Vec3 modifyExplosionKnockback(final Level world, final Explosion explosion, final Entity entity, final Vec3 original) {
|
||||||
|
return EventHooks.getExplosionKnockback(world, explosion, entity, original);
|
||||||
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public boolean hasCurrentlyLoadingChunk() {
|
public boolean hasCurrentlyLoadingChunk() {
|
||||||
return true;
|
return true;
|
||||||
@@ -87,12 +88,10 @@ public final class NeoForgeHooks extends BaseChunkSystemHooks implements Platfor
|
|||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void onChunkHolderTicketChange(final ServerLevel world, final ChunkHolder holder, final int oldLevel, final int newLevel) {
|
public void onChunkHolderTicketChange(final ServerLevel world, final NewChunkHolder holder, final int oldLevel, final int newLevel) {
|
||||||
final ChunkPos pos = holder.getPos();
|
|
||||||
|
|
||||||
EventHooks.fireChunkTicketLevelUpdated(
|
EventHooks.fireChunkTicketLevelUpdated(
|
||||||
world, CoordinateUtils.getChunkKey(pos.x, pos.z),
|
world, CoordinateUtils.getChunkKey(holder.chunkX, holder.chunkZ),
|
||||||
oldLevel, newLevel, holder
|
oldLevel, newLevel, holder.vanillaChunkHolder
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -102,7 +101,7 @@ public final class NeoForgeHooks extends BaseChunkSystemHooks implements Platfor
|
|||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void chunkSyncSave(final ServerLevel world, final ChunkAccess chunk, final SerializableChunkData data) {
|
public void chunkSyncSave(final ServerLevel world, final ChunkAccess chunk, final CompoundTag data) {
|
||||||
NeoForge.EVENT_BUS.post(new ChunkDataEvent.Save(chunk, world, data));
|
NeoForge.EVENT_BUS.post(new ChunkDataEvent.Save(chunk, world, data));
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -119,12 +118,7 @@ public final class NeoForgeHooks extends BaseChunkSystemHooks implements Platfor
|
|||||||
@Override
|
@Override
|
||||||
public void addToGetEntities(final Level world, final Entity entity, final AABB boundingBox, final Predicate<? super Entity> predicate,
|
public void addToGetEntities(final Level world, final Entity entity, final AABB boundingBox, final Predicate<? super Entity> predicate,
|
||||||
final List<Entity> into) {
|
final List<Entity> into) {
|
||||||
final Collection<PartEntity<?>> parts = world.dragonParts();
|
for (final PartEntity<?> part : world.getPartEntities()) {
|
||||||
if (parts.isEmpty()) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
for (final PartEntity<?> part : parts) {
|
|
||||||
if (part != entity && part.getBoundingBox().intersects(boundingBox) && (predicate == null || predicate.test(part))) {
|
if (part != entity && part.getBoundingBox().intersects(boundingBox) && (predicate == null || predicate.test(part))) {
|
||||||
into.add(part);
|
into.add(part);
|
||||||
}
|
}
|
||||||
@@ -139,18 +133,9 @@ public final class NeoForgeHooks extends BaseChunkSystemHooks implements Platfor
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
final Collection<PartEntity<?>> parts = world.dragonParts();
|
for (final PartEntity<?> part : world.getPartEntities()) {
|
||||||
if (parts.isEmpty()) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
for (final PartEntity<?> part : parts) {
|
|
||||||
if (!part.getBoundingBox().intersects(boundingBox)) {
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
|
|
||||||
final T casted = (T)entityTypeTest.tryCast(part);
|
final T casted = (T)entityTypeTest.tryCast(part);
|
||||||
if (casted != null && (predicate == null || predicate.test(casted))) {
|
if (casted != null && casted.getBoundingBox().intersects(boundingBox) && (predicate == null || predicate.test(casted))) {
|
||||||
into.add(casted);
|
into.add(casted);
|
||||||
if (into.size() >= maxCount) {
|
if (into.size() >= maxCount) {
|
||||||
break;
|
break;
|
||||||
@@ -208,12 +193,12 @@ public final class NeoForgeHooks extends BaseChunkSystemHooks implements Platfor
|
|||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public long configAutoSaveInterval(final ServerLevel world) {
|
public long configAutoSaveInterval() {
|
||||||
return ConfigHolder.getConfig().chunkSaving.autoSaveInterval.getTimeTicks();
|
return ConfigHolder.getConfig().chunkSaving.autoSaveInterval.getTimeTicks();
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public int configMaxAutoSavePerTick(final ServerLevel world) {
|
public int configMaxAutoSavePerTick() {
|
||||||
return ConfigHolder.getConfig().chunkSaving.maxAutoSaveChunksPerTick;
|
return ConfigHolder.getConfig().chunkSaving.maxAutoSaveChunksPerTick;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -221,61 +206,4 @@ public final class NeoForgeHooks extends BaseChunkSystemHooks implements Platfor
|
|||||||
public boolean configFixMC159283() {
|
public boolean configFixMC159283() {
|
||||||
return ConfigHolder.getConfig().bugFixes.fixMC159283;
|
return ConfigHolder.getConfig().bugFixes.fixMC159283;
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
|
||||||
public boolean forceNoSave(final ChunkAccess chunk) {
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public CompoundTag convertNBT(final DSL.TypeReference type, final DataFixer dataFixer, final CompoundTag nbt,
|
|
||||||
final int fromVersion, final int toVersion) {
|
|
||||||
return (CompoundTag)dataFixer.update(
|
|
||||||
type, new Dynamic<>(NbtOps.INSTANCE, nbt), fromVersion, toVersion
|
|
||||||
).getValue();
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public boolean hasMainChunkLoadHook() {
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public void mainChunkLoad(final ChunkAccess chunk, final SerializableChunkData chunkData) {
|
|
||||||
NeoForge.EVENT_BUS.post(new ChunkDataEvent.Load(chunk, chunkData));
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public List<Entity> modifySavedEntities(final ServerLevel world, final int chunkX, final int chunkZ, final List<Entity> entities) {
|
|
||||||
return entities;
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public void unloadEntity(final Entity entity) {
|
|
||||||
entity.setRemoved(Entity.RemovalReason.UNLOADED_TO_CHUNK);
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public void postLoadProtoChunk(final ServerLevel world, final ProtoChunk chunk) {
|
|
||||||
ChunkStatusTasks.postLoadProtoChunk(world, chunk.getEntities());
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public int modifyEntityTrackingRange(final Entity entity, final int currentRange) {
|
|
||||||
return currentRange;
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public long[] getCounterTypesUncached(final TicketType type) {
|
|
||||||
final LongArrayList ret = new LongArrayList();
|
|
||||||
|
|
||||||
if (type == TicketType.FORCED) {
|
|
||||||
ret.add(ChunkSystemTicketType.COUNTER_TYPE_FORCED);
|
|
||||||
}
|
|
||||||
if (type.forceNaturalSpawning()) {
|
|
||||||
ret.add(ChunkSystemTicketType.COUNTER_TYPER_NATURAL_SPAWNING_FORCED);
|
|
||||||
}
|
|
||||||
|
|
||||||
return ret.toLongArray();
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -0,0 +1,87 @@
|
|||||||
|
package ca.spottedleaf.moonrise.neoforge.mixin.chunk_system;
|
||||||
|
|
||||||
|
import ca.spottedleaf.moonrise.common.util.CoordinateUtils;
|
||||||
|
import ca.spottedleaf.moonrise.patches.chunk_system.level.chunk.ChunkSystemDistanceManager;
|
||||||
|
import ca.spottedleaf.moonrise.patches.chunk_tick_iteration.ChunkTickServerLevel;
|
||||||
|
import it.unimi.dsi.fastutil.longs.Long2ObjectOpenHashMap;
|
||||||
|
import net.minecraft.server.level.ChunkLevel;
|
||||||
|
import net.minecraft.server.level.DistanceManager;
|
||||||
|
import net.minecraft.server.level.FullChunkStatus;
|
||||||
|
import net.minecraft.server.level.Ticket;
|
||||||
|
import net.minecraft.server.level.TicketType;
|
||||||
|
import net.minecraft.util.SortedArraySet;
|
||||||
|
import net.minecraft.world.level.ChunkPos;
|
||||||
|
import org.spongepowered.asm.mixin.Final;
|
||||||
|
import org.spongepowered.asm.mixin.Mixin;
|
||||||
|
import org.spongepowered.asm.mixin.Overwrite;
|
||||||
|
import org.spongepowered.asm.mixin.Shadow;
|
||||||
|
|
||||||
|
@Mixin(DistanceManager.class)
|
||||||
|
abstract class NeoForgeDistanceManagerMixin implements ChunkSystemDistanceManager {
|
||||||
|
|
||||||
|
@Shadow
|
||||||
|
@Final
|
||||||
|
private Long2ObjectOpenHashMap<SortedArraySet<Ticket<?>>> forcedTickets;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @reason Route to new chunk system
|
||||||
|
* @author Spottedleaf
|
||||||
|
*/
|
||||||
|
@Overwrite
|
||||||
|
public <T> void addRegionTicket(final TicketType<T> type, final ChunkPos pos, final int radius, final T identifier, final boolean forceTicks) {
|
||||||
|
final int level = ChunkLevel.byStatus(FullChunkStatus.FULL) - radius;
|
||||||
|
this.moonrise$getChunkHolderManager().addTicketAtLevel(type, pos, level, identifier);
|
||||||
|
if (forceTicks) {
|
||||||
|
final Ticket<T> forceTicket = new Ticket<>(type, level, identifier, forceTicks);
|
||||||
|
|
||||||
|
this.forcedTickets.compute(pos.toLong(), (final Long keyInMap, final SortedArraySet<Ticket<?>> valueInMap) -> {
|
||||||
|
final SortedArraySet<Ticket<?>> ret;
|
||||||
|
if (valueInMap != null) {
|
||||||
|
ret = valueInMap;
|
||||||
|
} else {
|
||||||
|
ret = SortedArraySet.create(4);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (ret.add(forceTicket)) {
|
||||||
|
((ChunkTickServerLevel)NeoForgeDistanceManagerMixin.this.moonrise$getChunkMap().level).moonrise$addPlayerTickingRequest(
|
||||||
|
CoordinateUtils.getChunkX(keyInMap.longValue()), CoordinateUtils.getChunkZ(keyInMap.longValue())
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
return ret;
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @reason Route to new chunk system
|
||||||
|
* @author Spottedleaf
|
||||||
|
*/
|
||||||
|
@Overwrite
|
||||||
|
public <T> void removeRegionTicket(final TicketType<T> type, final ChunkPos pos, final int radius, final T identifier, final boolean forceTicks) {
|
||||||
|
final int level = ChunkLevel.byStatus(FullChunkStatus.FULL) - radius;
|
||||||
|
this.moonrise$getChunkHolderManager().removeTicketAtLevel(type, pos, level, identifier);
|
||||||
|
if (forceTicks) {
|
||||||
|
final Ticket<T> forceTicket = new Ticket<>(type, level, identifier, forceTicks);
|
||||||
|
|
||||||
|
this.forcedTickets.computeIfPresent(pos.toLong(), (final Long keyInMap, final SortedArraySet<Ticket<?>> valueInMap) -> {
|
||||||
|
if (valueInMap.remove(forceTicket)) {
|
||||||
|
((ChunkTickServerLevel)NeoForgeDistanceManagerMixin.this.moonrise$getChunkMap().level).moonrise$removePlayerTickingRequest(
|
||||||
|
CoordinateUtils.getChunkX(keyInMap.longValue()), CoordinateUtils.getChunkZ(keyInMap.longValue())
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
return valueInMap.isEmpty() ? null : valueInMap;
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @reason Only use containsKey, as we fix the leak with this impl
|
||||||
|
* @author Spottedleaf
|
||||||
|
*/
|
||||||
|
@Overwrite
|
||||||
|
public boolean shouldForceTicks(final long chunkPos) {
|
||||||
|
return this.forcedTickets.containsKey(chunkPos);
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -1,69 +0,0 @@
|
|||||||
package ca.spottedleaf.moonrise.neoforge.mixin.chunk_system;
|
|
||||||
|
|
||||||
import ca.spottedleaf.moonrise.common.util.CoordinateUtils;
|
|
||||||
import ca.spottedleaf.moonrise.patches.chunk_system.level.ChunkSystemServerLevel;
|
|
||||||
import ca.spottedleaf.moonrise.patches.chunk_system.ticket.ChunkSystemTicketStorage;
|
|
||||||
import ca.spottedleaf.moonrise.patches.chunk_system.ticket.ChunkSystemTicketType;
|
|
||||||
import it.unimi.dsi.fastutil.longs.Long2IntOpenHashMap;
|
|
||||||
import it.unimi.dsi.fastutil.longs.Long2ObjectOpenHashMap;
|
|
||||||
import it.unimi.dsi.fastutil.longs.LongSet;
|
|
||||||
import net.minecraft.world.level.ChunkPos;
|
|
||||||
import net.minecraft.world.level.TicketStorage;
|
|
||||||
import org.spongepowered.asm.mixin.Mixin;
|
|
||||||
import org.spongepowered.asm.mixin.Overwrite;
|
|
||||||
import org.spongepowered.asm.mixin.Shadow;
|
|
||||||
import org.spongepowered.asm.mixin.injection.At;
|
|
||||||
import org.spongepowered.asm.mixin.injection.Inject;
|
|
||||||
import org.spongepowered.asm.mixin.injection.Redirect;
|
|
||||||
import org.spongepowered.asm.mixin.injection.callback.CallbackInfo;
|
|
||||||
|
|
||||||
@Mixin(TicketStorage.class)
|
|
||||||
abstract class NeoForgeTicketStorageMixin implements ChunkSystemTicketStorage {
|
|
||||||
|
|
||||||
@Shadow
|
|
||||||
private LongSet chunksWithForceNaturalSpawning;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @reason Destroy old chunk system state
|
|
||||||
* @author Spottedleaf
|
|
||||||
*/
|
|
||||||
@Inject(
|
|
||||||
method = "<init>(Lit/unimi/dsi/fastutil/longs/Long2ObjectOpenHashMap;Lit/unimi/dsi/fastutil/longs/Long2ObjectOpenHashMap;)V",
|
|
||||||
at = @At(
|
|
||||||
value = "RETURN"
|
|
||||||
)
|
|
||||||
)
|
|
||||||
private void destroyFields(Long2ObjectOpenHashMap p_393873_, Long2ObjectOpenHashMap p_394615_, CallbackInfo ci) {
|
|
||||||
this.chunksWithForceNaturalSpawning = null;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @reason The forced natural spawning chunks would be empty, as tickets should always be empty.
|
|
||||||
* We need to do this to avoid throwing immediately.
|
|
||||||
* @author Spottedleaf
|
|
||||||
*/
|
|
||||||
@Redirect(
|
|
||||||
method = "<init>(Lit/unimi/dsi/fastutil/longs/Long2ObjectOpenHashMap;Lit/unimi/dsi/fastutil/longs/Long2ObjectOpenHashMap;)V",
|
|
||||||
at = @At(
|
|
||||||
value = "INVOKE",
|
|
||||||
target = "Lnet/minecraft/world/level/TicketStorage;updateForcedNaturalSpawning()V"
|
|
||||||
)
|
|
||||||
)
|
|
||||||
private void avoidUpdatingForcedNaturalChunks(final TicketStorage instance) {}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @reason Route to new chunk system
|
|
||||||
* @author Spottedleaf
|
|
||||||
*/
|
|
||||||
@Overwrite
|
|
||||||
public boolean shouldForceNaturalSpawning(final ChunkPos pos) {
|
|
||||||
final Long2IntOpenHashMap counters = ((ChunkSystemServerLevel)this.moonrise$getChunkMap().level).moonrise$getChunkTaskScheduler()
|
|
||||||
.chunkHolderManager.getTicketCounters(ChunkSystemTicketType.COUNTER_TYPER_NATURAL_SPAWNING_FORCED);
|
|
||||||
|
|
||||||
if (counters == null) {
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
return counters.containsKey(CoordinateUtils.getChunkKey(pos));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -28,7 +28,7 @@ side = "BOTH"
|
|||||||
[[dependencies.moonrise]]
|
[[dependencies.moonrise]]
|
||||||
modId = "minecraft"
|
modId = "minecraft"
|
||||||
type = "required"
|
type = "required"
|
||||||
versionRange = "(1.21.4,1.21.6)"
|
versionRange = "[1.21,1.21.2)"
|
||||||
ordering = "NONE"
|
ordering = "NONE"
|
||||||
side = "BOTH"
|
side = "BOTH"
|
||||||
|
|
||||||
@@ -57,6 +57,7 @@ config = "moonrise-neoforge.mixins.json"
|
|||||||
|
|
||||||
["lithium:options"]
|
["lithium:options"]
|
||||||
"mixin.ai.poi" = false
|
"mixin.ai.poi" = false
|
||||||
|
"mixin.alloc.chunk_ticking" = false
|
||||||
"mixin.alloc.deep_passengers" = false
|
"mixin.alloc.deep_passengers" = false
|
||||||
"mixin.alloc.entity_tracker" = false
|
"mixin.alloc.entity_tracker" = false
|
||||||
"mixin.block.flatten_states" = false
|
"mixin.block.flatten_states" = false
|
||||||
@@ -80,5 +81,6 @@ config = "moonrise-neoforge.mixins.json"
|
|||||||
"mixin.world.block_entity_ticking" = false
|
"mixin.world.block_entity_ticking" = false
|
||||||
"mixin.world.chunk_access" = false
|
"mixin.world.chunk_access" = false
|
||||||
"mixin.world.explosions.block_raycast" = false
|
"mixin.world.explosions.block_raycast" = false
|
||||||
|
"mixin.world.explosions.cache_exposure" = false
|
||||||
"mixin.world.temperature_cache" = false
|
"mixin.world.temperature_cache" = false
|
||||||
"mixin.world.tick_scheduler" = false
|
"mixin.world.tick_scheduler" = false
|
||||||
|
|||||||
@@ -2,9 +2,9 @@
|
|||||||
"parent": "moonrise.mixins.json",
|
"parent": "moonrise.mixins.json",
|
||||||
"package": "ca.spottedleaf.moonrise.neoforge.mixin",
|
"package": "ca.spottedleaf.moonrise.neoforge.mixin",
|
||||||
"mixins": [
|
"mixins": [
|
||||||
|
"chunk_system.NeoForgeDistanceManagerMixin",
|
||||||
"chunk_system.NeoForgeMinecraftServerMixin",
|
"chunk_system.NeoForgeMinecraftServerMixin",
|
||||||
"chunk_system.NeoForgeServerLevelMixin",
|
"chunk_system.NeoForgeServerLevelMixin",
|
||||||
"chunk_system.NeoForgeTicketStorageMixin",
|
|
||||||
"collisions.EntityMixin"
|
"collisions.EntityMixin"
|
||||||
],
|
],
|
||||||
"client": [
|
"client": [
|
||||||
|
|||||||
@@ -1,17 +0,0 @@
|
|||||||
package ca.spottedleaf.moonrise.neoforge;
|
|
||||||
|
|
||||||
import org.junit.jupiter.api.Test;
|
|
||||||
import org.spongepowered.asm.mixin.MixinEnvironment;
|
|
||||||
|
|
||||||
class MixinAuditTest {
|
|
||||||
@Test
|
|
||||||
void auditMixins() {
|
|
||||||
final ClassLoader old = Thread.currentThread().getContextClassLoader();
|
|
||||||
try {
|
|
||||||
Thread.currentThread().setContextClassLoader(MixinAuditTest.class.getClassLoader());
|
|
||||||
MixinEnvironment.getCurrentEnvironment().audit();
|
|
||||||
} finally {
|
|
||||||
Thread.currentThread().setContextClassLoader(old);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -13,6 +13,7 @@ pluginManagement {
|
|||||||
maven {
|
maven {
|
||||||
name = 'jmp'
|
name = 'jmp'
|
||||||
url = 'https://repo.jpenilla.xyz/snapshots'
|
url = 'https://repo.jpenilla.xyz/snapshots'
|
||||||
|
mavenContent { snapshotsOnly() }
|
||||||
}
|
}
|
||||||
maven {
|
maven {
|
||||||
name = 'architectury'
|
name = 'architectury'
|
||||||
@@ -22,10 +23,9 @@ pluginManagement {
|
|||||||
}
|
}
|
||||||
|
|
||||||
plugins {
|
plugins {
|
||||||
id("org.gradle.toolchains.foojay-resolver-convention") version "0.9.0"
|
id("org.gradle.toolchains.foojay-resolver-convention") version "0.8.0"
|
||||||
id("quiet-fabric-loom") version "1.10.317" apply false
|
id("xyz.jpenilla.quiet-architectury-loom") version "1.7.300" apply false
|
||||||
id("net.neoforged.moddev") version "2.0.80" apply false
|
id 'com.gradleup.shadow' version '8.3.0' apply false
|
||||||
id 'com.gradleup.shadow' version '8.3.6' apply false
|
|
||||||
}
|
}
|
||||||
|
|
||||||
dependencyResolutionManagement {
|
dependencyResolutionManagement {
|
||||||
@@ -37,7 +37,7 @@ dependencyResolutionManagement {
|
|||||||
}
|
}
|
||||||
versionCatalogs {
|
versionCatalogs {
|
||||||
create("fabricApiLibs") {
|
create("fabricApiLibs") {
|
||||||
from("net.fabricmc.fabric-api:fabric-api-catalog:${fabric_api_version}")
|
from("net.fabricmc.fabric-api:fabric-api-catalog:0.107.0+1.21.1")
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -48,6 +48,3 @@ include("fabric")
|
|||||||
findProject(":fabric").name = "Moonrise-Fabric"
|
findProject(":fabric").name = "Moonrise-Fabric"
|
||||||
include("neoforge")
|
include("neoforge")
|
||||||
findProject(":neoforge").name = "Moonrise-NeoForge"
|
findProject(":neoforge").name = "Moonrise-NeoForge"
|
||||||
|
|
||||||
// includeBuild("../YamlConfig") // Uncomment to use local YamlConfig
|
|
||||||
// includeBuild("../ConcurrentUtil") // Uncomment to use local ConcurrentUtil
|
|
||||||
|
|||||||
@@ -1,30 +1,28 @@
|
|||||||
package ca.spottedleaf.moonrise.common;
|
package ca.spottedleaf.moonrise.common;
|
||||||
|
|
||||||
import ca.spottedleaf.moonrise.common.util.ChunkSystemHooks;
|
import ca.spottedleaf.moonrise.patches.chunk_system.scheduling.NewChunkHolder;
|
||||||
import com.mojang.datafixers.DSL;
|
|
||||||
import com.mojang.datafixers.DataFixer;
|
|
||||||
import net.minecraft.core.BlockPos;
|
import net.minecraft.core.BlockPos;
|
||||||
import net.minecraft.nbt.CompoundTag;
|
import net.minecraft.nbt.CompoundTag;
|
||||||
import net.minecraft.server.level.ChunkHolder;
|
|
||||||
import net.minecraft.server.level.GenerationChunkHolder;
|
import net.minecraft.server.level.GenerationChunkHolder;
|
||||||
import net.minecraft.server.level.ServerLevel;
|
import net.minecraft.server.level.ServerLevel;
|
||||||
import net.minecraft.server.level.ServerPlayer;
|
import net.minecraft.server.level.ServerPlayer;
|
||||||
import net.minecraft.world.entity.Entity;
|
import net.minecraft.world.entity.Entity;
|
||||||
import net.minecraft.world.level.BlockGetter;
|
import net.minecraft.world.level.BlockGetter;
|
||||||
import net.minecraft.world.level.ChunkPos;
|
import net.minecraft.world.level.ChunkPos;
|
||||||
|
import net.minecraft.world.level.Explosion;
|
||||||
import net.minecraft.world.level.Level;
|
import net.minecraft.world.level.Level;
|
||||||
import net.minecraft.world.level.block.state.BlockState;
|
import net.minecraft.world.level.block.state.BlockState;
|
||||||
import net.minecraft.world.level.chunk.ChunkAccess;
|
import net.minecraft.world.level.chunk.ChunkAccess;
|
||||||
import net.minecraft.world.level.chunk.LevelChunk;
|
import net.minecraft.world.level.chunk.LevelChunk;
|
||||||
import net.minecraft.world.level.chunk.ProtoChunk;
|
import net.minecraft.world.level.chunk.ProtoChunk;
|
||||||
import net.minecraft.world.level.chunk.storage.SerializableChunkData;
|
|
||||||
import net.minecraft.world.level.entity.EntityTypeTest;
|
import net.minecraft.world.level.entity.EntityTypeTest;
|
||||||
import net.minecraft.world.phys.AABB;
|
import net.minecraft.world.phys.AABB;
|
||||||
|
import net.minecraft.world.phys.Vec3;
|
||||||
import java.util.List;
|
import java.util.List;
|
||||||
import java.util.ServiceLoader;
|
import java.util.ServiceLoader;
|
||||||
import java.util.function.Predicate;
|
import java.util.function.Predicate;
|
||||||
|
|
||||||
public interface PlatformHooks extends ChunkSystemHooks {
|
public interface PlatformHooks {
|
||||||
public static PlatformHooks get() {
|
public static PlatformHooks get() {
|
||||||
return Holder.INSTANCE;
|
return Holder.INSTANCE;
|
||||||
}
|
}
|
||||||
@@ -35,6 +33,10 @@ public interface PlatformHooks extends ChunkSystemHooks {
|
|||||||
|
|
||||||
public Predicate<BlockState> maybeHasLightEmission();
|
public Predicate<BlockState> maybeHasLightEmission();
|
||||||
|
|
||||||
|
public void onExplosion(final Level world, final Explosion explosion, final List<Entity> possiblyAffecting, final double diameter);
|
||||||
|
|
||||||
|
public Vec3 modifyExplosionKnockback(final Level world, final Explosion explosion, final Entity entity, final Vec3 original);
|
||||||
|
|
||||||
public boolean hasCurrentlyLoadingChunk();
|
public boolean hasCurrentlyLoadingChunk();
|
||||||
|
|
||||||
public LevelChunk getCurrentlyLoadingChunk(final GenerationChunkHolder holder);
|
public LevelChunk getCurrentlyLoadingChunk(final GenerationChunkHolder holder);
|
||||||
@@ -45,11 +47,11 @@ public interface PlatformHooks extends ChunkSystemHooks {
|
|||||||
|
|
||||||
public boolean allowAsyncTicketUpdates();
|
public boolean allowAsyncTicketUpdates();
|
||||||
|
|
||||||
public void onChunkHolderTicketChange(final ServerLevel world, final ChunkHolder holder, final int oldLevel, final int newLevel);
|
public void onChunkHolderTicketChange(final ServerLevel world, final NewChunkHolder holder, final int oldLevel, final int newLevel);
|
||||||
|
|
||||||
public void chunkUnloadFromWorld(final LevelChunk chunk);
|
public void chunkUnloadFromWorld(final LevelChunk chunk);
|
||||||
|
|
||||||
public void chunkSyncSave(final ServerLevel world, final ChunkAccess chunk, final SerializableChunkData data);
|
public void chunkSyncSave(final ServerLevel world, final ChunkAccess chunk, final CompoundTag data);
|
||||||
|
|
||||||
public void onChunkWatch(final ServerLevel world, final LevelChunk chunk, final ServerPlayer player);
|
public void onChunkWatch(final ServerLevel world, final LevelChunk chunk, final ServerPlayer player);
|
||||||
|
|
||||||
@@ -64,6 +66,8 @@ public interface PlatformHooks extends ChunkSystemHooks {
|
|||||||
|
|
||||||
public void entityMove(final Entity entity, final long oldSection, final long newSection);
|
public void entityMove(final Entity entity, final long oldSection, final long newSection);
|
||||||
|
|
||||||
|
public boolean screenEntity(final ServerLevel world, final Entity entity, final boolean fromDisk, final boolean event);
|
||||||
|
|
||||||
public boolean configFixMC224294();
|
public boolean configFixMC224294();
|
||||||
|
|
||||||
public boolean configAutoConfigSendDistance();
|
public boolean configAutoConfigSendDistance();
|
||||||
@@ -78,30 +82,12 @@ public interface PlatformHooks extends ChunkSystemHooks {
|
|||||||
|
|
||||||
public int configPlayerMaxConcurrentGens();
|
public int configPlayerMaxConcurrentGens();
|
||||||
|
|
||||||
public long configAutoSaveInterval(final ServerLevel world);
|
public long configAutoSaveInterval();
|
||||||
|
|
||||||
public int configMaxAutoSavePerTick(final ServerLevel world);
|
public int configMaxAutoSavePerTick();
|
||||||
|
|
||||||
public boolean configFixMC159283();
|
public boolean configFixMC159283();
|
||||||
|
|
||||||
// support for CB chunk mustNotSave
|
|
||||||
public boolean forceNoSave(final ChunkAccess chunk);
|
|
||||||
|
|
||||||
public CompoundTag convertNBT(final DSL.TypeReference type, final DataFixer dataFixer, final CompoundTag nbt,
|
|
||||||
final int fromVersion, final int toVersion);
|
|
||||||
|
|
||||||
public boolean hasMainChunkLoadHook();
|
|
||||||
|
|
||||||
public void mainChunkLoad(final ChunkAccess chunk, final SerializableChunkData chunkData);
|
|
||||||
|
|
||||||
public List<Entity> modifySavedEntities(final ServerLevel world, final int chunkX, final int chunkZ, final List<Entity> entities);
|
|
||||||
|
|
||||||
public void unloadEntity(final Entity entity);
|
|
||||||
|
|
||||||
public void postLoadProtoChunk(final ServerLevel world, final ProtoChunk chunk);
|
|
||||||
|
|
||||||
public int modifyEntityTrackingRange(final Entity entity, final int currentRange);
|
|
||||||
|
|
||||||
public static final class Holder {
|
public static final class Holder {
|
||||||
private Holder() {
|
private Holder() {
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -0,0 +1,7 @@
|
|||||||
|
package ca.spottedleaf.moonrise.common.config;
|
||||||
|
|
||||||
|
public interface InitialiseHook {
|
||||||
|
|
||||||
|
public void initialise();
|
||||||
|
|
||||||
|
}
|
||||||
@@ -0,0 +1,11 @@
|
|||||||
|
package ca.spottedleaf.moonrise.common.config.adapter;
|
||||||
|
|
||||||
|
import java.lang.reflect.Type;
|
||||||
|
|
||||||
|
public abstract class TypeAdapter<T, S> {
|
||||||
|
|
||||||
|
public abstract T deserialize(final TypeAdapterRegistry registry, final Object input, final Type type);
|
||||||
|
|
||||||
|
public abstract S serialize(final TypeAdapterRegistry registry, final T value, final Type type);
|
||||||
|
|
||||||
|
}
|
||||||
@@ -0,0 +1,307 @@
|
|||||||
|
package ca.spottedleaf.moonrise.common.config.adapter;
|
||||||
|
|
||||||
|
import ca.spottedleaf.moonrise.common.config.InitialiseHook;
|
||||||
|
import ca.spottedleaf.moonrise.common.config.adapter.collection.CollectionTypeAdapter;
|
||||||
|
import ca.spottedleaf.moonrise.common.config.adapter.collection.ListTypeAdapter;
|
||||||
|
import ca.spottedleaf.moonrise.common.config.adapter.collection.SortedMapTypeAdapter;
|
||||||
|
import ca.spottedleaf.moonrise.common.config.adapter.primitive.BooleanTypeAdapter;
|
||||||
|
import ca.spottedleaf.moonrise.common.config.adapter.primitive.ByteTypeAdapter;
|
||||||
|
import ca.spottedleaf.moonrise.common.config.adapter.primitive.DoubleTypeAdapter;
|
||||||
|
import ca.spottedleaf.moonrise.common.config.adapter.primitive.FloatTypeAdapter;
|
||||||
|
import ca.spottedleaf.moonrise.common.config.adapter.primitive.IntegerTypeAdapter;
|
||||||
|
import ca.spottedleaf.moonrise.common.config.adapter.primitive.LongTypeAdapter;
|
||||||
|
import ca.spottedleaf.moonrise.common.config.adapter.primitive.ShortTypeAdapter;
|
||||||
|
import ca.spottedleaf.moonrise.common.config.adapter.primitive.StringTypeAdapter;
|
||||||
|
import ca.spottedleaf.moonrise.common.config.adapter.type.BigDecimalTypeAdapter;
|
||||||
|
import ca.spottedleaf.moonrise.common.config.adapter.type.BigIntegerTypeAdapter;
|
||||||
|
import ca.spottedleaf.moonrise.common.config.adapter.type.DurationTypeAdapter;
|
||||||
|
import ca.spottedleaf.moonrise.common.config.annotation.Adaptable;
|
||||||
|
import ca.spottedleaf.moonrise.common.config.annotation.Serializable;
|
||||||
|
import ca.spottedleaf.moonrise.common.config.type.Duration;
|
||||||
|
import java.lang.annotation.Annotation;
|
||||||
|
import java.lang.reflect.Constructor;
|
||||||
|
import java.lang.reflect.Field;
|
||||||
|
import java.lang.reflect.ParameterizedType;
|
||||||
|
import java.lang.reflect.Type;
|
||||||
|
import java.math.BigDecimal;
|
||||||
|
import java.math.BigInteger;
|
||||||
|
import java.util.ArrayList;
|
||||||
|
import java.util.Collection;
|
||||||
|
import java.util.HashMap;
|
||||||
|
import java.util.LinkedHashMap;
|
||||||
|
import java.util.List;
|
||||||
|
import java.util.Map;
|
||||||
|
|
||||||
|
public final class TypeAdapterRegistry {
|
||||||
|
|
||||||
|
private final Map<Class<?>, TypeAdapter<?, ?>> adapters = new HashMap<>();
|
||||||
|
{
|
||||||
|
this.adapters.put(boolean.class, BooleanTypeAdapter.INSTANCE);
|
||||||
|
this.adapters.put(byte.class, ByteTypeAdapter.INSTANCE);
|
||||||
|
this.adapters.put(short.class, ShortTypeAdapter.INSTANCE);
|
||||||
|
this.adapters.put(int.class, IntegerTypeAdapter.INSTANCE);
|
||||||
|
this.adapters.put(long.class, LongTypeAdapter.INSTANCE);
|
||||||
|
this.adapters.put(float.class, FloatTypeAdapter.INSTANCE);
|
||||||
|
this.adapters.put(double.class, DoubleTypeAdapter.INSTANCE);
|
||||||
|
|
||||||
|
this.adapters.put(Boolean.class, BooleanTypeAdapter.INSTANCE);
|
||||||
|
this.adapters.put(Byte.class, ByteTypeAdapter.INSTANCE);
|
||||||
|
this.adapters.put(Short.class, ShortTypeAdapter.INSTANCE);
|
||||||
|
this.adapters.put(Integer.class, IntegerTypeAdapter.INSTANCE);
|
||||||
|
this.adapters.put(Long.class, LongTypeAdapter.INSTANCE);
|
||||||
|
this.adapters.put(Float.class, FloatTypeAdapter.INSTANCE);
|
||||||
|
this.adapters.put(Double.class, DoubleTypeAdapter.INSTANCE);
|
||||||
|
|
||||||
|
this.adapters.put(String.class, StringTypeAdapter.INSTANCE);
|
||||||
|
|
||||||
|
this.adapters.put(Collection.class, CollectionTypeAdapter.INSTANCE);
|
||||||
|
this.adapters.put(List.class, ListTypeAdapter.INSTANCE);
|
||||||
|
this.adapters.put(Map.class, SortedMapTypeAdapter.SORTED_CASE_INSENSITIVE);
|
||||||
|
this.adapters.put(LinkedHashMap.class, SortedMapTypeAdapter.SORTED_CASE_INSENSITIVE);
|
||||||
|
|
||||||
|
this.adapters.put(BigInteger.class, BigIntegerTypeAdapter.INSTANCE);
|
||||||
|
this.adapters.put(BigDecimal.class, BigDecimalTypeAdapter.INSTANCE);
|
||||||
|
this.adapters.put(Duration.class, DurationTypeAdapter.INSTANCE);
|
||||||
|
}
|
||||||
|
|
||||||
|
public TypeAdapter<?, ?> putAdapter(final Class<?> clazz, final TypeAdapter<?, ?> adapter) {
|
||||||
|
return this.adapters.put(clazz, adapter);
|
||||||
|
}
|
||||||
|
|
||||||
|
public TypeAdapter<?, ?> getAdapter(final Class<?> clazz) {
|
||||||
|
return this.adapters.get(clazz);
|
||||||
|
}
|
||||||
|
|
||||||
|
public Object deserialize(final Object input, final Type type) {
|
||||||
|
TypeAdapter<?, ?> adapter = null;
|
||||||
|
if (type instanceof Class<?> clazz) {
|
||||||
|
adapter = this.adapters.get(clazz);
|
||||||
|
}
|
||||||
|
if (adapter == null && (type instanceof ParameterizedType parameterizedType)) {
|
||||||
|
adapter = this.adapters.get((Class<?>)parameterizedType.getRawType());
|
||||||
|
}
|
||||||
|
|
||||||
|
if (adapter == null) {
|
||||||
|
throw new IllegalArgumentException("No adapter for " + input.getClass() + " with type " + type);
|
||||||
|
}
|
||||||
|
|
||||||
|
return ((TypeAdapter)adapter).deserialize(this, input, type);
|
||||||
|
}
|
||||||
|
|
||||||
|
public Object serialize(final Object input, final Type type) {
|
||||||
|
TypeAdapter<?, ?> adapter = null;
|
||||||
|
if (type instanceof Class<?> clazz) {
|
||||||
|
adapter = this.adapters.get(clazz);
|
||||||
|
}
|
||||||
|
if (adapter == null && (type instanceof ParameterizedType parameterizedType)) {
|
||||||
|
adapter = this.adapters.get((Class<?>)parameterizedType.getRawType());
|
||||||
|
}
|
||||||
|
if (adapter == null) {
|
||||||
|
adapter = this.adapters.get(input.getClass());
|
||||||
|
}
|
||||||
|
|
||||||
|
if (adapter == null) {
|
||||||
|
throw new IllegalArgumentException("No adapter for " + input.getClass() + " with type " + type);
|
||||||
|
}
|
||||||
|
|
||||||
|
return ((TypeAdapter)adapter).serialize(this, input, type);
|
||||||
|
}
|
||||||
|
|
||||||
|
public <T> TypeAdapter<T, Map<Object, Object>> makeAdapter(final Class<? extends T> clazz) throws Exception {
|
||||||
|
final TypeAdapter<T, Map<Object, Object>> ret = new AutoTypeAdapter<>(this, clazz);
|
||||||
|
|
||||||
|
this.putAdapter(clazz, ret);
|
||||||
|
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
|
|
||||||
|
public <T> void callInitialisers(final T object) {
|
||||||
|
if (object == null) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
final TypeAdapter<?, ?> adapter = this.getAdapter(object.getClass());
|
||||||
|
|
||||||
|
if (!(adapter instanceof AutoTypeAdapter<?> autoTypeAdapter)) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
((AutoTypeAdapter<T>)autoTypeAdapter).callInitialisers(object);
|
||||||
|
}
|
||||||
|
|
||||||
|
private static final class AutoTypeAdapter<T> extends TypeAdapter<T, Map<Object, Object>> {
|
||||||
|
|
||||||
|
private final TypeAdapterRegistry registry;
|
||||||
|
private final Constructor<? extends T> constructor;
|
||||||
|
private final SerializableField[] fields;
|
||||||
|
|
||||||
|
public AutoTypeAdapter(final TypeAdapterRegistry registry, final Class<? extends T> clazz) throws Exception {
|
||||||
|
this.registry = registry;
|
||||||
|
this.constructor = clazz.getConstructor();
|
||||||
|
this.fields = findSerializableFields(registry, clazz);
|
||||||
|
}
|
||||||
|
|
||||||
|
private static TypeAdapter<?, ?> findOrMakeAdapter(final TypeAdapterRegistry registry, final Class<?> clazz) throws Exception {
|
||||||
|
final TypeAdapter<?, ?> ret = registry.getAdapter(clazz);
|
||||||
|
if (ret != null) {
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
|
|
||||||
|
for (final Annotation annotation : clazz.getAnnotations()) {
|
||||||
|
if (annotation instanceof Adaptable adaptable) {
|
||||||
|
return registry.makeAdapter(clazz);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
throw new IllegalArgumentException("No type adapter for " + clazz + " (Forgot @Adaptable?)");
|
||||||
|
}
|
||||||
|
|
||||||
|
private static String makeSerializedKey(final String input) {
|
||||||
|
final StringBuilder ret = new StringBuilder();
|
||||||
|
|
||||||
|
for (final char c : input.toCharArray()) {
|
||||||
|
if (!Character.isUpperCase(c)) {
|
||||||
|
ret.append(c);
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
ret.append('-');
|
||||||
|
ret.append(Character.toLowerCase(c));
|
||||||
|
}
|
||||||
|
|
||||||
|
return ret.toString();
|
||||||
|
}
|
||||||
|
|
||||||
|
private static record SerializableField(
|
||||||
|
Field field,
|
||||||
|
boolean required,
|
||||||
|
String comment,
|
||||||
|
TypeAdapter<?, ?> adapter,
|
||||||
|
boolean serialize,
|
||||||
|
String serializedKey
|
||||||
|
) {}
|
||||||
|
|
||||||
|
private static SerializableField[] findSerializableFields(final TypeAdapterRegistry registry, Class<?> clazz) throws Exception {
|
||||||
|
final List<SerializableField> ret = new ArrayList<>();
|
||||||
|
do {
|
||||||
|
for (final Field field : clazz.getDeclaredFields()) {
|
||||||
|
field.setAccessible(true);
|
||||||
|
|
||||||
|
for (final Annotation annotation : field.getAnnotations()) {
|
||||||
|
if (!(annotation instanceof Serializable serializable)) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
final TypeAdapter<?, ?> adapter;
|
||||||
|
|
||||||
|
if (serializable.adapter() != TypeAdapter.class) {
|
||||||
|
adapter = serializable.adapter().getConstructor().newInstance();
|
||||||
|
} else {
|
||||||
|
adapter = findOrMakeAdapter(registry, field.getType());
|
||||||
|
}
|
||||||
|
|
||||||
|
String serializedKey = serializable.serializedKey();
|
||||||
|
if (serializedKey.isEmpty()) {
|
||||||
|
serializedKey = makeSerializedKey(field.getName());
|
||||||
|
}
|
||||||
|
|
||||||
|
ret.add(new SerializableField(
|
||||||
|
field, serializable.required(), serializable.comment(), adapter,
|
||||||
|
serializable.serialize(), serializedKey
|
||||||
|
));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} while ((clazz = clazz.getSuperclass()) != Object.class);
|
||||||
|
|
||||||
|
ret.sort((final SerializableField c1, final SerializableField c2) -> {
|
||||||
|
return c1.serializedKey.compareTo(c2.serializedKey);
|
||||||
|
});
|
||||||
|
|
||||||
|
return ret.toArray(new SerializableField[0]);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public T deserialize(final TypeAdapterRegistry registry, final Object input, final Type type) {
|
||||||
|
if (!(input instanceof Map<?,?> inputMap)) {
|
||||||
|
throw new IllegalArgumentException("Not a map type: " + input.getClass());
|
||||||
|
}
|
||||||
|
|
||||||
|
try {
|
||||||
|
final T ret = this.constructor.newInstance();
|
||||||
|
|
||||||
|
for (final SerializableField field : this.fields) {
|
||||||
|
final Object fieldValue = inputMap.get(field.serializedKey);
|
||||||
|
|
||||||
|
if (fieldValue == null) {
|
||||||
|
if (field.required) {
|
||||||
|
throw new IllegalArgumentException("Missing required field '" + field.serializedKey + "' in " + this.constructor.getDeclaringClass());
|
||||||
|
}
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
field.field.set(ret, field.adapter.deserialize(registry, fieldValue, field.field.getGenericType()));
|
||||||
|
}
|
||||||
|
|
||||||
|
return ret;
|
||||||
|
} catch (final Exception ex) {
|
||||||
|
throw new RuntimeException(ex);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public Map<Object, Object> serialize(final TypeAdapterRegistry registry, final T value, final Type type) {
|
||||||
|
final LinkedHashMap<Object, Object> ret = new LinkedHashMap<>();
|
||||||
|
|
||||||
|
for (final SerializableField field : this.fields) {
|
||||||
|
if (!field.serialize) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
final Object fieldValue;
|
||||||
|
try {
|
||||||
|
fieldValue = field.field.get(value);
|
||||||
|
} catch (final Exception ex) {
|
||||||
|
throw new RuntimeException(ex);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (fieldValue != null) {
|
||||||
|
ret.put(
|
||||||
|
field.comment.isBlank() ? field.serializedKey : new CommentedData(field.comment, field.serializedKey),
|
||||||
|
((TypeAdapter)field.adapter).serialize(
|
||||||
|
registry, fieldValue, field.field.getGenericType()
|
||||||
|
)
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void callInitialisers(final T value) {
|
||||||
|
for (final SerializableField field : this.fields) {
|
||||||
|
final Object fieldValue;
|
||||||
|
try {
|
||||||
|
fieldValue = field.field.get(value);
|
||||||
|
} catch (final Exception ex) {
|
||||||
|
throw new RuntimeException(ex);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (fieldValue instanceof InitialiseHook initialiseHook) {
|
||||||
|
initialiseHook.initialise();
|
||||||
|
}
|
||||||
|
|
||||||
|
this.registry.callInitialisers(fieldValue);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public static final class CommentedData {
|
||||||
|
|
||||||
|
public final String comment;
|
||||||
|
public final Object data;
|
||||||
|
|
||||||
|
public CommentedData(final String comment, final Object data) {
|
||||||
|
this.comment = comment;
|
||||||
|
this.data = data;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,46 @@
|
|||||||
|
package ca.spottedleaf.moonrise.common.config.adapter.collection;
|
||||||
|
|
||||||
|
import ca.spottedleaf.moonrise.common.config.adapter.TypeAdapter;
|
||||||
|
import ca.spottedleaf.moonrise.common.config.adapter.TypeAdapterRegistry;
|
||||||
|
import ca.spottedleaf.moonrise.common.config.adapter.primitive.StringTypeAdapter;
|
||||||
|
import java.lang.reflect.ParameterizedType;
|
||||||
|
import java.lang.reflect.Type;
|
||||||
|
import java.util.ArrayList;
|
||||||
|
import java.util.Collection;
|
||||||
|
import java.util.List;
|
||||||
|
|
||||||
|
public final class CollectionTypeAdapter extends TypeAdapter<Collection<Object>, List<Object>> {
|
||||||
|
|
||||||
|
public static final CollectionTypeAdapter INSTANCE = new CollectionTypeAdapter();
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public Collection<Object> deserialize(final TypeAdapterRegistry registry, final Object input, final Type type) {
|
||||||
|
if (!(type instanceof ParameterizedType parameterizedType)) {
|
||||||
|
throw new IllegalArgumentException("Collection field must specify generic type");
|
||||||
|
}
|
||||||
|
final Type elementType = parameterizedType.getActualTypeArguments()[0];
|
||||||
|
if (input instanceof Collection<?> collection) {
|
||||||
|
final List<Object> ret = new ArrayList<>(collection.size());
|
||||||
|
|
||||||
|
for (final Object v : collection) {
|
||||||
|
ret.add(registry.deserialize(v, elementType));
|
||||||
|
}
|
||||||
|
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
|
throw new IllegalArgumentException("Not a collection type: " + input.getClass());
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public List<Object> serialize(final TypeAdapterRegistry registry, final Collection<Object> value, final Type type) {
|
||||||
|
final List<Object> ret = new ArrayList<>(value.size());
|
||||||
|
|
||||||
|
final Type elementType = type instanceof ParameterizedType parameterizedType ? parameterizedType.getActualTypeArguments()[0] : null;
|
||||||
|
|
||||||
|
for (final Object v : value) {
|
||||||
|
ret.add(registry.serialize(v, elementType == null ? v.getClass() : elementType));
|
||||||
|
}
|
||||||
|
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,45 @@
|
|||||||
|
package ca.spottedleaf.moonrise.common.config.adapter.collection;
|
||||||
|
|
||||||
|
import ca.spottedleaf.moonrise.common.config.adapter.TypeAdapter;
|
||||||
|
import ca.spottedleaf.moonrise.common.config.adapter.TypeAdapterRegistry;
|
||||||
|
import java.lang.reflect.ParameterizedType;
|
||||||
|
import java.lang.reflect.Type;
|
||||||
|
import java.util.ArrayList;
|
||||||
|
import java.util.Collection;
|
||||||
|
import java.util.List;
|
||||||
|
|
||||||
|
public final class ListTypeAdapter extends TypeAdapter<List<Object>, List<Object>> {
|
||||||
|
|
||||||
|
public static final ListTypeAdapter INSTANCE = new ListTypeAdapter();
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public List<Object> deserialize(final TypeAdapterRegistry registry, final Object input, final Type type) {
|
||||||
|
if (!(type instanceof ParameterizedType parameterizedType)) {
|
||||||
|
throw new IllegalArgumentException("Collection field must specify generic type");
|
||||||
|
}
|
||||||
|
final Type elementType = parameterizedType.getActualTypeArguments()[0];
|
||||||
|
if (input instanceof Collection<?> collection) {
|
||||||
|
final List<Object> ret = new ArrayList<>(collection.size());
|
||||||
|
|
||||||
|
for (final Object v : collection) {
|
||||||
|
ret.add(registry.deserialize(v, elementType));
|
||||||
|
}
|
||||||
|
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
|
throw new IllegalArgumentException("Not a collection type: " + input.getClass());
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public List<Object> serialize(final TypeAdapterRegistry registry, final List<Object> value, final Type type) {
|
||||||
|
final List<Object> ret = new ArrayList<>(value.size());
|
||||||
|
|
||||||
|
final Type elementType = type instanceof ParameterizedType parameterizedType ? parameterizedType.getActualTypeArguments()[0] : null;
|
||||||
|
|
||||||
|
for (final Object v : value) {
|
||||||
|
ret.add(registry.serialize(v, elementType));
|
||||||
|
}
|
||||||
|
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,59 @@
|
|||||||
|
package ca.spottedleaf.moonrise.common.config.adapter.collection;
|
||||||
|
|
||||||
|
import ca.spottedleaf.moonrise.common.config.adapter.TypeAdapter;
|
||||||
|
import ca.spottedleaf.moonrise.common.config.adapter.TypeAdapterRegistry;
|
||||||
|
import java.lang.reflect.ParameterizedType;
|
||||||
|
import java.lang.reflect.Type;
|
||||||
|
import java.util.Comparator;
|
||||||
|
import java.util.LinkedHashMap;
|
||||||
|
import java.util.Map;
|
||||||
|
import java.util.TreeMap;
|
||||||
|
|
||||||
|
public final class SortedMapTypeAdapter extends TypeAdapter<Map<String, Object>, Map<String, Object>> {
|
||||||
|
|
||||||
|
public static final SortedMapTypeAdapter SORTED_CASE_INSENSITIVE = new SortedMapTypeAdapter(String.CASE_INSENSITIVE_ORDER);
|
||||||
|
public static final SortedMapTypeAdapter SORTED_CASE_SENSITIVE = new SortedMapTypeAdapter(null);
|
||||||
|
|
||||||
|
private final Comparator<String> keyComparator;
|
||||||
|
|
||||||
|
public SortedMapTypeAdapter(final Comparator<String> keyComparator) {
|
||||||
|
this.keyComparator = keyComparator;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public Map<String, Object> deserialize(final TypeAdapterRegistry registry, final Object input, final Type type) {
|
||||||
|
if (!(type instanceof ParameterizedType parameterizedType)) {
|
||||||
|
throw new IllegalArgumentException("Collection field must specify generic type");
|
||||||
|
}
|
||||||
|
final Type valueType = parameterizedType.getActualTypeArguments()[1];
|
||||||
|
if (input instanceof Map<?,?> inputMap) {
|
||||||
|
final Map<String, Object> castedInput = (Map<String, Object>)inputMap;
|
||||||
|
|
||||||
|
final TreeMap<String, Object> ret = new TreeMap<>(this.keyComparator);
|
||||||
|
|
||||||
|
for (final Map.Entry<String, Object> entry : castedInput.entrySet()) {
|
||||||
|
ret.put(entry.getKey(), registry.deserialize(entry.getValue(), valueType));
|
||||||
|
}
|
||||||
|
|
||||||
|
// transform to linked so that get() is O(1)
|
||||||
|
return new LinkedHashMap<>(ret);
|
||||||
|
}
|
||||||
|
|
||||||
|
throw new IllegalArgumentException("Not a map type: " + input.getClass());
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public Map<String, Object> serialize(final TypeAdapterRegistry registry, final Map<String, Object> value, final Type type) {
|
||||||
|
final TreeMap<String, Object> ret = new TreeMap<>(this.keyComparator);
|
||||||
|
|
||||||
|
final Type valueType = type instanceof ParameterizedType parameterizedType ? parameterizedType.getActualTypeArguments()[1] : null;
|
||||||
|
|
||||||
|
for (final Map.Entry<String, Object> entry : value.entrySet()) {
|
||||||
|
ret.put(entry.getKey(), registry.serialize(entry.getValue(), valueType));
|
||||||
|
}
|
||||||
|
|
||||||
|
// transform to linked so that get() is O(1)
|
||||||
|
return new LinkedHashMap<>(ret);
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
@@ -0,0 +1,47 @@
|
|||||||
|
package ca.spottedleaf.moonrise.common.config.adapter.collection;
|
||||||
|
|
||||||
|
import ca.spottedleaf.moonrise.common.config.adapter.TypeAdapter;
|
||||||
|
import ca.spottedleaf.moonrise.common.config.adapter.TypeAdapterRegistry;
|
||||||
|
import java.lang.reflect.ParameterizedType;
|
||||||
|
import java.lang.reflect.Type;
|
||||||
|
import java.util.LinkedHashMap;
|
||||||
|
import java.util.Map;
|
||||||
|
|
||||||
|
public final class UnsortedMapTypeAdapter extends TypeAdapter<Map<String, Object>, Map<String, Object>> {
|
||||||
|
|
||||||
|
public static final UnsortedMapTypeAdapter INSTANCE = new UnsortedMapTypeAdapter();
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public Map<String, Object> deserialize(final TypeAdapterRegistry registry, final Object input, final Type type) {
|
||||||
|
if (!(type instanceof ParameterizedType parameterizedType)) {
|
||||||
|
throw new IllegalArgumentException("Collection field must specify generic type");
|
||||||
|
}
|
||||||
|
final Type valueType = parameterizedType.getActualTypeArguments()[1];
|
||||||
|
if (input instanceof Map<?,?> inputMap) {
|
||||||
|
final Map<String, Object> castedInput = (Map<String, Object>)inputMap;
|
||||||
|
|
||||||
|
final LinkedHashMap<String, Object> ret = new LinkedHashMap<>();
|
||||||
|
|
||||||
|
for (final Map.Entry<String, Object> entry : castedInput.entrySet()) {
|
||||||
|
ret.put(entry.getKey(), registry.deserialize(entry.getValue(), valueType));
|
||||||
|
}
|
||||||
|
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
|
|
||||||
|
throw new IllegalArgumentException("Not a map type: " + input.getClass());
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public Map<String, Object> serialize(final TypeAdapterRegistry registry, final Map<String, Object> value, final Type type) {
|
||||||
|
final LinkedHashMap<String, Object> ret = new LinkedHashMap<>();
|
||||||
|
|
||||||
|
final Type valueType = type instanceof ParameterizedType parameterizedType ? parameterizedType.getActualTypeArguments()[1] : null;
|
||||||
|
|
||||||
|
for (final Map.Entry<String, Object> entry : value.entrySet()) {
|
||||||
|
ret.put(entry.getKey(), registry.serialize(entry.getValue(), valueType));
|
||||||
|
}
|
||||||
|
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,33 @@
|
|||||||
|
package ca.spottedleaf.moonrise.common.config.adapter.primitive;
|
||||||
|
|
||||||
|
import ca.spottedleaf.moonrise.common.config.adapter.TypeAdapter;
|
||||||
|
import ca.spottedleaf.moonrise.common.config.adapter.TypeAdapterRegistry;
|
||||||
|
import java.lang.reflect.Type;
|
||||||
|
|
||||||
|
public final class BooleanTypeAdapter extends TypeAdapter<Boolean, Boolean> {
|
||||||
|
|
||||||
|
public static final BooleanTypeAdapter INSTANCE = new BooleanTypeAdapter();
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public Boolean deserialize(final TypeAdapterRegistry registry, final Object input, final Type type) {
|
||||||
|
if (input instanceof Boolean ret) {
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
|
if (input instanceof String str) {
|
||||||
|
if (str.equalsIgnoreCase("false")) {
|
||||||
|
return Boolean.FALSE;
|
||||||
|
}
|
||||||
|
if (str.equalsIgnoreCase("true")) {
|
||||||
|
return Boolean.TRUE;
|
||||||
|
}
|
||||||
|
throw new IllegalArgumentException("Not a boolean: " + str);
|
||||||
|
}
|
||||||
|
|
||||||
|
throw new IllegalArgumentException("Not a boolean type: " + input.getClass());
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public Boolean serialize(final TypeAdapterRegistry registry, final Boolean value, final Type type) {
|
||||||
|
return value;
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,36 @@
|
|||||||
|
package ca.spottedleaf.moonrise.common.config.adapter.primitive;
|
||||||
|
|
||||||
|
import ca.spottedleaf.moonrise.common.config.adapter.TypeAdapter;
|
||||||
|
import ca.spottedleaf.moonrise.common.config.adapter.TypeAdapterRegistry;
|
||||||
|
import java.lang.reflect.Type;
|
||||||
|
import java.math.BigInteger;
|
||||||
|
|
||||||
|
public final class ByteTypeAdapter extends TypeAdapter<Byte, Byte> {
|
||||||
|
|
||||||
|
public static final ByteTypeAdapter INSTANCE = new ByteTypeAdapter();
|
||||||
|
|
||||||
|
private static Byte cast(final Object original, final long value) {
|
||||||
|
if (value < (long)Byte.MIN_VALUE || value > (long)Byte.MAX_VALUE) {
|
||||||
|
throw new IllegalArgumentException("Byte value is out of range: " + original.toString());
|
||||||
|
}
|
||||||
|
return Byte.valueOf((byte)value);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public Byte deserialize(final TypeAdapterRegistry registry, final Object input, final Type type) {
|
||||||
|
if (input instanceof Number number) {
|
||||||
|
// note: silently discard floating point significand
|
||||||
|
return cast(input, number instanceof BigInteger bigInteger ? bigInteger.longValueExact() : number.longValue());
|
||||||
|
}
|
||||||
|
if (input instanceof String string) {
|
||||||
|
return cast(input, (long)Double.parseDouble(string));
|
||||||
|
}
|
||||||
|
|
||||||
|
throw new IllegalArgumentException("Not a byte type: " + input.getClass());
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public Byte serialize(final TypeAdapterRegistry registry, final Byte value, final Type type) {
|
||||||
|
return value;
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,27 @@
|
|||||||
|
package ca.spottedleaf.moonrise.common.config.adapter.primitive;
|
||||||
|
|
||||||
|
import ca.spottedleaf.moonrise.common.config.adapter.TypeAdapter;
|
||||||
|
import ca.spottedleaf.moonrise.common.config.adapter.TypeAdapterRegistry;
|
||||||
|
import java.lang.reflect.Type;
|
||||||
|
|
||||||
|
public final class DoubleTypeAdapter extends TypeAdapter<Double, Double> {
|
||||||
|
|
||||||
|
public static final DoubleTypeAdapter INSTANCE = new DoubleTypeAdapter();
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public Double deserialize(final TypeAdapterRegistry registry, final Object input, final Type type) {
|
||||||
|
if (input instanceof Number number) {
|
||||||
|
return Double.valueOf(number.doubleValue());
|
||||||
|
}
|
||||||
|
if (input instanceof String string) {
|
||||||
|
return Double.valueOf(Double.parseDouble(string));
|
||||||
|
}
|
||||||
|
|
||||||
|
throw new IllegalArgumentException("Not a byte type: " + input.getClass());
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public Double serialize(final TypeAdapterRegistry registry, final Double value, final Type type) {
|
||||||
|
return value;
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,35 @@
|
|||||||
|
package ca.spottedleaf.moonrise.common.config.adapter.primitive;
|
||||||
|
|
||||||
|
import ca.spottedleaf.moonrise.common.config.adapter.TypeAdapter;
|
||||||
|
import ca.spottedleaf.moonrise.common.config.adapter.TypeAdapterRegistry;
|
||||||
|
import java.lang.reflect.Type;
|
||||||
|
|
||||||
|
public final class FloatTypeAdapter extends TypeAdapter<Float, Float> {
|
||||||
|
|
||||||
|
public static final FloatTypeAdapter INSTANCE = new FloatTypeAdapter();
|
||||||
|
|
||||||
|
private static Float cast(final Object original, final double value) {
|
||||||
|
if (value < -(double)Float.MAX_VALUE || value > (double)Float.MAX_VALUE) {
|
||||||
|
throw new IllegalArgumentException("Byte value is out of range: " + original.toString());
|
||||||
|
}
|
||||||
|
// note: silently ignore precision loss
|
||||||
|
return Float.valueOf((float)value);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public Float deserialize(final TypeAdapterRegistry registry, final Object input, final Type type) {
|
||||||
|
if (input instanceof Number number) {
|
||||||
|
return cast(input, number.doubleValue());
|
||||||
|
}
|
||||||
|
if (input instanceof String string) {
|
||||||
|
return cast(input, Double.parseDouble(string));
|
||||||
|
}
|
||||||
|
|
||||||
|
throw new IllegalArgumentException("Not a byte type: " + input.getClass());
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public Float serialize(final TypeAdapterRegistry registry, final Float value, final Type type) {
|
||||||
|
return value;
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,36 @@
|
|||||||
|
package ca.spottedleaf.moonrise.common.config.adapter.primitive;
|
||||||
|
|
||||||
|
import ca.spottedleaf.moonrise.common.config.adapter.TypeAdapter;
|
||||||
|
import ca.spottedleaf.moonrise.common.config.adapter.TypeAdapterRegistry;
|
||||||
|
import java.lang.reflect.Type;
|
||||||
|
import java.math.BigInteger;
|
||||||
|
|
||||||
|
public final class IntegerTypeAdapter extends TypeAdapter<Integer, Integer> {
|
||||||
|
|
||||||
|
public static final IntegerTypeAdapter INSTANCE = new IntegerTypeAdapter();
|
||||||
|
|
||||||
|
private static Integer cast(final Object original, final long value) {
|
||||||
|
if (value < (long)Integer.MIN_VALUE || value > (long)Integer.MAX_VALUE) {
|
||||||
|
throw new IllegalArgumentException("Integer value is out of range: " + original.toString());
|
||||||
|
}
|
||||||
|
return Integer.valueOf((int)value);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public Integer deserialize(final TypeAdapterRegistry registry, final Object input, final Type type) {
|
||||||
|
if (input instanceof Number number) {
|
||||||
|
// note: silently discard floating point significand
|
||||||
|
return cast(input, number instanceof BigInteger bigInteger ? bigInteger.longValueExact() : number.longValue());
|
||||||
|
}
|
||||||
|
if (input instanceof String string) {
|
||||||
|
return cast(input, (long)Double.parseDouble(string));
|
||||||
|
}
|
||||||
|
|
||||||
|
throw new IllegalArgumentException("Not an integer type: " + input.getClass());
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public Integer serialize(final TypeAdapterRegistry registry, final Integer value, final Type type) {
|
||||||
|
return value;
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,33 @@
|
|||||||
|
package ca.spottedleaf.moonrise.common.config.adapter.primitive;
|
||||||
|
|
||||||
|
import ca.spottedleaf.moonrise.common.config.adapter.TypeAdapter;
|
||||||
|
import ca.spottedleaf.moonrise.common.config.adapter.TypeAdapterRegistry;
|
||||||
|
import java.lang.reflect.Type;
|
||||||
|
import java.math.BigInteger;
|
||||||
|
|
||||||
|
public final class LongTypeAdapter extends TypeAdapter<Long, Long> {
|
||||||
|
|
||||||
|
public static final LongTypeAdapter INSTANCE = new LongTypeAdapter();
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public Long deserialize(final TypeAdapterRegistry registry, final Object input, final Type type) {
|
||||||
|
if (input instanceof Number number) {
|
||||||
|
// note: silently discard floating point significand
|
||||||
|
return number instanceof BigInteger bigInteger ? bigInteger.longValueExact() : number.longValue();
|
||||||
|
}
|
||||||
|
if (input instanceof String string) {
|
||||||
|
try {
|
||||||
|
return Long.valueOf(Long.parseLong(string));
|
||||||
|
} catch (final NumberFormatException ex) {
|
||||||
|
return Long.valueOf((long)Double.parseDouble(string));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
throw new IllegalArgumentException("Not a long type: " + input.getClass());
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public Long serialize(final TypeAdapterRegistry registry, final Long value, final Type type) {
|
||||||
|
return value;
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,36 @@
|
|||||||
|
package ca.spottedleaf.moonrise.common.config.adapter.primitive;
|
||||||
|
|
||||||
|
import ca.spottedleaf.moonrise.common.config.adapter.TypeAdapter;
|
||||||
|
import ca.spottedleaf.moonrise.common.config.adapter.TypeAdapterRegistry;
|
||||||
|
import java.lang.reflect.Type;
|
||||||
|
import java.math.BigInteger;
|
||||||
|
|
||||||
|
public final class ShortTypeAdapter extends TypeAdapter<Short, Short> {
|
||||||
|
|
||||||
|
public static final ShortTypeAdapter INSTANCE = new ShortTypeAdapter();
|
||||||
|
|
||||||
|
private static Short cast(final Object original, final long value) {
|
||||||
|
if (value < (long)Short.MIN_VALUE || value > (long)Short.MAX_VALUE) {
|
||||||
|
throw new IllegalArgumentException("Short value is out of range: " + original.toString());
|
||||||
|
}
|
||||||
|
return Short.valueOf((short)value);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public Short deserialize(final TypeAdapterRegistry registry, final Object input, final Type type) {
|
||||||
|
if (input instanceof Number number) {
|
||||||
|
// note: silently discard floating point significand
|
||||||
|
return cast(input, number instanceof BigInteger bigInteger ? bigInteger.longValueExact() : number.longValue());
|
||||||
|
}
|
||||||
|
if (input instanceof String string) {
|
||||||
|
return cast(input, (long)Double.parseDouble(string));
|
||||||
|
}
|
||||||
|
|
||||||
|
throw new IllegalArgumentException("Not a short type: " + input.getClass());
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public Short serialize(final TypeAdapterRegistry registry, final Short value, final Type type) {
|
||||||
|
return value;
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,29 @@
|
|||||||
|
package ca.spottedleaf.moonrise.common.config.adapter.primitive;
|
||||||
|
|
||||||
|
import ca.spottedleaf.moonrise.common.config.adapter.TypeAdapter;
|
||||||
|
import ca.spottedleaf.moonrise.common.config.adapter.TypeAdapterRegistry;
|
||||||
|
import java.lang.reflect.Type;
|
||||||
|
|
||||||
|
public final class StringTypeAdapter extends TypeAdapter<String, String> {
|
||||||
|
|
||||||
|
public static final StringTypeAdapter INSTANCE = new StringTypeAdapter();
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public String deserialize(final TypeAdapterRegistry registry, final Object input, final Type type) {
|
||||||
|
if (input instanceof Boolean bool) {
|
||||||
|
return String.valueOf(bool.booleanValue());
|
||||||
|
}
|
||||||
|
if (input instanceof Number number) {
|
||||||
|
return number.toString();
|
||||||
|
}
|
||||||
|
if (input instanceof String string) {
|
||||||
|
return string;
|
||||||
|
}
|
||||||
|
throw new IllegalArgumentException("Not a string type: " + input.getClass());
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public String serialize(final TypeAdapterRegistry registry, final String value, final Type type) {
|
||||||
|
return value;
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,30 @@
|
|||||||
|
package ca.spottedleaf.moonrise.common.config.adapter.type;
|
||||||
|
|
||||||
|
import ca.spottedleaf.moonrise.common.config.adapter.TypeAdapter;
|
||||||
|
import ca.spottedleaf.moonrise.common.config.adapter.TypeAdapterRegistry;
|
||||||
|
import java.lang.reflect.Type;
|
||||||
|
import java.math.BigDecimal;
|
||||||
|
import java.math.BigInteger;
|
||||||
|
|
||||||
|
public final class BigDecimalTypeAdapter extends TypeAdapter<BigDecimal, String> {
|
||||||
|
|
||||||
|
public static final BigDecimalTypeAdapter INSTANCE = new BigDecimalTypeAdapter();
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public BigDecimal deserialize(final TypeAdapterRegistry registry, final Object input, final Type type) {
|
||||||
|
if (input instanceof Number number) {
|
||||||
|
// safest to catch all number impls is to use toString
|
||||||
|
return new BigDecimal(number.toString());
|
||||||
|
}
|
||||||
|
if (input instanceof String string) {
|
||||||
|
return new BigDecimal(string);
|
||||||
|
}
|
||||||
|
|
||||||
|
throw new IllegalArgumentException("Not an BigDecimal type: " + input.getClass());
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public String serialize(final TypeAdapterRegistry registry, final BigDecimal value, final Type type) {
|
||||||
|
return value.toString();
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,37 @@
|
|||||||
|
package ca.spottedleaf.moonrise.common.config.adapter.type;
|
||||||
|
|
||||||
|
import ca.spottedleaf.moonrise.common.config.adapter.TypeAdapter;
|
||||||
|
import ca.spottedleaf.moonrise.common.config.adapter.TypeAdapterRegistry;
|
||||||
|
import java.lang.reflect.Type;
|
||||||
|
import java.math.BigDecimal;
|
||||||
|
import java.math.BigInteger;
|
||||||
|
|
||||||
|
public final class BigIntegerTypeAdapter extends TypeAdapter<BigInteger, String> {
|
||||||
|
|
||||||
|
public static final BigIntegerTypeAdapter INSTANCE = new BigIntegerTypeAdapter();
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public BigInteger deserialize(final TypeAdapterRegistry registry, final Object input, final Type type) {
|
||||||
|
if (input instanceof Number number) {
|
||||||
|
if (number instanceof BigInteger bigInteger) {
|
||||||
|
return bigInteger;
|
||||||
|
}
|
||||||
|
// note: silently discard floating point significand
|
||||||
|
if (number instanceof BigDecimal bigDecimal) {
|
||||||
|
return bigDecimal.toBigInteger();
|
||||||
|
}
|
||||||
|
|
||||||
|
return BigInteger.valueOf(number.longValue());
|
||||||
|
}
|
||||||
|
if (input instanceof String string) {
|
||||||
|
return new BigDecimal(string).toBigInteger();
|
||||||
|
}
|
||||||
|
|
||||||
|
throw new IllegalArgumentException("Not an BigInteger type: " + input.getClass());
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public String serialize(final TypeAdapterRegistry registry, final BigInteger value, final Type type) {
|
||||||
|
return value.toString();
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,24 @@
|
|||||||
|
package ca.spottedleaf.moonrise.common.config.adapter.type;
|
||||||
|
|
||||||
|
import ca.spottedleaf.moonrise.common.config.adapter.TypeAdapter;
|
||||||
|
import ca.spottedleaf.moonrise.common.config.adapter.TypeAdapterRegistry;
|
||||||
|
import ca.spottedleaf.moonrise.common.config.type.Duration;
|
||||||
|
import java.lang.reflect.Type;
|
||||||
|
|
||||||
|
public final class DurationTypeAdapter extends TypeAdapter<Duration, String> {
|
||||||
|
|
||||||
|
public static final DurationTypeAdapter INSTANCE = new DurationTypeAdapter();
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public Duration deserialize(final TypeAdapterRegistry registry, final Object input, final Type type) {
|
||||||
|
if (!(input instanceof String string)) {
|
||||||
|
throw new IllegalArgumentException("Not a string: " + input.getClass());
|
||||||
|
}
|
||||||
|
return Duration.parse(string);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public String serialize(final TypeAdapterRegistry registry, final Duration value, final Type type) {
|
||||||
|
return value.toString();
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,15 @@
|
|||||||
|
package ca.spottedleaf.moonrise.common.config.annotation;
|
||||||
|
|
||||||
|
import java.lang.annotation.ElementType;
|
||||||
|
import java.lang.annotation.Retention;
|
||||||
|
import java.lang.annotation.RetentionPolicy;
|
||||||
|
import java.lang.annotation.Target;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Annotation used on a class to indicate that its type adapter may automatically be generated. The class must have
|
||||||
|
* a public no-args constructor.
|
||||||
|
*/
|
||||||
|
@Retention(RetentionPolicy.RUNTIME)
|
||||||
|
@Target(ElementType.TYPE)
|
||||||
|
public @interface Adaptable {
|
||||||
|
}
|
||||||
@@ -0,0 +1,45 @@
|
|||||||
|
package ca.spottedleaf.moonrise.common.config.annotation;
|
||||||
|
|
||||||
|
import ca.spottedleaf.moonrise.common.config.adapter.TypeAdapter;
|
||||||
|
import java.lang.annotation.ElementType;
|
||||||
|
import java.lang.annotation.Retention;
|
||||||
|
import java.lang.annotation.RetentionPolicy;
|
||||||
|
import java.lang.annotation.Target;
|
||||||
|
import java.util.function.Function;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Annotation indicating that a field should be deserialized or serialized from the config.
|
||||||
|
* By default, this annotation is not assumed.
|
||||||
|
*/
|
||||||
|
@Retention(RetentionPolicy.RUNTIME)
|
||||||
|
@Target(ElementType.FIELD)
|
||||||
|
public @interface Serializable {
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Indicates whether this field is required to be present in the config. If the field is not present,
|
||||||
|
* and {@code required = true}, then an exception will be thrown during deserialization. If {@code required = false}
|
||||||
|
* and the field is not present, then the field value will remain unmodified.
|
||||||
|
*/
|
||||||
|
public boolean required() default false;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* The comment to apply before the element when serializing.
|
||||||
|
*/
|
||||||
|
public String comment() default "";
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Adapter override class. The class must have a public no-args constructor.
|
||||||
|
*/
|
||||||
|
public Class<? extends TypeAdapter> adapter() default TypeAdapter.class;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Whether to serialize the value to the config.
|
||||||
|
*/
|
||||||
|
public boolean serialize() default true;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* When not empty, this value overrides the auto generated serialized key in the config.
|
||||||
|
*/
|
||||||
|
public String serializedKey() default "";
|
||||||
|
|
||||||
|
}
|
||||||
@@ -0,0 +1,178 @@
|
|||||||
|
package ca.spottedleaf.moonrise.common.config.config;
|
||||||
|
|
||||||
|
import ca.spottedleaf.moonrise.common.config.adapter.TypeAdapterRegistry;
|
||||||
|
import org.yaml.snakeyaml.DumperOptions;
|
||||||
|
import org.yaml.snakeyaml.LoaderOptions;
|
||||||
|
import org.yaml.snakeyaml.Yaml;
|
||||||
|
import org.yaml.snakeyaml.comments.CommentLine;
|
||||||
|
import org.yaml.snakeyaml.comments.CommentType;
|
||||||
|
import org.yaml.snakeyaml.constructor.Constructor;
|
||||||
|
import org.yaml.snakeyaml.nodes.Node;
|
||||||
|
import org.yaml.snakeyaml.representer.Represent;
|
||||||
|
import org.yaml.snakeyaml.representer.Representer;
|
||||||
|
import java.io.BufferedInputStream;
|
||||||
|
import java.io.BufferedOutputStream;
|
||||||
|
import java.io.File;
|
||||||
|
import java.io.FileInputStream;
|
||||||
|
import java.io.FileOutputStream;
|
||||||
|
import java.io.IOException;
|
||||||
|
import java.io.InputStream;
|
||||||
|
import java.io.InputStreamReader;
|
||||||
|
import java.io.OutputStream;
|
||||||
|
import java.io.StringReader;
|
||||||
|
import java.nio.charset.StandardCharsets;
|
||||||
|
import java.nio.file.AtomicMoveNotSupportedException;
|
||||||
|
import java.nio.file.Files;
|
||||||
|
import java.nio.file.StandardCopyOption;
|
||||||
|
import java.util.ArrayList;
|
||||||
|
import java.util.List;
|
||||||
|
import java.util.Map;
|
||||||
|
import java.util.TreeMap;
|
||||||
|
|
||||||
|
public final class YamlConfig<T> {
|
||||||
|
|
||||||
|
public final TypeAdapterRegistry typeAdapters;
|
||||||
|
|
||||||
|
private final Class<? extends T> clazz;
|
||||||
|
|
||||||
|
public volatile T config;
|
||||||
|
|
||||||
|
private final Yaml yaml;
|
||||||
|
private final LoaderOptions loaderOptions;
|
||||||
|
private final DumperOptions dumperOptions;
|
||||||
|
|
||||||
|
public YamlConfig(final Class<? extends T> clazz, final T dfl) throws Exception {
|
||||||
|
this(clazz, dfl, new TypeAdapterRegistry());
|
||||||
|
}
|
||||||
|
|
||||||
|
public YamlConfig(final Class<? extends T> clazz, final T dfl, final TypeAdapterRegistry registry) throws Exception {
|
||||||
|
this.clazz = clazz;
|
||||||
|
this.config = dfl;
|
||||||
|
this.typeAdapters = registry;
|
||||||
|
this.typeAdapters.makeAdapter(clazz);
|
||||||
|
|
||||||
|
final LoaderOptions loaderOptions = new LoaderOptions();
|
||||||
|
loaderOptions.setProcessComments(true);
|
||||||
|
|
||||||
|
final DumperOptions dumperOptions = new DumperOptions();
|
||||||
|
dumperOptions.setProcessComments(true);
|
||||||
|
dumperOptions.setDefaultFlowStyle(DumperOptions.FlowStyle.BLOCK);
|
||||||
|
|
||||||
|
this.loaderOptions = loaderOptions;
|
||||||
|
this.dumperOptions = dumperOptions;
|
||||||
|
this.yaml = new Yaml(new YamlConstructor(loaderOptions), new YamlRepresenter(dumperOptions), dumperOptions, loaderOptions);
|
||||||
|
}
|
||||||
|
|
||||||
|
public void load(final File file) throws IOException {
|
||||||
|
try (final InputStream is = new BufferedInputStream(new FileInputStream(file))) {
|
||||||
|
this.load(is);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public void load(final InputStream is) throws IOException {
|
||||||
|
final Object serialized = this.yaml.load(new InputStreamReader(is, StandardCharsets.UTF_8));
|
||||||
|
|
||||||
|
this.config = (T)this.typeAdapters.deserialize(serialized, this.clazz);
|
||||||
|
}
|
||||||
|
|
||||||
|
public void save(final File file) throws IOException {
|
||||||
|
this.save(file, "");
|
||||||
|
}
|
||||||
|
|
||||||
|
public void save(final File file, final String header) throws IOException {
|
||||||
|
if (file.isDirectory()) {
|
||||||
|
throw new IOException("File is a directory");
|
||||||
|
}
|
||||||
|
|
||||||
|
final File parent = file.getParentFile();
|
||||||
|
if (parent != null) {
|
||||||
|
parent.mkdirs();
|
||||||
|
}
|
||||||
|
|
||||||
|
final File tmp = new File(parent, file.getName() + ".tmp");
|
||||||
|
tmp.delete();
|
||||||
|
tmp.createNewFile();
|
||||||
|
try {
|
||||||
|
try (final OutputStream os = new BufferedOutputStream(new FileOutputStream(tmp))) {
|
||||||
|
this.save(os, header);
|
||||||
|
}
|
||||||
|
|
||||||
|
try {
|
||||||
|
Files.move(tmp.toPath(), file.toPath(), StandardCopyOption.REPLACE_EXISTING, StandardCopyOption.ATOMIC_MOVE);
|
||||||
|
} catch (final AtomicMoveNotSupportedException ex) {
|
||||||
|
Files.move(tmp.toPath(), file.toPath(), StandardCopyOption.REPLACE_EXISTING);
|
||||||
|
}
|
||||||
|
} finally {
|
||||||
|
tmp.delete();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public void save(final OutputStream os) throws IOException {
|
||||||
|
os.write(this.saveToString().getBytes(StandardCharsets.UTF_8));
|
||||||
|
}
|
||||||
|
|
||||||
|
public void save(final OutputStream os, final String header) throws IOException {
|
||||||
|
os.write(this.saveToString(header).getBytes(StandardCharsets.UTF_8));
|
||||||
|
}
|
||||||
|
|
||||||
|
public String saveToString() {
|
||||||
|
return this.yaml.dump(this.typeAdapters.serialize(this.config, this.clazz));
|
||||||
|
}
|
||||||
|
|
||||||
|
public String saveToString(final String header) {
|
||||||
|
if (header.isBlank()) {
|
||||||
|
return this.saveToString();
|
||||||
|
}
|
||||||
|
|
||||||
|
final StringBuilder ret = new StringBuilder();
|
||||||
|
final String lineBreak = this.dumperOptions.getLineBreak().getString();
|
||||||
|
|
||||||
|
for (final String line : header.split("\n")) {
|
||||||
|
ret.append("# ").append(line.trim()).append(lineBreak);
|
||||||
|
}
|
||||||
|
|
||||||
|
ret.append(lineBreak);
|
||||||
|
|
||||||
|
return ret.append(this.saveToString()).toString();
|
||||||
|
}
|
||||||
|
|
||||||
|
public void callInitialisers() {
|
||||||
|
this.typeAdapters.callInitialisers(this.config);
|
||||||
|
}
|
||||||
|
|
||||||
|
private static final class YamlConstructor extends Constructor {
|
||||||
|
|
||||||
|
public YamlConstructor(final LoaderOptions loadingConfig) {
|
||||||
|
super(loadingConfig);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private static final class YamlRepresenter extends Representer {
|
||||||
|
|
||||||
|
public YamlRepresenter(final DumperOptions options) {
|
||||||
|
super(options);
|
||||||
|
|
||||||
|
this.representers.put(TypeAdapterRegistry.CommentedData.class, new CommentedDataRepresenter());
|
||||||
|
}
|
||||||
|
|
||||||
|
private final class CommentedDataRepresenter implements Represent {
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public Node representData(final Object data0) {
|
||||||
|
final TypeAdapterRegistry.CommentedData commentedData = (TypeAdapterRegistry.CommentedData)data0;
|
||||||
|
|
||||||
|
final Node node = YamlRepresenter.this.representData(commentedData.data);
|
||||||
|
|
||||||
|
final List<CommentLine> comments = new ArrayList<>();
|
||||||
|
|
||||||
|
for (final String line : commentedData.comment.split("\n")) {
|
||||||
|
comments.add(new CommentLine(null, null, " ".concat(line.trim()), CommentType.BLOCK));
|
||||||
|
}
|
||||||
|
|
||||||
|
node.setBlockComments(comments);
|
||||||
|
|
||||||
|
return node;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -1,12 +1,12 @@
|
|||||||
package ca.spottedleaf.moonrise.common.config.moonrise;
|
package ca.spottedleaf.moonrise.common.config.moonrise;
|
||||||
|
|
||||||
|
import ca.spottedleaf.moonrise.common.config.InitialiseHook;
|
||||||
|
import ca.spottedleaf.moonrise.common.config.annotation.Adaptable;
|
||||||
import ca.spottedleaf.moonrise.common.config.ui.ClothConfig;
|
import ca.spottedleaf.moonrise.common.config.ui.ClothConfig;
|
||||||
|
import ca.spottedleaf.moonrise.common.config.annotation.Serializable;
|
||||||
|
import ca.spottedleaf.moonrise.common.config.type.Duration;
|
||||||
import ca.spottedleaf.moonrise.common.util.MoonriseCommon;
|
import ca.spottedleaf.moonrise.common.util.MoonriseCommon;
|
||||||
import ca.spottedleaf.moonrise.patches.chunk_system.scheduling.ChunkTaskScheduler;
|
import ca.spottedleaf.moonrise.patches.chunk_system.scheduling.ChunkTaskScheduler;
|
||||||
import ca.spottedleaf.yamlconfig.InitialiseHook;
|
|
||||||
import ca.spottedleaf.yamlconfig.annotation.Adaptable;
|
|
||||||
import ca.spottedleaf.yamlconfig.annotation.Serializable;
|
|
||||||
import ca.spottedleaf.yamlconfig.type.Duration;
|
|
||||||
|
|
||||||
@Adaptable
|
@Adaptable
|
||||||
public final class MoonriseConfig {
|
public final class MoonriseConfig {
|
||||||
@@ -251,21 +251,4 @@ public final class MoonriseConfig {
|
|||||||
)
|
)
|
||||||
public boolean fixMC159283 = false;
|
public boolean fixMC159283 = false;
|
||||||
}
|
}
|
||||||
|
|
||||||
@Serializable
|
|
||||||
public Misc misc = new Misc();
|
|
||||||
|
|
||||||
@Adaptable
|
|
||||||
public static final class Misc {
|
|
||||||
|
|
||||||
@Serializable(
|
|
||||||
serializedKey = "immediately-close-loading-screen",
|
|
||||||
comment = """
|
|
||||||
Whether the loading screen should be closed immediately when joining servers/SP worlds.
|
|
||||||
This will let you in game faster, but may result in getting in game before enough chunks are
|
|
||||||
loaded for rendering.
|
|
||||||
"""
|
|
||||||
)
|
|
||||||
public boolean immediatelyCloseLoadingScreen = false;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -0,0 +1,38 @@
|
|||||||
|
package ca.spottedleaf.moonrise.common.config.moonrise.adapter;
|
||||||
|
|
||||||
|
import ca.spottedleaf.moonrise.common.config.adapter.TypeAdapter;
|
||||||
|
import ca.spottedleaf.moonrise.common.config.adapter.TypeAdapterRegistry;
|
||||||
|
import ca.spottedleaf.moonrise.common.config.moonrise.type.DefaultedValue;
|
||||||
|
import java.lang.reflect.ParameterizedType;
|
||||||
|
import java.lang.reflect.Type;
|
||||||
|
|
||||||
|
public final class DefaultedTypeAdapter extends TypeAdapter<DefaultedValue<?>, Object> {
|
||||||
|
|
||||||
|
private static final String DEFAULT_STRING = "default";
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public DefaultedValue<?> deserialize(final TypeAdapterRegistry registry, final Object input, final Type type) {
|
||||||
|
if (input instanceof String string && string.equalsIgnoreCase(DEFAULT_STRING)) {
|
||||||
|
return new DefaultedValue<>();
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!(type instanceof ParameterizedType parameterizedType)) {
|
||||||
|
throw new IllegalArgumentException("DefaultedValue field must specify generic type");
|
||||||
|
}
|
||||||
|
final Type valueType = parameterizedType.getActualTypeArguments()[0];
|
||||||
|
|
||||||
|
return new DefaultedValue<>(registry.deserialize(input, valueType));
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public Object serialize(final TypeAdapterRegistry registry, final DefaultedValue<?> value, final Type type) {
|
||||||
|
final Object raw = value.getValueRaw();
|
||||||
|
if (raw == null) {
|
||||||
|
return DEFAULT_STRING;
|
||||||
|
}
|
||||||
|
|
||||||
|
final Type valueType = type instanceof ParameterizedType parameterizedType ? parameterizedType.getActualTypeArguments()[0] : null;
|
||||||
|
|
||||||
|
return registry.serialize(raw, valueType);
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,22 @@
|
|||||||
|
package ca.spottedleaf.moonrise.common.config.moonrise.type;
|
||||||
|
|
||||||
|
public final class DefaultedValue<T> {
|
||||||
|
|
||||||
|
private final T value;
|
||||||
|
|
||||||
|
public DefaultedValue() {
|
||||||
|
this(null);
|
||||||
|
}
|
||||||
|
|
||||||
|
public DefaultedValue(final T value) {
|
||||||
|
this.value = value;
|
||||||
|
}
|
||||||
|
|
||||||
|
public T getValueRaw() {
|
||||||
|
return value;
|
||||||
|
}
|
||||||
|
|
||||||
|
public T getOrDefault(final T dfl) {
|
||||||
|
return this.value != null ? this.value : dfl;
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,76 @@
|
|||||||
|
package ca.spottedleaf.moonrise.common.config.type;
|
||||||
|
|
||||||
|
import java.math.BigDecimal;
|
||||||
|
|
||||||
|
public final class Duration {
|
||||||
|
|
||||||
|
private final String string;
|
||||||
|
private final long timeNS;
|
||||||
|
|
||||||
|
private Duration(final String string, final long timeNS) {
|
||||||
|
this.string = string;
|
||||||
|
this.timeNS = timeNS;
|
||||||
|
}
|
||||||
|
|
||||||
|
public static Duration parse(final String value) {
|
||||||
|
if (value.length() < 2) {
|
||||||
|
throw new IllegalArgumentException("Invalid duration: " + value);
|
||||||
|
}
|
||||||
|
|
||||||
|
final char last = value.charAt(value.length() - 1);
|
||||||
|
|
||||||
|
final long multiplier;
|
||||||
|
|
||||||
|
switch (last) {
|
||||||
|
case 's': {
|
||||||
|
multiplier = (1000L * 1000L * 1000L) * 1L;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
case 't': {
|
||||||
|
multiplier = (1000L * 1000L * 1000L) / 20L;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
case 'm': {
|
||||||
|
multiplier = (1000L * 1000L * 1000L) * 60L;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
case 'h': {
|
||||||
|
multiplier = (1000L * 1000L * 1000L) * 60L * 60L;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
case 'd': {
|
||||||
|
multiplier = (1000L * 1000L * 1000L) * 24L * 60L * 60L;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
default: {
|
||||||
|
throw new IllegalArgumentException("Duration must end with one of: [s, t, m, h, d]");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
final BigDecimal parsed = new BigDecimal(value.substring(0, value.length() - 1))
|
||||||
|
.multiply(new BigDecimal(multiplier));
|
||||||
|
|
||||||
|
return new Duration(value, parsed.toBigInteger().longValueExact());
|
||||||
|
}
|
||||||
|
|
||||||
|
public long getTimeNS() {
|
||||||
|
return this.timeNS;
|
||||||
|
}
|
||||||
|
|
||||||
|
public long getTimeMS() {
|
||||||
|
return this.timeNS / (1000L * 1000L);
|
||||||
|
}
|
||||||
|
|
||||||
|
public long getTimeS() {
|
||||||
|
return this.timeNS / (1000L * 1000L * 1000L);
|
||||||
|
}
|
||||||
|
|
||||||
|
public long getTimeTicks() {
|
||||||
|
return this.timeNS / ((1000L * 1000L * 1000L) / (20L));
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public String toString() {
|
||||||
|
return this.string;
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -2,7 +2,6 @@ package ca.spottedleaf.moonrise.common.list;
|
|||||||
|
|
||||||
import it.unimi.dsi.fastutil.objects.Reference2IntLinkedOpenHashMap;
|
import it.unimi.dsi.fastutil.objects.Reference2IntLinkedOpenHashMap;
|
||||||
import it.unimi.dsi.fastutil.objects.Reference2IntMap;
|
import it.unimi.dsi.fastutil.objects.Reference2IntMap;
|
||||||
import java.lang.reflect.Array;
|
|
||||||
import java.util.Arrays;
|
import java.util.Arrays;
|
||||||
import java.util.NoSuchElementException;
|
import java.util.NoSuchElementException;
|
||||||
|
|
||||||
@@ -22,34 +21,15 @@ public final class IteratorSafeOrderedReferenceSet<E> {
|
|||||||
private int iteratorCount;
|
private int iteratorCount;
|
||||||
|
|
||||||
public IteratorSafeOrderedReferenceSet() {
|
public IteratorSafeOrderedReferenceSet() {
|
||||||
this(Object.class);
|
this(16, 0.75f, 16, 0.2);
|
||||||
}
|
|
||||||
|
|
||||||
public IteratorSafeOrderedReferenceSet(final Class<? super E> arrComponent) {
|
|
||||||
this(16, 0.75f, 16, 0.2, arrComponent);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
public IteratorSafeOrderedReferenceSet(final int setCapacity, final float setLoadFactor, final int arrayCapacity,
|
public IteratorSafeOrderedReferenceSet(final int setCapacity, final float setLoadFactor, final int arrayCapacity,
|
||||||
final double maxFragFactor) {
|
final double maxFragFactor) {
|
||||||
this(setCapacity, setLoadFactor, arrayCapacity, maxFragFactor, Object.class);
|
|
||||||
}
|
|
||||||
|
|
||||||
public IteratorSafeOrderedReferenceSet(final int setCapacity, final float setLoadFactor, final int arrayCapacity,
|
|
||||||
final double maxFragFactor, final Class<? super E> arrComponent) {
|
|
||||||
this.indexMap = new Reference2IntLinkedOpenHashMap<>(setCapacity, setLoadFactor);
|
this.indexMap = new Reference2IntLinkedOpenHashMap<>(setCapacity, setLoadFactor);
|
||||||
this.indexMap.defaultReturnValue(-1);
|
this.indexMap.defaultReturnValue(-1);
|
||||||
this.maxFragFactor = maxFragFactor;
|
this.maxFragFactor = maxFragFactor;
|
||||||
this.listElements = (E[])Array.newInstance(arrComponent, arrayCapacity);
|
this.listElements = (E[])new Object[arrayCapacity];
|
||||||
}
|
|
||||||
|
|
||||||
// includes null (gravestone) elements
|
|
||||||
public E[] getListRaw() {
|
|
||||||
return this.listElements;
|
|
||||||
}
|
|
||||||
|
|
||||||
// includes null (gravestone) elements
|
|
||||||
public int getListSize() {
|
|
||||||
return this.listSize;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
@@ -101,7 +81,7 @@ public final class IteratorSafeOrderedReferenceSet<E> {
|
|||||||
public int createRawIterator() {
|
public int createRawIterator() {
|
||||||
++this.iteratorCount;
|
++this.iteratorCount;
|
||||||
if (this.indexMap.isEmpty()) {
|
if (this.indexMap.isEmpty()) {
|
||||||
return Integer.MAX_VALUE;
|
return -1;
|
||||||
} else {
|
} else {
|
||||||
return this.firstInvalidIndex == 0 ? this.indexMap.getInt(this.indexMap.firstKey()) : 0;
|
return this.firstInvalidIndex == 0 ? this.indexMap.getInt(this.indexMap.firstKey()) : 0;
|
||||||
}
|
}
|
||||||
@@ -116,7 +96,7 @@ public final class IteratorSafeOrderedReferenceSet<E> {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
return Integer.MAX_VALUE;
|
return -1;
|
||||||
}
|
}
|
||||||
|
|
||||||
public void finishRawIterator() {
|
public void finishRawIterator() {
|
||||||
@@ -225,6 +205,10 @@ public final class IteratorSafeOrderedReferenceSet<E> {
|
|||||||
//this.check();
|
//this.check();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public E rawGet(final int index) {
|
||||||
|
return this.listElements[index];
|
||||||
|
}
|
||||||
|
|
||||||
public int size() {
|
public int size() {
|
||||||
// always returns the correct amount - listSize can be different
|
// always returns the correct amount - listSize can be different
|
||||||
return this.indexMap.size();
|
return this.indexMap.size();
|
||||||
|
|||||||
@@ -1,9 +1,9 @@
|
|||||||
package ca.spottedleaf.moonrise.common.misc;
|
package ca.spottedleaf.moonrise.common.misc;
|
||||||
|
|
||||||
import ca.spottedleaf.moonrise.common.PlatformHooks;
|
|
||||||
import ca.spottedleaf.moonrise.common.list.ReferenceList;
|
import ca.spottedleaf.moonrise.common.list.ReferenceList;
|
||||||
import ca.spottedleaf.moonrise.common.util.CoordinateUtils;
|
import ca.spottedleaf.moonrise.common.util.CoordinateUtils;
|
||||||
import ca.spottedleaf.moonrise.common.util.MoonriseConstants;
|
import ca.spottedleaf.moonrise.common.util.MoonriseConstants;
|
||||||
|
import ca.spottedleaf.moonrise.common.util.ChunkSystem;
|
||||||
import ca.spottedleaf.moonrise.patches.chunk_system.level.ChunkSystemLevel;
|
import ca.spottedleaf.moonrise.patches.chunk_system.level.ChunkSystemLevel;
|
||||||
import ca.spottedleaf.moonrise.patches.chunk_system.level.chunk.ChunkData;
|
import ca.spottedleaf.moonrise.patches.chunk_system.level.chunk.ChunkData;
|
||||||
import ca.spottedleaf.moonrise.patches.chunk_tick_iteration.ChunkTickConstants;
|
import ca.spottedleaf.moonrise.patches.chunk_tick_iteration.ChunkTickConstants;
|
||||||
@@ -121,8 +121,8 @@ public final class NearbyPlayers {
|
|||||||
players[NearbyMapType.GENERAL.ordinal()].update(chunk.x, chunk.z, GENERAL_AREA_VIEW_DISTANCE);
|
players[NearbyMapType.GENERAL.ordinal()].update(chunk.x, chunk.z, GENERAL_AREA_VIEW_DISTANCE);
|
||||||
players[NearbyMapType.GENERAL_SMALL.ordinal()].update(chunk.x, chunk.z, GENERAL_SMALL_VIEW_DISTANCE);
|
players[NearbyMapType.GENERAL_SMALL.ordinal()].update(chunk.x, chunk.z, GENERAL_SMALL_VIEW_DISTANCE);
|
||||||
players[NearbyMapType.GENERAL_REALLY_SMALL.ordinal()].update(chunk.x, chunk.z, GENERAL_REALLY_SMALL_VIEW_DISTANCE);
|
players[NearbyMapType.GENERAL_REALLY_SMALL.ordinal()].update(chunk.x, chunk.z, GENERAL_REALLY_SMALL_VIEW_DISTANCE);
|
||||||
players[NearbyMapType.TICK_VIEW_DISTANCE.ordinal()].update(chunk.x, chunk.z, PlatformHooks.get().getTickViewDistance(player));
|
players[NearbyMapType.TICK_VIEW_DISTANCE.ordinal()].update(chunk.x, chunk.z, ChunkSystem.getTickViewDistance(player));
|
||||||
players[NearbyMapType.VIEW_DISTANCE.ordinal()].update(chunk.x, chunk.z, PlatformHooks.get().getViewDistance(player));
|
players[NearbyMapType.VIEW_DISTANCE.ordinal()].update(chunk.x, chunk.z, ChunkSystem.getLoadViewDistance(player));
|
||||||
players[NearbyMapType.SPAWN_RANGE.ordinal()].update(chunk.x, chunk.z, ChunkTickConstants.PLAYER_SPAWN_TRACK_RANGE); // Moonrise - chunk tick iteration
|
players[NearbyMapType.SPAWN_RANGE.ordinal()].update(chunk.x, chunk.z, ChunkTickConstants.PLAYER_SPAWN_TRACK_RANGE); // Moonrise - chunk tick iteration
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -1,8 +1,7 @@
|
|||||||
package ca.spottedleaf.moonrise.common.misc;
|
package ca.spottedleaf.moonrise.common.misc;
|
||||||
|
|
||||||
import ca.spottedleaf.moonrise.common.util.CoordinateUtils;
|
import ca.spottedleaf.concurrentutil.util.IntPairUtil;
|
||||||
import it.unimi.dsi.fastutil.longs.Long2IntOpenHashMap;
|
import it.unimi.dsi.fastutil.longs.Long2IntOpenHashMap;
|
||||||
import it.unimi.dsi.fastutil.longs.LongSet;
|
|
||||||
import it.unimi.dsi.fastutil.objects.Reference2ReferenceOpenHashMap;
|
import it.unimi.dsi.fastutil.objects.Reference2ReferenceOpenHashMap;
|
||||||
import it.unimi.dsi.fastutil.objects.ReferenceSet;
|
import it.unimi.dsi.fastutil.objects.ReferenceSet;
|
||||||
|
|
||||||
@@ -15,24 +14,16 @@ public final class PositionCountingAreaMap<T> {
|
|||||||
return this.counters.keySet();
|
return this.counters.keySet();
|
||||||
}
|
}
|
||||||
|
|
||||||
public LongSet getPositions() {
|
|
||||||
return this.positions.keySet();
|
|
||||||
}
|
|
||||||
|
|
||||||
public int getTotalPositions() {
|
public int getTotalPositions() {
|
||||||
return this.positions.size();
|
return this.positions.size();
|
||||||
}
|
}
|
||||||
|
|
||||||
public boolean hasObjectsNear(final long pos) {
|
|
||||||
return this.positions.containsKey(pos);
|
|
||||||
}
|
|
||||||
|
|
||||||
public boolean hasObjectsNear(final int toX, final int toZ) {
|
public boolean hasObjectsNear(final int toX, final int toZ) {
|
||||||
return this.positions.containsKey(CoordinateUtils.getChunkKey(toX, toZ));
|
return this.positions.containsKey(IntPairUtil.key(toX, toZ));
|
||||||
}
|
}
|
||||||
|
|
||||||
public int getObjectsNear(final int toX, final int toZ) {
|
public int getObjectsNear(final int toX, final int toZ) {
|
||||||
return this.positions.get(CoordinateUtils.getChunkKey(toX, toZ));
|
return this.positions.get(IntPairUtil.key(toX, toZ));
|
||||||
}
|
}
|
||||||
|
|
||||||
public boolean add(final T parameter, final int toX, final int toZ, final int distance) {
|
public boolean add(final T parameter, final int toX, final int toZ, final int distance) {
|
||||||
@@ -89,12 +80,12 @@ public final class PositionCountingAreaMap<T> {
|
|||||||
|
|
||||||
@Override
|
@Override
|
||||||
protected void addCallback(final T parameter, final int toX, final int toZ) {
|
protected void addCallback(final T parameter, final int toX, final int toZ) {
|
||||||
PositionCountingAreaMap.this.positions.addTo(CoordinateUtils.getChunkKey(toX, toZ), 1);
|
PositionCountingAreaMap.this.positions.addTo(IntPairUtil.key(toX, toZ), 1);
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
protected void removeCallback(final T parameter, final int toX, final int toZ) {
|
protected void removeCallback(final T parameter, final int toX, final int toZ) {
|
||||||
final long key = CoordinateUtils.getChunkKey(toX, toZ);
|
final long key = IntPairUtil.key(toX, toZ);
|
||||||
if (PositionCountingAreaMap.this.positions.addTo(key, -1) == 1) {
|
if (PositionCountingAreaMap.this.positions.addTo(key, -1) == 1) {
|
||||||
PositionCountingAreaMap.this.positions.remove(key);
|
PositionCountingAreaMap.this.positions.remove(key);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,5 +1,7 @@
|
|||||||
package ca.spottedleaf.moonrise.common.misc;
|
package ca.spottedleaf.moonrise.common.misc;
|
||||||
|
|
||||||
|
import ca.spottedleaf.concurrentutil.util.IntegerUtil;
|
||||||
|
|
||||||
public abstract class SingleUserAreaMap<T> {
|
public abstract class SingleUserAreaMap<T> {
|
||||||
|
|
||||||
public static final int NOT_SET = Integer.MIN_VALUE;
|
public static final int NOT_SET = Integer.MIN_VALUE;
|
||||||
@@ -97,8 +99,8 @@ public abstract class SingleUserAreaMap<T> {
|
|||||||
final int dx = toX - fromX;
|
final int dx = toX - fromX;
|
||||||
final int dz = toZ - fromZ;
|
final int dz = toZ - fromZ;
|
||||||
|
|
||||||
final int totalX = Math.abs(fromX - toX);
|
final int totalX = IntegerUtil.branchlessAbs(fromX - toX);
|
||||||
final int totalZ = Math.abs(fromZ - toZ);
|
final int totalZ = IntegerUtil.branchlessAbs(fromZ - toZ);
|
||||||
|
|
||||||
if (Math.max(totalX, totalZ) > (2 * Math.max(newViewDistance, oldViewDistance))) {
|
if (Math.max(totalX, totalZ) > (2 * Math.max(newViewDistance, oldViewDistance))) {
|
||||||
// teleported
|
// teleported
|
||||||
@@ -118,7 +120,7 @@ public abstract class SingleUserAreaMap<T> {
|
|||||||
for (int currZ = oldMinZ; currZ <= oldMaxZ; ++currZ) {
|
for (int currZ = oldMinZ; currZ <= oldMaxZ; ++currZ) {
|
||||||
|
|
||||||
// only remove if we're outside the new view distance...
|
// only remove if we're outside the new view distance...
|
||||||
if (Math.max(Math.abs(currX - toX), Math.abs(currZ - toZ)) > newViewDistance) {
|
if (Math.max(IntegerUtil.branchlessAbs(currX - toX), IntegerUtil.branchlessAbs(currZ - toZ)) > newViewDistance) {
|
||||||
this.removeCallback(parameter, currX, currZ);
|
this.removeCallback(parameter, currX, currZ);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -134,7 +136,7 @@ public abstract class SingleUserAreaMap<T> {
|
|||||||
for (int currZ = newMinZ; currZ <= newMaxZ; ++currZ) {
|
for (int currZ = newMinZ; currZ <= newMaxZ; ++currZ) {
|
||||||
|
|
||||||
// only add if we're outside the old view distance...
|
// only add if we're outside the old view distance...
|
||||||
if (Math.max(Math.abs(currX - fromX), Math.abs(currZ - fromZ)) > oldViewDistance) {
|
if (Math.max(IntegerUtil.branchlessAbs(currX - fromX), IntegerUtil.branchlessAbs(currZ - fromZ)) > oldViewDistance) {
|
||||||
this.addCallback(parameter, currX, currZ);
|
this.addCallback(parameter, currX, currZ);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,176 +0,0 @@
|
|||||||
package ca.spottedleaf.moonrise.common.util;
|
|
||||||
|
|
||||||
import ca.spottedleaf.concurrentutil.util.Priority;
|
|
||||||
import ca.spottedleaf.moonrise.patches.chunk_system.level.ChunkSystemServerLevel;
|
|
||||||
import ca.spottedleaf.moonrise.patches.chunk_system.level.chunk.ChunkSystemLevelChunk;
|
|
||||||
import ca.spottedleaf.moonrise.patches.chunk_system.player.RegionizedPlayerChunkLoader;
|
|
||||||
import ca.spottedleaf.moonrise.patches.chunk_system.world.ChunkSystemServerChunkCache;
|
|
||||||
import ca.spottedleaf.moonrise.patches.chunk_tick_iteration.ChunkTickServerLevel;
|
|
||||||
import net.minecraft.server.level.ChunkHolder;
|
|
||||||
import net.minecraft.server.level.FullChunkStatus;
|
|
||||||
import net.minecraft.server.level.ServerLevel;
|
|
||||||
import net.minecraft.server.level.ServerPlayer;
|
|
||||||
import net.minecraft.server.level.progress.ChunkProgressListener;
|
|
||||||
import net.minecraft.world.level.chunk.ChunkAccess;
|
|
||||||
import net.minecraft.world.level.chunk.LevelChunk;
|
|
||||||
import net.minecraft.world.level.chunk.status.ChunkStatus;
|
|
||||||
import java.util.List;
|
|
||||||
import java.util.function.Consumer;
|
|
||||||
|
|
||||||
public abstract class BaseChunkSystemHooks implements ChunkSystemHooks {
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public void scheduleChunkTask(final ServerLevel level, final int chunkX, final int chunkZ, final Runnable run) {
|
|
||||||
this.scheduleChunkTask(level, chunkX, chunkZ, run, Priority.NORMAL);
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public void scheduleChunkTask(final ServerLevel level, final int chunkX, final int chunkZ, final Runnable run, final Priority priority) {
|
|
||||||
((ChunkSystemServerLevel)level).moonrise$getChunkTaskScheduler().scheduleChunkTask(chunkX, chunkZ, run, priority);
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public void scheduleChunkLoad(final ServerLevel level, final int chunkX, final int chunkZ, final boolean gen,
|
|
||||||
final ChunkStatus toStatus, final boolean addTicket, final Priority priority,
|
|
||||||
final Consumer<ChunkAccess> onComplete) {
|
|
||||||
((ChunkSystemServerLevel)level).moonrise$getChunkTaskScheduler().scheduleChunkLoad(chunkX, chunkZ, gen, toStatus, addTicket, priority, onComplete);
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public void scheduleChunkLoad(final ServerLevel level, final int chunkX, final int chunkZ, final ChunkStatus toStatus,
|
|
||||||
final boolean addTicket, final Priority priority, final Consumer<ChunkAccess> onComplete) {
|
|
||||||
((ChunkSystemServerLevel)level).moonrise$getChunkTaskScheduler().scheduleChunkLoad(chunkX, chunkZ, toStatus, addTicket, priority, onComplete);
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public void scheduleTickingState(final ServerLevel level, final int chunkX, final int chunkZ,
|
|
||||||
final FullChunkStatus toStatus, final boolean addTicket,
|
|
||||||
final Priority priority, final Consumer<LevelChunk> onComplete) {
|
|
||||||
((ChunkSystemServerLevel)level).moonrise$getChunkTaskScheduler().scheduleTickingState(chunkX, chunkZ, toStatus, addTicket, priority, onComplete);
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public List<ChunkHolder> getVisibleChunkHolders(final ServerLevel level) {
|
|
||||||
return ((ChunkSystemServerLevel)level).moonrise$getChunkTaskScheduler().chunkHolderManager.getOldChunkHolders();
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public List<ChunkHolder> getUpdatingChunkHolders(final ServerLevel level) {
|
|
||||||
return ((ChunkSystemServerLevel)level).moonrise$getChunkTaskScheduler().chunkHolderManager.getOldChunkHolders();
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public int getVisibleChunkHolderCount(final ServerLevel level) {
|
|
||||||
return ((ChunkSystemServerLevel)level).moonrise$getChunkTaskScheduler().chunkHolderManager.size();
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public int getUpdatingChunkHolderCount(final ServerLevel level) {
|
|
||||||
return ((ChunkSystemServerLevel)level).moonrise$getChunkTaskScheduler().chunkHolderManager.size();
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public boolean hasAnyChunkHolders(final ServerLevel level) {
|
|
||||||
return this.getUpdatingChunkHolderCount(level) != 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public void onChunkHolderCreate(final ServerLevel level, final ChunkHolder holder) {
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public void onChunkHolderDelete(final ServerLevel level, final ChunkHolder holder) {
|
|
||||||
// Update progress listener for LevelLoadingScreen
|
|
||||||
final ChunkProgressListener progressListener = level.getChunkSource().chunkMap.progressListener;
|
|
||||||
if (progressListener != null) {
|
|
||||||
this.scheduleChunkTask(level, holder.getPos().x, holder.getPos().z, () -> {
|
|
||||||
progressListener.onStatusChange(holder.getPos(), null);
|
|
||||||
});
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public void onChunkPreBorder(final LevelChunk chunk, final ChunkHolder holder) {
|
|
||||||
((ChunkSystemServerChunkCache)((ServerLevel)chunk.getLevel()).getChunkSource())
|
|
||||||
.moonrise$setFullChunk(chunk.getPos().x, chunk.getPos().z, chunk);
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public void onChunkBorder(final LevelChunk chunk, final ChunkHolder holder) {
|
|
||||||
((ChunkSystemServerLevel)((ServerLevel)chunk.getLevel())).moonrise$getLoadedChunks().add(chunk);
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public void onChunkNotBorder(final LevelChunk chunk, final ChunkHolder holder) {
|
|
||||||
((ChunkSystemServerLevel)((ServerLevel)chunk.getLevel())).moonrise$getLoadedChunks().remove(chunk);
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public void onChunkPostNotBorder(final LevelChunk chunk, final ChunkHolder holder) {
|
|
||||||
((ChunkSystemServerChunkCache)((ServerLevel)chunk.getLevel()).getChunkSource())
|
|
||||||
.moonrise$setFullChunk(chunk.getPos().x, chunk.getPos().z, null);
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public void onChunkTicking(final LevelChunk chunk, final ChunkHolder holder) {
|
|
||||||
((ChunkSystemServerLevel)((ServerLevel)chunk.getLevel())).moonrise$getTickingChunks().add(chunk);
|
|
||||||
if (!((ChunkSystemLevelChunk)chunk).moonrise$isPostProcessingDone()) {
|
|
||||||
chunk.postProcessGeneration((ServerLevel)chunk.getLevel());
|
|
||||||
}
|
|
||||||
((ServerLevel)chunk.getLevel()).startTickingChunk(chunk);
|
|
||||||
((ServerLevel)chunk.getLevel()).getChunkSource().chunkMap.tickingGenerated.incrementAndGet();
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public void onChunkNotTicking(final LevelChunk chunk, final ChunkHolder holder) {
|
|
||||||
((ChunkSystemServerLevel)((ServerLevel)chunk.getLevel())).moonrise$getTickingChunks().remove(chunk);
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public void onChunkEntityTicking(final LevelChunk chunk, final ChunkHolder holder) {
|
|
||||||
((ChunkSystemServerLevel)((ServerLevel)chunk.getLevel())).moonrise$getEntityTickingChunks().add(chunk);
|
|
||||||
((ChunkTickServerLevel)(ServerLevel)chunk.getLevel()).moonrise$markChunkForPlayerTicking(chunk); // Moonrise - chunk tick iteration
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public void onChunkNotEntityTicking(final LevelChunk chunk, final ChunkHolder holder) {
|
|
||||||
((ChunkSystemServerLevel)((ServerLevel)chunk.getLevel())).moonrise$getEntityTickingChunks().remove(chunk);
|
|
||||||
((ChunkTickServerLevel)(ServerLevel)chunk.getLevel()).moonrise$removeChunkForPlayerTicking(chunk); // Moonrise - chunk tick iteration
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public ChunkHolder getUnloadingChunkHolder(final ServerLevel level, final int chunkX, final int chunkZ) {
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public int getSendViewDistance(final ServerPlayer player) {
|
|
||||||
return RegionizedPlayerChunkLoader.getAPISendViewDistance(player);
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public int getViewDistance(final ServerPlayer player) {
|
|
||||||
return RegionizedPlayerChunkLoader.getAPIViewDistance(player);
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public int getTickViewDistance(final ServerPlayer player) {
|
|
||||||
return RegionizedPlayerChunkLoader.getAPITickViewDistance(player);
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public void addPlayerToDistanceMaps(final ServerLevel world, final ServerPlayer player) {
|
|
||||||
((ChunkSystemServerLevel)world).moonrise$getPlayerChunkLoader().addPlayer(player);
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public void removePlayerFromDistanceMaps(final ServerLevel world, final ServerPlayer player) {
|
|
||||||
((ChunkSystemServerLevel)world).moonrise$getPlayerChunkLoader().removePlayer(player);
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public void updateMaps(final ServerLevel world, final ServerPlayer player) {
|
|
||||||
((ChunkSystemServerLevel)world).moonrise$getPlayerChunkLoader().updatePlayer(player);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -0,0 +1,176 @@
|
|||||||
|
package ca.spottedleaf.moonrise.common.util;
|
||||||
|
|
||||||
|
import ca.spottedleaf.concurrentutil.util.Priority;
|
||||||
|
import ca.spottedleaf.moonrise.common.PlatformHooks;
|
||||||
|
import ca.spottedleaf.moonrise.patches.chunk_system.level.ChunkSystemServerLevel;
|
||||||
|
import ca.spottedleaf.moonrise.patches.chunk_system.level.chunk.ChunkSystemLevelChunk;
|
||||||
|
import ca.spottedleaf.moonrise.patches.chunk_system.player.RegionizedPlayerChunkLoader;
|
||||||
|
import ca.spottedleaf.moonrise.patches.chunk_system.world.ChunkSystemServerChunkCache;
|
||||||
|
import ca.spottedleaf.moonrise.patches.chunk_tick_iteration.ChunkTickServerLevel;
|
||||||
|
import com.mojang.logging.LogUtils;
|
||||||
|
import net.minecraft.server.level.ChunkHolder;
|
||||||
|
import net.minecraft.server.level.FullChunkStatus;
|
||||||
|
import net.minecraft.server.level.ServerLevel;
|
||||||
|
import net.minecraft.server.level.ServerPlayer;
|
||||||
|
import net.minecraft.server.level.progress.ChunkProgressListener;
|
||||||
|
import net.minecraft.world.entity.Entity;
|
||||||
|
import net.minecraft.world.level.chunk.ChunkAccess;
|
||||||
|
import net.minecraft.world.level.chunk.LevelChunk;
|
||||||
|
import net.minecraft.world.level.chunk.status.ChunkStatus;
|
||||||
|
import org.slf4j.Logger;
|
||||||
|
import java.util.List;
|
||||||
|
import java.util.function.Consumer;
|
||||||
|
|
||||||
|
public final class ChunkSystem {
|
||||||
|
|
||||||
|
private static final Logger LOGGER = LogUtils.getLogger();
|
||||||
|
|
||||||
|
public static void scheduleChunkTask(final ServerLevel level, final int chunkX, final int chunkZ, final Runnable run) {
|
||||||
|
scheduleChunkTask(level, chunkX, chunkZ, run, Priority.NORMAL);
|
||||||
|
}
|
||||||
|
|
||||||
|
public static void scheduleChunkTask(final ServerLevel level, final int chunkX, final int chunkZ, final Runnable run, final Priority priority) {
|
||||||
|
((ChunkSystemServerLevel)level).moonrise$getChunkTaskScheduler().scheduleChunkTask(chunkX, chunkZ, run, priority);
|
||||||
|
}
|
||||||
|
|
||||||
|
public static void scheduleChunkLoad(final ServerLevel level, final int chunkX, final int chunkZ, final boolean gen,
|
||||||
|
final ChunkStatus toStatus, final boolean addTicket, final Priority priority,
|
||||||
|
final Consumer<ChunkAccess> onComplete) {
|
||||||
|
((ChunkSystemServerLevel)level).moonrise$getChunkTaskScheduler().scheduleChunkLoad(chunkX, chunkZ, gen, toStatus, addTicket, priority, onComplete);
|
||||||
|
}
|
||||||
|
|
||||||
|
public static void scheduleChunkLoad(final ServerLevel level, final int chunkX, final int chunkZ, final ChunkStatus toStatus,
|
||||||
|
final boolean addTicket, final Priority priority, final Consumer<ChunkAccess> onComplete) {
|
||||||
|
((ChunkSystemServerLevel)level).moonrise$getChunkTaskScheduler().scheduleChunkLoad(chunkX, chunkZ, toStatus, addTicket, priority, onComplete);
|
||||||
|
}
|
||||||
|
|
||||||
|
public static void scheduleTickingState(final ServerLevel level, final int chunkX, final int chunkZ,
|
||||||
|
final FullChunkStatus toStatus, final boolean addTicket,
|
||||||
|
final Priority priority, final Consumer<LevelChunk> onComplete) {
|
||||||
|
((ChunkSystemServerLevel)level).moonrise$getChunkTaskScheduler().scheduleTickingState(chunkX, chunkZ, toStatus, addTicket, priority, onComplete);
|
||||||
|
}
|
||||||
|
|
||||||
|
public static List<ChunkHolder> getVisibleChunkHolders(final ServerLevel level) {
|
||||||
|
return ((ChunkSystemServerLevel)level).moonrise$getChunkTaskScheduler().chunkHolderManager.getOldChunkHolders();
|
||||||
|
}
|
||||||
|
|
||||||
|
public static List<ChunkHolder> getUpdatingChunkHolders(final ServerLevel level) {
|
||||||
|
return ((ChunkSystemServerLevel)level).moonrise$getChunkTaskScheduler().chunkHolderManager.getOldChunkHolders();
|
||||||
|
}
|
||||||
|
|
||||||
|
public static int getVisibleChunkHolderCount(final ServerLevel level) {
|
||||||
|
return ((ChunkSystemServerLevel)level).moonrise$getChunkTaskScheduler().chunkHolderManager.size();
|
||||||
|
}
|
||||||
|
|
||||||
|
public static int getUpdatingChunkHolderCount(final ServerLevel level) {
|
||||||
|
return ((ChunkSystemServerLevel)level).moonrise$getChunkTaskScheduler().chunkHolderManager.size();
|
||||||
|
}
|
||||||
|
|
||||||
|
public static boolean hasAnyChunkHolders(final ServerLevel level) {
|
||||||
|
return getUpdatingChunkHolderCount(level) != 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
public static boolean screenEntity(final ServerLevel level, final Entity entity, final boolean fromDisk, final boolean event) {
|
||||||
|
if (!PlatformHooks.get().screenEntity(level, entity, fromDisk, event)) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
public static void onChunkHolderCreate(final ServerLevel level, final ChunkHolder holder) {
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
public static void onChunkHolderDelete(final ServerLevel level, final ChunkHolder holder) {
|
||||||
|
// Update progress listener for LevelLoadingScreen
|
||||||
|
final ChunkProgressListener progressListener = level.getChunkSource().chunkMap.progressListener;
|
||||||
|
if (progressListener != null) {
|
||||||
|
ChunkSystem.scheduleChunkTask(level, holder.getPos().x, holder.getPos().z, () -> {
|
||||||
|
progressListener.onStatusChange(holder.getPos(), null);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public static void onChunkPreBorder(final LevelChunk chunk, final ChunkHolder holder) {
|
||||||
|
((ChunkSystemServerChunkCache)((ServerLevel)chunk.getLevel()).getChunkSource())
|
||||||
|
.moonrise$setFullChunk(chunk.getPos().x, chunk.getPos().z, chunk);
|
||||||
|
}
|
||||||
|
|
||||||
|
public static void onChunkBorder(final LevelChunk chunk, final ChunkHolder holder) {
|
||||||
|
((ChunkSystemServerLevel)((ServerLevel)chunk.getLevel())).moonrise$getLoadedChunks().add(
|
||||||
|
((ChunkSystemLevelChunk)chunk).moonrise$getChunkAndHolder()
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
public static void onChunkNotBorder(final LevelChunk chunk, final ChunkHolder holder) {
|
||||||
|
((ChunkSystemServerLevel)((ServerLevel)chunk.getLevel())).moonrise$getLoadedChunks().remove(
|
||||||
|
((ChunkSystemLevelChunk)chunk).moonrise$getChunkAndHolder()
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
public static void onChunkPostNotBorder(final LevelChunk chunk, final ChunkHolder holder) {
|
||||||
|
((ChunkSystemServerChunkCache)((ServerLevel)chunk.getLevel()).getChunkSource())
|
||||||
|
.moonrise$setFullChunk(chunk.getPos().x, chunk.getPos().z, null);
|
||||||
|
}
|
||||||
|
|
||||||
|
public static void onChunkTicking(final LevelChunk chunk, final ChunkHolder holder) {
|
||||||
|
((ChunkSystemServerLevel)((ServerLevel)chunk.getLevel())).moonrise$getTickingChunks().add(
|
||||||
|
((ChunkSystemLevelChunk)chunk).moonrise$getChunkAndHolder()
|
||||||
|
);
|
||||||
|
if (!((ChunkSystemLevelChunk)chunk).moonrise$isPostProcessingDone()) {
|
||||||
|
chunk.postProcessGeneration();
|
||||||
|
}
|
||||||
|
((ServerLevel)chunk.getLevel()).startTickingChunk(chunk);
|
||||||
|
((ServerLevel)chunk.getLevel()).getChunkSource().chunkMap.tickingGenerated.incrementAndGet();
|
||||||
|
((ChunkTickServerLevel)(ServerLevel)chunk.getLevel()).moonrise$markChunkForPlayerTicking(chunk); // Moonrise - chunk tick iteration
|
||||||
|
}
|
||||||
|
|
||||||
|
public static void onChunkNotTicking(final LevelChunk chunk, final ChunkHolder holder) {
|
||||||
|
((ChunkSystemServerLevel)((ServerLevel)chunk.getLevel())).moonrise$getTickingChunks().remove(
|
||||||
|
((ChunkSystemLevelChunk)chunk).moonrise$getChunkAndHolder()
|
||||||
|
);
|
||||||
|
((ChunkTickServerLevel)(ServerLevel)chunk.getLevel()).moonrise$removeChunkForPlayerTicking(chunk); // Moonrise - chunk tick iteration
|
||||||
|
}
|
||||||
|
|
||||||
|
public static void onChunkEntityTicking(final LevelChunk chunk, final ChunkHolder holder) {
|
||||||
|
((ChunkSystemServerLevel)((ServerLevel)chunk.getLevel())).moonrise$getEntityTickingChunks().add(
|
||||||
|
((ChunkSystemLevelChunk)chunk).moonrise$getChunkAndHolder()
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
public static void onChunkNotEntityTicking(final LevelChunk chunk, final ChunkHolder holder) {
|
||||||
|
((ChunkSystemServerLevel)((ServerLevel)chunk.getLevel())).moonrise$getEntityTickingChunks().remove(
|
||||||
|
((ChunkSystemLevelChunk)chunk).moonrise$getChunkAndHolder()
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
public static ChunkHolder getUnloadingChunkHolder(final ServerLevel level, final int chunkX, final int chunkZ) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
public static int getSendViewDistance(final ServerPlayer player) {
|
||||||
|
return RegionizedPlayerChunkLoader.getAPISendViewDistance(player);
|
||||||
|
}
|
||||||
|
|
||||||
|
public static int getLoadViewDistance(final ServerPlayer player) {
|
||||||
|
return RegionizedPlayerChunkLoader.getLoadViewDistance(player);
|
||||||
|
}
|
||||||
|
|
||||||
|
public static int getTickViewDistance(final ServerPlayer player) {
|
||||||
|
return RegionizedPlayerChunkLoader.getAPITickViewDistance(player);
|
||||||
|
}
|
||||||
|
|
||||||
|
public static void addPlayerToDistanceMaps(final ServerLevel world, final ServerPlayer player) {
|
||||||
|
((ChunkSystemServerLevel)world).moonrise$getPlayerChunkLoader().addPlayer(player);
|
||||||
|
}
|
||||||
|
|
||||||
|
public static void removePlayerFromDistanceMaps(final ServerLevel world, final ServerPlayer player) {
|
||||||
|
((ChunkSystemServerLevel)world).moonrise$getPlayerChunkLoader().removePlayer(player);
|
||||||
|
}
|
||||||
|
|
||||||
|
public static void updateMaps(final ServerLevel world, final ServerPlayer player) {
|
||||||
|
((ChunkSystemServerLevel)world).moonrise$getPlayerChunkLoader().updatePlayer(player);
|
||||||
|
}
|
||||||
|
|
||||||
|
private ChunkSystem() {}
|
||||||
|
}
|
||||||
@@ -1,80 +0,0 @@
|
|||||||
package ca.spottedleaf.moonrise.common.util;
|
|
||||||
|
|
||||||
import ca.spottedleaf.concurrentutil.util.Priority;
|
|
||||||
import net.minecraft.server.level.ChunkHolder;
|
|
||||||
import net.minecraft.server.level.FullChunkStatus;
|
|
||||||
import net.minecraft.server.level.ServerLevel;
|
|
||||||
import net.minecraft.server.level.ServerPlayer;
|
|
||||||
import net.minecraft.server.level.TicketType;
|
|
||||||
import net.minecraft.world.entity.Entity;
|
|
||||||
import net.minecraft.world.level.chunk.ChunkAccess;
|
|
||||||
import net.minecraft.world.level.chunk.LevelChunk;
|
|
||||||
import net.minecraft.world.level.chunk.status.ChunkStatus;
|
|
||||||
import java.util.List;
|
|
||||||
import java.util.function.Consumer;
|
|
||||||
|
|
||||||
public interface ChunkSystemHooks {
|
|
||||||
|
|
||||||
public void scheduleChunkTask(final ServerLevel level, final int chunkX, final int chunkZ, final Runnable run);
|
|
||||||
|
|
||||||
public void scheduleChunkTask(final ServerLevel level, final int chunkX, final int chunkZ, final Runnable run, final Priority priority);
|
|
||||||
|
|
||||||
public void scheduleChunkLoad(final ServerLevel level, final int chunkX, final int chunkZ, final boolean gen,
|
|
||||||
final ChunkStatus toStatus, final boolean addTicket, final Priority priority,
|
|
||||||
final Consumer<ChunkAccess> onComplete);
|
|
||||||
|
|
||||||
public void scheduleChunkLoad(final ServerLevel level, final int chunkX, final int chunkZ, final ChunkStatus toStatus,
|
|
||||||
final boolean addTicket, final Priority priority, final Consumer<ChunkAccess> onComplete);
|
|
||||||
|
|
||||||
public void scheduleTickingState(final ServerLevel level, final int chunkX, final int chunkZ,
|
|
||||||
final FullChunkStatus toStatus, final boolean addTicket,
|
|
||||||
final Priority priority, final Consumer<LevelChunk> onComplete);
|
|
||||||
|
|
||||||
public List<ChunkHolder> getVisibleChunkHolders(final ServerLevel level);
|
|
||||||
|
|
||||||
public List<ChunkHolder> getUpdatingChunkHolders(final ServerLevel level);
|
|
||||||
|
|
||||||
public int getVisibleChunkHolderCount(final ServerLevel level);
|
|
||||||
|
|
||||||
public int getUpdatingChunkHolderCount(final ServerLevel level);
|
|
||||||
|
|
||||||
public boolean hasAnyChunkHolders(final ServerLevel level);
|
|
||||||
|
|
||||||
public boolean screenEntity(final ServerLevel level, final Entity entity, final boolean fromDisk, final boolean event);
|
|
||||||
|
|
||||||
public void onChunkHolderCreate(final ServerLevel level, final ChunkHolder holder);
|
|
||||||
|
|
||||||
public void onChunkHolderDelete(final ServerLevel level, final ChunkHolder holder);
|
|
||||||
|
|
||||||
public void onChunkPreBorder(final LevelChunk chunk, final ChunkHolder holder);
|
|
||||||
|
|
||||||
public void onChunkBorder(final LevelChunk chunk, final ChunkHolder holder);
|
|
||||||
|
|
||||||
public void onChunkNotBorder(final LevelChunk chunk, final ChunkHolder holder);
|
|
||||||
|
|
||||||
public void onChunkPostNotBorder(final LevelChunk chunk, final ChunkHolder holder);
|
|
||||||
|
|
||||||
public void onChunkTicking(final LevelChunk chunk, final ChunkHolder holder);
|
|
||||||
|
|
||||||
public void onChunkNotTicking(final LevelChunk chunk, final ChunkHolder holder);
|
|
||||||
|
|
||||||
public void onChunkEntityTicking(final LevelChunk chunk, final ChunkHolder holder);
|
|
||||||
|
|
||||||
public void onChunkNotEntityTicking(final LevelChunk chunk, final ChunkHolder holder);
|
|
||||||
|
|
||||||
public ChunkHolder getUnloadingChunkHolder(final ServerLevel level, final int chunkX, final int chunkZ);
|
|
||||||
|
|
||||||
public int getSendViewDistance(final ServerPlayer player);
|
|
||||||
|
|
||||||
public int getViewDistance(final ServerPlayer player);
|
|
||||||
|
|
||||||
public int getTickViewDistance(final ServerPlayer player);
|
|
||||||
|
|
||||||
public void addPlayerToDistanceMaps(final ServerLevel world, final ServerPlayer player);
|
|
||||||
|
|
||||||
public void removePlayerFromDistanceMaps(final ServerLevel world, final ServerPlayer player);
|
|
||||||
|
|
||||||
public void updateMaps(final ServerLevel world, final ServerPlayer player);
|
|
||||||
|
|
||||||
public long[] getCounterTypesUncached(final TicketType type);
|
|
||||||
}
|
|
||||||
@@ -1,9 +1,10 @@
|
|||||||
package ca.spottedleaf.moonrise.common.util;
|
package ca.spottedleaf.moonrise.common.util;
|
||||||
|
|
||||||
import ca.spottedleaf.moonrise.common.PlatformHooks;
|
import ca.spottedleaf.moonrise.common.config.adapter.TypeAdapterRegistry;
|
||||||
|
import ca.spottedleaf.moonrise.common.config.config.YamlConfig;
|
||||||
import ca.spottedleaf.moonrise.common.config.moonrise.MoonriseConfig;
|
import ca.spottedleaf.moonrise.common.config.moonrise.MoonriseConfig;
|
||||||
import ca.spottedleaf.yamlconfig.adapter.TypeAdapterRegistry;
|
import ca.spottedleaf.moonrise.common.config.moonrise.adapter.DefaultedTypeAdapter;
|
||||||
import ca.spottedleaf.yamlconfig.config.YamlConfig;
|
import ca.spottedleaf.moonrise.common.config.moonrise.type.DefaultedValue;
|
||||||
import org.slf4j.Logger;
|
import org.slf4j.Logger;
|
||||||
import org.slf4j.LoggerFactory;
|
import org.slf4j.LoggerFactory;
|
||||||
import java.io.File;
|
import java.io.File;
|
||||||
@@ -12,17 +13,19 @@ public final class ConfigHolder {
|
|||||||
|
|
||||||
private static final Logger LOGGER = LoggerFactory.getLogger(ConfigHolder.class);
|
private static final Logger LOGGER = LoggerFactory.getLogger(ConfigHolder.class);
|
||||||
|
|
||||||
private static final File CONFIG_FILE = new File(System.getProperty(PlatformHooks.get().getBrand() + ".ConfigFile", "config/moonrise.yml"));
|
private static final File CONFIG_FILE = new File(System.getProperty("Moonrise.ConfigFile", "config/moonrise.yml"));
|
||||||
private static final TypeAdapterRegistry CONFIG_ADAPTERS = new TypeAdapterRegistry();
|
private static final TypeAdapterRegistry CONFIG_ADAPTERS = new TypeAdapterRegistry();
|
||||||
private static final YamlConfig<MoonriseConfig> CONFIG;
|
private static final YamlConfig<MoonriseConfig> CONFIG;
|
||||||
static {
|
static {
|
||||||
|
CONFIG_ADAPTERS.putAdapter(DefaultedValue.class, new DefaultedTypeAdapter());
|
||||||
|
|
||||||
try {
|
try {
|
||||||
CONFIG = new YamlConfig<>(MoonriseConfig.class, new MoonriseConfig(), CONFIG_ADAPTERS);
|
CONFIG = new YamlConfig<>(MoonriseConfig.class, new MoonriseConfig(), CONFIG_ADAPTERS);
|
||||||
} catch (final Exception ex) {
|
} catch (final Exception ex) {
|
||||||
throw new RuntimeException(ex);
|
throw new RuntimeException(ex);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
private static final String CONFIG_HEADER = String.format("""
|
private static final String CONFIG_HEADER = """
|
||||||
This is the configuration file for Moonrise.
|
This is the configuration file for Moonrise.
|
||||||
|
|
||||||
Each configuration option is prefixed with a comment to explain what it does. Additional changes to this file
|
Each configuration option is prefixed with a comment to explain what it does. Additional changes to this file
|
||||||
@@ -30,10 +33,10 @@ public final class ConfigHolder {
|
|||||||
|
|
||||||
Below are the Moonrise startup flags. Note that startup flags must be placed in the JVM arguments, not
|
Below are the Moonrise startup flags. Note that startup flags must be placed in the JVM arguments, not
|
||||||
program arguments.
|
program arguments.
|
||||||
-D%1$s.ConfigFile=<file> - Override the config file location. Might be useful for multiple game versions.
|
-DMoonrise.ConfigFile=<file> - Override the config file location. Might be useful for multiple game versions.
|
||||||
-D%1$s.WorkerThreadCount=<number> - Override the auto configured worker thread counts (worker-threads).
|
-DMoonrise.WorkerThreadCount=<number> - Override the auto configured worker thread counts (worker-threads).
|
||||||
-D%1$s.MaxViewDistance=<number> - Overrides the maximum view distance, should only use for debugging purposes.
|
-DMoonrise.MaxViewDistance=<number> - Overrides the maximum view distance, should only use for debugging purposes.
|
||||||
""", PlatformHooks.get().getBrand());
|
""";
|
||||||
|
|
||||||
static {
|
static {
|
||||||
reloadConfig();
|
reloadConfig();
|
||||||
@@ -48,6 +51,14 @@ public final class ConfigHolder {
|
|||||||
}
|
}
|
||||||
|
|
||||||
public static boolean reloadConfig() {
|
public static boolean reloadConfig() {
|
||||||
|
final boolean ret = reloadConfig0();
|
||||||
|
|
||||||
|
CONFIG.callInitialisers();
|
||||||
|
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
|
|
||||||
|
private static boolean reloadConfig0() {
|
||||||
synchronized (CONFIG) {
|
synchronized (CONFIG) {
|
||||||
if (CONFIG_FILE.exists()) {
|
if (CONFIG_FILE.exists()) {
|
||||||
try {
|
try {
|
||||||
@@ -58,8 +69,6 @@ public final class ConfigHolder {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
CONFIG.callInitialisers();
|
|
||||||
|
|
||||||
// write back any changes, or create if needed
|
// write back any changes, or create if needed
|
||||||
return saveConfig();
|
return saveConfig();
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,44 +0,0 @@
|
|||||||
package ca.spottedleaf.moonrise.common.util;
|
|
||||||
|
|
||||||
import net.minecraft.world.entity.Entity;
|
|
||||||
import net.minecraft.world.phys.Vec3;
|
|
||||||
import java.text.DecimalFormat;
|
|
||||||
import java.util.ArrayList;
|
|
||||||
import java.util.List;
|
|
||||||
|
|
||||||
public final class EntityUtil {
|
|
||||||
|
|
||||||
private static final ThreadLocal<DecimalFormat> THREE_DECIMAL_PLACES = ThreadLocal.withInitial(() -> {
|
|
||||||
return new DecimalFormat("#,##0.000");
|
|
||||||
});
|
|
||||||
|
|
||||||
private static String formatVec(final Vec3 vec) {
|
|
||||||
final DecimalFormat format = THREE_DECIMAL_PLACES.get();
|
|
||||||
|
|
||||||
return "(" + format.format(vec.x) + "," + format.format(vec.y) + "," + format.format(vec.z) + ")";
|
|
||||||
}
|
|
||||||
|
|
||||||
private static String dumpEntityWithoutReferences(final Entity entity) {
|
|
||||||
if (entity == null) {
|
|
||||||
return "{null}";
|
|
||||||
}
|
|
||||||
|
|
||||||
return "{type=" + entity.getClass().getSimpleName() + ",id=" + entity.getId() + ",uuid=" + entity.getUUID() + ",pos=" + formatVec(entity.position())
|
|
||||||
+ ",mot=" + formatVec(entity.getDeltaMovement()) + ",aabb=" + entity.getBoundingBox() + ",removed=" + entity.getRemovalReason() + ",has_vehicle=" + (entity.getVehicle() != null)
|
|
||||||
+ ",passenger_count=" + entity.getPassengers().size();
|
|
||||||
}
|
|
||||||
|
|
||||||
public static String dumpEntity(final Entity entity) {
|
|
||||||
final List<Entity> passengers = entity.getPassengers();
|
|
||||||
final List<String> passengerStrings = new ArrayList<>(passengers.size());
|
|
||||||
|
|
||||||
for (final Entity passenger : passengers) {
|
|
||||||
passengerStrings.add("(" + dumpEntityWithoutReferences(passenger) + ")");
|
|
||||||
}
|
|
||||||
|
|
||||||
return "{root=[" + dumpEntityWithoutReferences(entity) + "], vehicle=[" + dumpEntityWithoutReferences(entity.getVehicle())
|
|
||||||
+ "], passengers=[" + String.join(",", passengerStrings) + "]";
|
|
||||||
}
|
|
||||||
|
|
||||||
private EntityUtil() {}
|
|
||||||
}
|
|
||||||
@@ -1,7 +1,6 @@
|
|||||||
package ca.spottedleaf.moonrise.common.util;
|
package ca.spottedleaf.moonrise.common.util;
|
||||||
|
|
||||||
import com.google.gson.JsonElement;
|
import com.google.gson.JsonElement;
|
||||||
import com.google.gson.Strictness;
|
|
||||||
import com.google.gson.internal.Streams;
|
import com.google.gson.internal.Streams;
|
||||||
import com.google.gson.stream.JsonWriter;
|
import com.google.gson.stream.JsonWriter;
|
||||||
import java.io.File;
|
import java.io.File;
|
||||||
@@ -17,7 +16,7 @@ public final class JsonUtil {
|
|||||||
final StringWriter stringWriter = new StringWriter();
|
final StringWriter stringWriter = new StringWriter();
|
||||||
final JsonWriter jsonWriter = new JsonWriter(stringWriter);
|
final JsonWriter jsonWriter = new JsonWriter(stringWriter);
|
||||||
jsonWriter.setIndent(" ");
|
jsonWriter.setIndent(" ");
|
||||||
jsonWriter.setStrictness(Strictness.LENIENT);
|
jsonWriter.setLenient(false);
|
||||||
Streams.write(element, jsonWriter);
|
Streams.write(element, jsonWriter);
|
||||||
|
|
||||||
final String jsonString = stringWriter.toString();
|
final String jsonString = stringWriter.toString();
|
||||||
|
|||||||
@@ -3,12 +3,8 @@ package ca.spottedleaf.moonrise.common.util;
|
|||||||
public final class MixinWorkarounds {
|
public final class MixinWorkarounds {
|
||||||
|
|
||||||
// mixins tries to find the owner of the clone() method, which doesn't exist and NPEs
|
// mixins tries to find the owner of the clone() method, which doesn't exist and NPEs
|
||||||
// https://github.com/FabricMC/Mixin/pull/147
|
|
||||||
public static long[] clone(final long[] values) {
|
public static long[] clone(final long[] values) {
|
||||||
return values.clone();
|
return values.clone();
|
||||||
}
|
}
|
||||||
|
|
||||||
public static byte[] clone(final byte[] values) {
|
|
||||||
return values.clone();
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -19,7 +19,7 @@ public final class MoonriseCommon {
|
|||||||
@Override
|
@Override
|
||||||
public void accept(Thread thread) {
|
public void accept(Thread thread) {
|
||||||
thread.setDaemon(true);
|
thread.setDaemon(true);
|
||||||
thread.setName(PlatformHooks.get().getBrand() + " Common Worker #" + this.idGenerator.getAndIncrement());
|
thread.setName("Moonrise Common Worker #" + this.idGenerator.getAndIncrement());
|
||||||
thread.setUncaughtExceptionHandler(new Thread.UncaughtExceptionHandler() {
|
thread.setUncaughtExceptionHandler(new Thread.UncaughtExceptionHandler() {
|
||||||
@Override
|
@Override
|
||||||
public void uncaughtException(final Thread thread, final Throwable throwable) {
|
public void uncaughtException(final Thread thread, final Throwable throwable) {
|
||||||
@@ -44,7 +44,7 @@ public final class MoonriseCommon {
|
|||||||
} else {
|
} else {
|
||||||
defaultWorkerThreads = defaultWorkerThreads / 2;
|
defaultWorkerThreads = defaultWorkerThreads / 2;
|
||||||
}
|
}
|
||||||
defaultWorkerThreads = Integer.getInteger(PlatformHooks.get().getBrand() + ".WorkerThreadCount", Integer.valueOf(defaultWorkerThreads));
|
defaultWorkerThreads = Integer.getInteger("Moonrise.WorkerThreadCount", Integer.valueOf(defaultWorkerThreads));
|
||||||
|
|
||||||
int workerThreads = configWorkerThreads;
|
int workerThreads = configWorkerThreads;
|
||||||
|
|
||||||
|
|||||||
@@ -1,10 +1,8 @@
|
|||||||
package ca.spottedleaf.moonrise.common.util;
|
package ca.spottedleaf.moonrise.common.util;
|
||||||
|
|
||||||
import ca.spottedleaf.moonrise.common.PlatformHooks;
|
|
||||||
|
|
||||||
public final class MoonriseConstants {
|
public final class MoonriseConstants {
|
||||||
|
|
||||||
public static final int MAX_VIEW_DISTANCE = Integer.getInteger(PlatformHooks.get().getBrand() + ".MaxViewDistance", 32);
|
public static final int MAX_VIEW_DISTANCE = Integer.getInteger("Moonrise.MaxViewDistance", 32);
|
||||||
|
|
||||||
private MoonriseConstants() {}
|
private MoonriseConstants() {}
|
||||||
|
|
||||||
|
|||||||
@@ -0,0 +1,53 @@
|
|||||||
|
package ca.spottedleaf.moonrise.common.util;
|
||||||
|
|
||||||
|
import net.minecraft.world.level.levelgen.LegacyRandomSource;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Avoid costly CAS of superclass
|
||||||
|
*/
|
||||||
|
public final class SimpleRandom extends LegacyRandomSource {
|
||||||
|
|
||||||
|
private static final long MULTIPLIER = 25214903917L;
|
||||||
|
private static final long ADDEND = 11L;
|
||||||
|
private static final int BITS = 48;
|
||||||
|
private static final long MASK = (1L << BITS) - 1;
|
||||||
|
|
||||||
|
private long value;
|
||||||
|
|
||||||
|
public SimpleRandom(final long seed) {
|
||||||
|
super(0L);
|
||||||
|
this.value = seed;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void setSeed(final long seed) {
|
||||||
|
this.value = (seed ^ MULTIPLIER) & MASK;
|
||||||
|
}
|
||||||
|
|
||||||
|
private long advanceSeed() {
|
||||||
|
return this.value = ((this.value * MULTIPLIER) + ADDEND) & MASK;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public int next(final int bits) {
|
||||||
|
return (int)(this.advanceSeed() >>> (BITS - bits));
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public int nextInt() {
|
||||||
|
final long seed = this.advanceSeed();
|
||||||
|
return (int)(seed >>> (BITS - Integer.SIZE));
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public int nextInt(final int bound) {
|
||||||
|
if (bound <= 0) {
|
||||||
|
throw new IllegalArgumentException();
|
||||||
|
}
|
||||||
|
|
||||||
|
// https://lemire.me/blog/2016/06/27/a-fast-alternative-to-the-modulo-reduction/
|
||||||
|
final long value = this.advanceSeed() >>> (BITS - Integer.SIZE);
|
||||||
|
return (int)((value * (long)bound) >>> Integer.SIZE);
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -1,105 +0,0 @@
|
|||||||
package ca.spottedleaf.moonrise.common.util;
|
|
||||||
|
|
||||||
import net.minecraft.util.Mth;
|
|
||||||
import net.minecraft.util.RandomSource;
|
|
||||||
import net.minecraft.world.level.levelgen.BitRandomSource;
|
|
||||||
import net.minecraft.world.level.levelgen.MarsagliaPolarGaussian;
|
|
||||||
import net.minecraft.world.level.levelgen.PositionalRandomFactory;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Avoid costly CAS of superclass + division in nextInt
|
|
||||||
*/
|
|
||||||
public final class SimpleThreadUnsafeRandom implements BitRandomSource {
|
|
||||||
|
|
||||||
private static final long MULTIPLIER = 25214903917L;
|
|
||||||
private static final long ADDEND = 11L;
|
|
||||||
private static final int BITS = 48;
|
|
||||||
private static final long MASK = (1L << BITS) - 1L;
|
|
||||||
|
|
||||||
private long value;
|
|
||||||
private final MarsagliaPolarGaussian gaussianSource = new MarsagliaPolarGaussian(this);
|
|
||||||
|
|
||||||
public SimpleThreadUnsafeRandom(final long seed) {
|
|
||||||
this.setSeed(seed);
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public void setSeed(final long seed) {
|
|
||||||
this.value = (seed ^ MULTIPLIER) & MASK;
|
|
||||||
this.gaussianSource.reset();
|
|
||||||
}
|
|
||||||
|
|
||||||
private long advanceSeed() {
|
|
||||||
return this.value = ((this.value * MULTIPLIER) + ADDEND) & MASK;
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public int next(final int bits) {
|
|
||||||
return (int)(this.advanceSeed() >>> (BITS - bits));
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public int nextInt() {
|
|
||||||
final long seed = this.advanceSeed();
|
|
||||||
return (int)(seed >>> (BITS - Integer.SIZE));
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public int nextInt(final int bound) {
|
|
||||||
if (bound <= 0) {
|
|
||||||
throw new IllegalArgumentException();
|
|
||||||
}
|
|
||||||
|
|
||||||
// https://lemire.me/blog/2016/06/27/a-fast-alternative-to-the-modulo-reduction/
|
|
||||||
final long value = this.advanceSeed() >>> (BITS - Integer.SIZE);
|
|
||||||
return (int)((value * (long)bound) >>> Integer.SIZE);
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public double nextGaussian() {
|
|
||||||
return this.gaussianSource.nextGaussian();
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public RandomSource fork() {
|
|
||||||
return new SimpleThreadUnsafeRandom(this.nextLong());
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public PositionalRandomFactory forkPositional() {
|
|
||||||
return new SimpleRandomPositionalFactory(this.nextLong());
|
|
||||||
}
|
|
||||||
|
|
||||||
public static final class SimpleRandomPositionalFactory implements PositionalRandomFactory {
|
|
||||||
|
|
||||||
private final long seed;
|
|
||||||
|
|
||||||
public SimpleRandomPositionalFactory(final long seed) {
|
|
||||||
this.seed = seed;
|
|
||||||
}
|
|
||||||
|
|
||||||
public long getSeed() {
|
|
||||||
return this.seed;
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public RandomSource fromHashOf(final String string) {
|
|
||||||
return new SimpleThreadUnsafeRandom((long)string.hashCode() ^ this.seed);
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public RandomSource fromSeed(final long seed) {
|
|
||||||
return new SimpleThreadUnsafeRandom(seed);
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public RandomSource at(final int x, final int y, final int z) {
|
|
||||||
return new SimpleThreadUnsafeRandom(Mth.getSeed(x, y, z) ^ this.seed);
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public void parityConfigString(final StringBuilder stringBuilder) {
|
|
||||||
stringBuilder.append("SimpleRandomPositionalFactory{").append(this.seed).append('}');
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,94 +0,0 @@
|
|||||||
package ca.spottedleaf.moonrise.common.util;
|
|
||||||
|
|
||||||
import net.minecraft.util.Mth;
|
|
||||||
import net.minecraft.util.RandomSource;
|
|
||||||
import net.minecraft.world.level.levelgen.BitRandomSource;
|
|
||||||
import net.minecraft.world.level.levelgen.MarsagliaPolarGaussian;
|
|
||||||
import net.minecraft.world.level.levelgen.PositionalRandomFactory;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Avoid costly CAS of superclass
|
|
||||||
*/
|
|
||||||
public final class ThreadUnsafeRandom implements BitRandomSource {
|
|
||||||
|
|
||||||
private static final long MULTIPLIER = 25214903917L;
|
|
||||||
private static final long ADDEND = 11L;
|
|
||||||
private static final int BITS = 48;
|
|
||||||
private static final long MASK = (1L << BITS) - 1L;
|
|
||||||
|
|
||||||
private long value;
|
|
||||||
private final MarsagliaPolarGaussian gaussianSource = new MarsagliaPolarGaussian(this);
|
|
||||||
|
|
||||||
public ThreadUnsafeRandom(final long seed) {
|
|
||||||
this.setSeed(seed);
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public void setSeed(final long seed) {
|
|
||||||
this.value = (seed ^ MULTIPLIER) & MASK;
|
|
||||||
this.gaussianSource.reset();
|
|
||||||
}
|
|
||||||
|
|
||||||
private long advanceSeed() {
|
|
||||||
return this.value = ((this.value * MULTIPLIER) + ADDEND) & MASK;
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public int next(final int bits) {
|
|
||||||
return (int)(this.advanceSeed() >>> (BITS - bits));
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public int nextInt() {
|
|
||||||
final long seed = this.advanceSeed();
|
|
||||||
return (int)(seed >>> (BITS - Integer.SIZE));
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public double nextGaussian() {
|
|
||||||
return this.gaussianSource.nextGaussian();
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public RandomSource fork() {
|
|
||||||
return new ThreadUnsafeRandom(this.nextLong());
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public PositionalRandomFactory forkPositional() {
|
|
||||||
return new ThreadUnsafeRandomPositionalFactory(this.nextLong());
|
|
||||||
}
|
|
||||||
|
|
||||||
public static final class ThreadUnsafeRandomPositionalFactory implements PositionalRandomFactory {
|
|
||||||
|
|
||||||
private final long seed;
|
|
||||||
|
|
||||||
public ThreadUnsafeRandomPositionalFactory(final long seed) {
|
|
||||||
this.seed = seed;
|
|
||||||
}
|
|
||||||
|
|
||||||
public long getSeed() {
|
|
||||||
return this.seed;
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public RandomSource fromHashOf(final String string) {
|
|
||||||
return new ThreadUnsafeRandom((long)string.hashCode() ^ this.seed);
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public RandomSource fromSeed(final long seed) {
|
|
||||||
return new ThreadUnsafeRandom(seed);
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public RandomSource at(final int x, final int y, final int z) {
|
|
||||||
return new ThreadUnsafeRandom(Mth.getSeed(x, y, z) ^ this.seed);
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public void parityConfigString(final StringBuilder stringBuilder) {
|
|
||||||
stringBuilder.append("ThreadUnsafeRandomPositionalFactory{").append(this.seed).append('}');
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -15,81 +15,56 @@ public class TickThread extends Thread {
|
|||||||
|
|
||||||
private static final Logger LOGGER = LoggerFactory.getLogger(TickThread.class);
|
private static final Logger LOGGER = LoggerFactory.getLogger(TickThread.class);
|
||||||
|
|
||||||
private static String getThreadContext() {
|
|
||||||
return "thread=" + Thread.currentThread().getName();
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @deprecated
|
* @deprecated
|
||||||
*/
|
*/
|
||||||
@Deprecated
|
@Deprecated
|
||||||
public static void ensureTickThread(final String reason) {
|
public static void ensureTickThread(final String reason) {
|
||||||
if (!isTickThread()) {
|
if (!isTickThread()) {
|
||||||
LOGGER.error("Thread failed main thread check: " + reason + ", context=" + getThreadContext(), new Throwable());
|
LOGGER.error("Thread " + Thread.currentThread().getName() + " failed main thread check: " + reason, new Throwable());
|
||||||
throw new IllegalStateException(reason);
|
throw new IllegalStateException(reason);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public static void ensureTickThread(final Level world, final BlockPos pos, final String reason) {
|
public static void ensureTickThread(final Level world, final BlockPos pos, final String reason) {
|
||||||
if (!isTickThreadFor(world, pos)) {
|
if (!isTickThreadFor(world, pos)) {
|
||||||
final String ex = "Thread failed main thread check: " +
|
LOGGER.error("Thread " + Thread.currentThread().getName() + " failed main thread check: " + reason, new Throwable());
|
||||||
reason + ", context=" + getThreadContext() + ", world=" + WorldUtil.getWorldName(world) + ", block_pos=" + pos;
|
throw new IllegalStateException(reason);
|
||||||
LOGGER.error(ex, new Throwable());
|
|
||||||
throw new IllegalStateException(ex);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public static void ensureTickThread(final Level world, final BlockPos pos, final int blockRadius, final String reason) {
|
|
||||||
if (!isTickThreadFor(world, pos, blockRadius)) {
|
|
||||||
final String ex = "Thread failed main thread check: " +
|
|
||||||
reason + ", context=" + getThreadContext() + ", world=" + WorldUtil.getWorldName(world) + ", block_pos=" + pos + ", block_radius=" + blockRadius;
|
|
||||||
LOGGER.error(ex, new Throwable());
|
|
||||||
throw new IllegalStateException(ex);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public static void ensureTickThread(final Level world, final ChunkPos pos, final String reason) {
|
public static void ensureTickThread(final Level world, final ChunkPos pos, final String reason) {
|
||||||
if (!isTickThreadFor(world, pos)) {
|
if (!isTickThreadFor(world, pos)) {
|
||||||
final String ex = "Thread failed main thread check: " +
|
LOGGER.error("Thread " + Thread.currentThread().getName() + " failed main thread check: " + reason, new Throwable());
|
||||||
reason + ", context=" + getThreadContext() + ", world=" + WorldUtil.getWorldName(world) + ", chunk_pos=" + pos;
|
throw new IllegalStateException(reason);
|
||||||
LOGGER.error(ex, new Throwable());
|
|
||||||
throw new IllegalStateException(ex);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public static void ensureTickThread(final Level world, final int chunkX, final int chunkZ, final String reason) {
|
public static void ensureTickThread(final Level world, final int chunkX, final int chunkZ, final String reason) {
|
||||||
if (!isTickThreadFor(world, chunkX, chunkZ)) {
|
if (!isTickThreadFor(world, chunkX, chunkZ)) {
|
||||||
final String ex = "Thread failed main thread check: " +
|
LOGGER.error("Thread " + Thread.currentThread().getName() + " failed main thread check: " + reason, new Throwable());
|
||||||
reason + ", context=" + getThreadContext() + ", world=" + WorldUtil.getWorldName(world) + ", chunk_pos=" + new ChunkPos(chunkX, chunkZ);
|
throw new IllegalStateException(reason);
|
||||||
LOGGER.error(ex, new Throwable());
|
|
||||||
throw new IllegalStateException(ex);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public static void ensureTickThread(final Entity entity, final String reason) {
|
public static void ensureTickThread(final Entity entity, final String reason) {
|
||||||
if (!isTickThreadFor(entity)) {
|
if (!isTickThreadFor(entity)) {
|
||||||
final String ex = "Thread failed main thread check: " +
|
LOGGER.error("Thread " + Thread.currentThread().getName() + " failed main thread check: " + reason, new Throwable());
|
||||||
reason + ", context=" + getThreadContext() + ", entity=" + EntityUtil.dumpEntity(entity);
|
throw new IllegalStateException(reason);
|
||||||
LOGGER.error(ex, new Throwable());
|
|
||||||
throw new IllegalStateException(ex);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public static void ensureTickThread(final Level world, final AABB aabb, final String reason) {
|
public static void ensureTickThread(final Level world, final AABB aabb, final String reason) {
|
||||||
if (!isTickThreadFor(world, aabb)) {
|
if (!isTickThreadFor(world, aabb)) {
|
||||||
final String ex = "Thread failed main thread check: " +
|
LOGGER.error("Thread " + Thread.currentThread().getName() + " failed main thread check: " + reason, new Throwable());
|
||||||
reason + ", context=" + getThreadContext() + ", world=" + WorldUtil.getWorldName(world) + ", aabb=" + aabb;
|
throw new IllegalStateException(reason);
|
||||||
LOGGER.error(ex, new Throwable());
|
|
||||||
throw new IllegalStateException(ex);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public static void ensureTickThread(final Level world, final double blockX, final double blockZ, final String reason) {
|
public static void ensureTickThread(final Level world, final double blockX, final double blockZ, final String reason) {
|
||||||
if (!isTickThreadFor(world, blockX, blockZ)) {
|
if (!isTickThreadFor(world, blockX, blockZ)) {
|
||||||
final String ex = "Thread failed main thread check: " +
|
LOGGER.error("Thread " + Thread.currentThread().getName() + " failed main thread check: " + reason, new Throwable());
|
||||||
reason + ", context=" + getThreadContext() + ", world=" + WorldUtil.getWorldName(world) + ", block_pos=" + new Vec3(blockX, 0.0, blockZ);
|
throw new IllegalStateException(reason);
|
||||||
LOGGER.error(ex, new Throwable());
|
|
||||||
throw new IllegalStateException(ex);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -130,10 +105,6 @@ public class TickThread extends Thread {
|
|||||||
return isTickThread();
|
return isTickThread();
|
||||||
}
|
}
|
||||||
|
|
||||||
public static boolean isTickThreadFor(final Level world, final BlockPos pos, final int blockRadius) {
|
|
||||||
return isTickThread();
|
|
||||||
}
|
|
||||||
|
|
||||||
public static boolean isTickThreadFor(final Level world, final ChunkPos pos) {
|
public static boolean isTickThreadFor(final Level world, final ChunkPos pos) {
|
||||||
return isTickThread();
|
return isTickThread();
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -8,19 +8,19 @@ public final class WorldUtil {
|
|||||||
// min, max are inclusive
|
// min, max are inclusive
|
||||||
|
|
||||||
public static int getMaxSection(final LevelHeightAccessor world) {
|
public static int getMaxSection(final LevelHeightAccessor world) {
|
||||||
return world.getMaxSectionY();
|
return world.getMaxSection() - 1; // getMaxSection() is exclusive
|
||||||
}
|
}
|
||||||
|
|
||||||
public static int getMaxSection(final Level world) {
|
public static int getMaxSection(final Level world) {
|
||||||
return world.getMaxSectionY();
|
return world.getMaxSection() - 1; // getMaxSection() is exclusive
|
||||||
}
|
}
|
||||||
|
|
||||||
public static int getMinSection(final LevelHeightAccessor world) {
|
public static int getMinSection(final LevelHeightAccessor world) {
|
||||||
return world.getMinSectionY();
|
return world.getMinSection();
|
||||||
}
|
}
|
||||||
|
|
||||||
public static int getMinSection(final Level world) {
|
public static int getMinSection(final Level world) {
|
||||||
return world.getMinSectionY();
|
return world.getMinSection();
|
||||||
}
|
}
|
||||||
|
|
||||||
public static int getMaxLightSection(final LevelHeightAccessor world) {
|
public static int getMaxLightSection(final LevelHeightAccessor world) {
|
||||||
|
|||||||
@@ -154,9 +154,9 @@ abstract class LevelChunkSectionMixin implements BlockCountingChunkSection {
|
|||||||
|
|
||||||
if (this.maybeHas((final BlockState state) -> !state.isAir())) {
|
if (this.maybeHas((final BlockState state) -> !state.isAir())) {
|
||||||
final PalettedContainer.Data<BlockState> data = this.states.data;
|
final PalettedContainer.Data<BlockState> data = this.states.data;
|
||||||
final Palette<BlockState> palette = data.palette();
|
final Palette<BlockState> palette = data.palette;
|
||||||
final int paletteSize = palette.getSize();
|
final int paletteSize = palette.getSize();
|
||||||
final BitStorage storage = data.storage();
|
final BitStorage storage = data.storage;
|
||||||
|
|
||||||
final Int2ObjectOpenHashMap<ShortArrayList> counts;
|
final Int2ObjectOpenHashMap<ShortArrayList> counts;
|
||||||
if (paletteSize == 1) {
|
if (paletteSize == 1) {
|
||||||
|
|||||||
@@ -87,7 +87,7 @@ abstract class LevelMixin implements LevelAccessor, AutoCloseable {
|
|||||||
|
|
||||||
if (doTick && this.shouldTickBlocksAt(tileEntity.getPos())) {
|
if (doTick && this.shouldTickBlocksAt(tileEntity.getPos())) {
|
||||||
tileEntity.tick();
|
tileEntity.tick();
|
||||||
// call mid-tick tasks for chunk system
|
// call mid tick tasks for chunk system
|
||||||
if ((++tickedEntities & 7) == 0) {
|
if ((++tickedEntities & 7) == 0) {
|
||||||
((ChunkSystemLevel)(Level)(Object)this).moonrise$midTickTasks();
|
((ChunkSystemLevel)(Level)(Object)this).moonrise$midTickTasks();
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -6,7 +6,6 @@ import com.llamalad7.mixinextras.injector.wrapoperation.WrapOperation;
|
|||||||
import net.minecraft.world.level.block.state.properties.BooleanProperty;
|
import net.minecraft.world.level.block.state.properties.BooleanProperty;
|
||||||
import net.minecraft.world.level.block.state.properties.Property;
|
import net.minecraft.world.level.block.state.properties.Property;
|
||||||
import org.spongepowered.asm.mixin.Mixin;
|
import org.spongepowered.asm.mixin.Mixin;
|
||||||
import org.spongepowered.asm.mixin.Unique;
|
|
||||||
import org.spongepowered.asm.mixin.injection.At;
|
import org.spongepowered.asm.mixin.injection.At;
|
||||||
import org.spongepowered.asm.mixin.injection.Constant;
|
import org.spongepowered.asm.mixin.injection.Constant;
|
||||||
import org.spongepowered.asm.mixin.injection.Inject;
|
import org.spongepowered.asm.mixin.injection.Inject;
|
||||||
@@ -18,9 +17,6 @@ abstract class BooleanPropertyMixin extends Property<Boolean> implements Propert
|
|||||||
super(string, class_);
|
super(string, class_);
|
||||||
}
|
}
|
||||||
|
|
||||||
@Unique
|
|
||||||
private static final Boolean[] BY_ID = new Boolean[]{ Boolean.FALSE, Boolean.TRUE };
|
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public final int moonrise$getIdFor(final Boolean value) {
|
public final int moonrise$getIdFor(final Boolean value) {
|
||||||
return value.booleanValue() ? 1 : 0;
|
return value.booleanValue() ? 1 : 0;
|
||||||
@@ -37,6 +33,23 @@ abstract class BooleanPropertyMixin extends Property<Boolean> implements Propert
|
|||||||
)
|
)
|
||||||
)
|
)
|
||||||
private void init(final CallbackInfo ci) {
|
private void init(final CallbackInfo ci) {
|
||||||
this.moonrise$setById(BY_ID);
|
this.moonrise$setById(new Boolean[]{ Boolean.FALSE, Boolean.TRUE });
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* This skips all ops after the identity comparison in the original code.
|
||||||
|
*
|
||||||
|
* @reason Properties are identity comparable
|
||||||
|
* @author Spottedleaf
|
||||||
|
*/
|
||||||
|
@WrapOperation(
|
||||||
|
method = "equals",
|
||||||
|
constant = @Constant(
|
||||||
|
classValue = BooleanProperty.class,
|
||||||
|
ordinal = 0
|
||||||
|
)
|
||||||
|
)
|
||||||
|
private boolean skipFurtherComparison(final Object obj, final Operation<Boolean> orig) {
|
||||||
|
return false;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -18,6 +18,9 @@ import java.util.Collection;
|
|||||||
@Mixin(EnumProperty.class)
|
@Mixin(EnumProperty.class)
|
||||||
abstract class EnumPropertyMixin<T extends Enum<T> & StringRepresentable> extends Property<T> implements PropertyAccess<T> {
|
abstract class EnumPropertyMixin<T extends Enum<T> & StringRepresentable> extends Property<T> implements PropertyAccess<T> {
|
||||||
|
|
||||||
|
@Shadow
|
||||||
|
public abstract Collection<T> getPossibleValues();
|
||||||
|
|
||||||
@Unique
|
@Unique
|
||||||
private int[] idLookupTable;
|
private int[] idLookupTable;
|
||||||
|
|
||||||
|
|||||||
@@ -83,6 +83,7 @@ abstract class StateHolderMixin<O, S> implements PropertyAccessStateHolder {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// remove values arrays
|
// remove values arrays
|
||||||
|
this.values = null;
|
||||||
for (final Map.Entry<Map<Property<?>, Comparable<?>>, S> entry : map.entrySet()) {
|
for (final Map.Entry<Map<Property<?>, Comparable<?>>, S> entry : map.entrySet()) {
|
||||||
final S value = entry.getValue();
|
final S value = entry.getValue();
|
||||||
((StateHolderMixin<O, S>)(Object)(StateHolder<O, S>)value).values = null;
|
((StateHolderMixin<O, S>)(Object)(StateHolder<O, S>)value).values = null;
|
||||||
@@ -125,8 +126,8 @@ abstract class StateHolderMixin<O, S> implements PropertyAccessStateHolder {
|
|||||||
* @author Spottedleaf
|
* @author Spottedleaf
|
||||||
*/
|
*/
|
||||||
@Overwrite
|
@Overwrite
|
||||||
public <T extends Comparable<T>> T getNullableValue(final Property<T> property) {
|
public <T extends Comparable<T>> Optional<T> getOptionalValue(final Property<T> property) {
|
||||||
return property == null ? null : this.optimisedTable.get(this.tableIndex, property);
|
return property == null ? Optional.empty() : Optional.ofNullable(this.optimisedTable.get(this.tableIndex, property));
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -166,7 +167,7 @@ abstract class StateHolderMixin<O, S> implements PropertyAccessStateHolder {
|
|||||||
*/
|
*/
|
||||||
@Overwrite
|
@Overwrite
|
||||||
public Map<Property<?>, Comparable<?>> getValues() {
|
public Map<Property<?>, Comparable<?>> getValues() {
|
||||||
final ZeroCollidingReferenceStateTable<O, S> table = this.optimisedTable;
|
ZeroCollidingReferenceStateTable<O, S> table = this.optimisedTable;
|
||||||
// We have to use this.values until the table is loaded
|
// We have to use this.values until the table is loaded
|
||||||
return table.isLoaded() ? table.getMapView(this.tableIndex) : this.values;
|
return table.isLoaded() ? table.getMapView(this.tableIndex) : this.values;
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -66,6 +66,9 @@ abstract class ChunkHolderMixin extends GenerationChunkHolder implements ChunkSy
|
|||||||
@Unique
|
@Unique
|
||||||
private final ReferenceList<ServerPlayer> playersSentChunkTo = new ReferenceList<>(EMPTY_PLAYER_ARRAY);
|
private final ReferenceList<ServerPlayer> playersSentChunkTo = new ReferenceList<>(EMPTY_PLAYER_ARRAY);
|
||||||
|
|
||||||
|
@Unique
|
||||||
|
private boolean isMarkedDirtyForPlayers;
|
||||||
|
|
||||||
@Unique
|
@Unique
|
||||||
private ChunkMap getChunkMap() {
|
private ChunkMap getChunkMap() {
|
||||||
return (ChunkMap)this.playerProvider;
|
return (ChunkMap)this.playerProvider;
|
||||||
@@ -120,6 +123,16 @@ abstract class ChunkHolderMixin extends GenerationChunkHolder implements ChunkSy
|
|||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public final boolean moonrise$isMarkedDirtyForPlayers() {
|
||||||
|
return this.isMarkedDirtyForPlayers;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public final void moonrise$markDirtyForPlayers(final boolean value) {
|
||||||
|
this.isMarkedDirtyForPlayers = value;
|
||||||
|
}
|
||||||
|
|
||||||
@Unique
|
@Unique
|
||||||
private static final ServerPlayer[] EMPTY_PLAYER_ARRAY = new ServerPlayer[0];
|
private static final ServerPlayer[] EMPTY_PLAYER_ARRAY = new ServerPlayer[0];
|
||||||
|
|
||||||
@@ -287,7 +300,14 @@ abstract class ChunkHolderMixin extends GenerationChunkHolder implements ChunkSy
|
|||||||
// no players to sent to, so don't need to update anything
|
// no players to sent to, so don't need to update anything
|
||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
return this.getChunkToSend();
|
final LevelChunk ret = this.getChunkToSend();
|
||||||
|
|
||||||
|
if (ret != null) {
|
||||||
|
((ChunkSystemServerLevel)this.getChunkMap().level).moonrise$addUnsyncedChunk((ChunkHolder)(Object)this);
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
|
|
||||||
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -306,7 +326,14 @@ abstract class ChunkHolderMixin extends GenerationChunkHolder implements ChunkSy
|
|||||||
// no players to sent to, so don't need to update anything
|
// no players to sent to, so don't need to update anything
|
||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
return this.getChunkToSend();
|
final LevelChunk ret = this.getChunkToSend();
|
||||||
|
|
||||||
|
if (ret != null) {
|
||||||
|
((ChunkSystemServerLevel)this.getChunkMap().level).moonrise$addUnsyncedChunk((ChunkHolder)(Object)this);
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
|
|
||||||
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -366,7 +393,7 @@ abstract class ChunkHolderMixin extends GenerationChunkHolder implements ChunkSy
|
|||||||
/**
|
/**
|
||||||
* @reason Use ticket system to control ticket levels
|
* @reason Use ticket system to control ticket levels
|
||||||
* @author Spottedleaf
|
* @author Spottedleaf
|
||||||
* @see net.minecraft.server.level.ServerChunkCache#addTicketWithRadius(TicketType, ChunkPos, int)
|
* @see net.minecraft.server.level.ServerChunkCache#addRegionTicket(TicketType, ChunkPos, int, Object)
|
||||||
*/
|
*/
|
||||||
@Overwrite
|
@Overwrite
|
||||||
public void setTicketLevel(int i) {
|
public void setTicketLevel(int i) {
|
||||||
@@ -397,14 +424,8 @@ abstract class ChunkHolderMixin extends GenerationChunkHolder implements ChunkSy
|
|||||||
* @reason Chunk system hooks for ticket level updating now in {@link NewChunkHolder#processTicketLevelUpdate(List, List)}
|
* @reason Chunk system hooks for ticket level updating now in {@link NewChunkHolder#processTicketLevelUpdate(List, List)}
|
||||||
* @author Spottedleaf
|
* @author Spottedleaf
|
||||||
*/
|
*/
|
||||||
// inject to avoid conflicting with fabric API's mixin here, we call their event in FabricHooks
|
@Overwrite
|
||||||
@Inject(
|
public void updateFutures(final ChunkMap chunkMap, final Executor executor) {
|
||||||
method = "updateFutures",
|
|
||||||
at = @At(
|
|
||||||
value = "HEAD"
|
|
||||||
)
|
|
||||||
)
|
|
||||||
public void clobberUpdateFutures(final ChunkMap chunkMap, final Executor executor, final CallbackInfo ci) {
|
|
||||||
throw new UnsupportedOperationException();
|
throw new UnsupportedOperationException();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -2,7 +2,6 @@ package ca.spottedleaf.moonrise.mixin.chunk_system;
|
|||||||
|
|
||||||
import ca.spottedleaf.moonrise.patches.chunk_system.level.chunk.ChunkSystemDistanceManager;
|
import ca.spottedleaf.moonrise.patches.chunk_system.level.chunk.ChunkSystemDistanceManager;
|
||||||
import net.minecraft.server.level.ChunkMap;
|
import net.minecraft.server.level.ChunkMap;
|
||||||
import net.minecraft.world.level.TicketStorage;
|
|
||||||
import org.spongepowered.asm.mixin.Final;
|
import org.spongepowered.asm.mixin.Final;
|
||||||
import org.spongepowered.asm.mixin.Mixin;
|
import org.spongepowered.asm.mixin.Mixin;
|
||||||
import org.spongepowered.asm.mixin.Overwrite;
|
import org.spongepowered.asm.mixin.Overwrite;
|
||||||
@@ -16,8 +15,8 @@ abstract class ChunkMap$DistanceManagerMixin extends net.minecraft.server.level.
|
|||||||
@Final
|
@Final
|
||||||
ChunkMap field_17443;
|
ChunkMap field_17443;
|
||||||
|
|
||||||
protected ChunkMap$DistanceManagerMixin(final TicketStorage p_394060_, final Executor p_140774_, final Executor p_140775_) {
|
protected ChunkMap$DistanceManagerMixin(Executor executor, Executor executor2) {
|
||||||
super(p_394060_, p_140774_, p_140775_);
|
super(executor, executor2);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
package ca.spottedleaf.moonrise.mixin.chunk_system;
|
package ca.spottedleaf.moonrise.mixin.chunk_system;
|
||||||
|
|
||||||
import ca.spottedleaf.moonrise.common.PlatformHooks;
|
|
||||||
import ca.spottedleaf.moonrise.common.util.MoonriseConstants;
|
import ca.spottedleaf.moonrise.common.util.MoonriseConstants;
|
||||||
|
import ca.spottedleaf.moonrise.common.util.ChunkSystem;
|
||||||
import ca.spottedleaf.moonrise.patches.chunk_system.io.MoonriseRegionFileIO;
|
import ca.spottedleaf.moonrise.patches.chunk_system.io.MoonriseRegionFileIO;
|
||||||
import ca.spottedleaf.moonrise.patches.chunk_system.level.ChunkSystemChunkMap;
|
import ca.spottedleaf.moonrise.patches.chunk_system.level.ChunkSystemChunkMap;
|
||||||
import ca.spottedleaf.moonrise.patches.chunk_system.level.ChunkSystemServerLevel;
|
import ca.spottedleaf.moonrise.patches.chunk_system.level.ChunkSystemServerLevel;
|
||||||
@@ -11,17 +11,14 @@ import ca.spottedleaf.moonrise.patches.chunk_system.scheduling.NewChunkHolder;
|
|||||||
import com.mojang.datafixers.DataFixer;
|
import com.mojang.datafixers.DataFixer;
|
||||||
import it.unimi.dsi.fastutil.longs.Long2ObjectLinkedOpenHashMap;
|
import it.unimi.dsi.fastutil.longs.Long2ObjectLinkedOpenHashMap;
|
||||||
import java.util.concurrent.Executor;
|
import java.util.concurrent.Executor;
|
||||||
import java.util.concurrent.atomic.AtomicInteger;
|
|
||||||
import java.util.function.IntConsumer;
|
|
||||||
import java.util.function.Supplier;
|
import java.util.function.Supplier;
|
||||||
import it.unimi.dsi.fastutil.longs.LongSet;
|
|
||||||
import net.minecraft.nbt.CompoundTag;
|
import net.minecraft.nbt.CompoundTag;
|
||||||
import net.minecraft.nbt.StreamTagVisitor;
|
import net.minecraft.nbt.StreamTagVisitor;
|
||||||
import net.minecraft.server.level.ChunkGenerationTask;
|
import net.minecraft.server.level.ChunkGenerationTask;
|
||||||
import net.minecraft.server.level.ChunkHolder;
|
import net.minecraft.server.level.ChunkHolder;
|
||||||
import net.minecraft.server.level.ChunkMap;
|
import net.minecraft.server.level.ChunkMap;
|
||||||
import net.minecraft.server.level.ChunkResult;
|
import net.minecraft.server.level.ChunkResult;
|
||||||
import net.minecraft.server.level.ChunkTaskDispatcher;
|
import net.minecraft.server.level.ChunkTaskPriorityQueueSorter;
|
||||||
import net.minecraft.server.level.ChunkTrackingView;
|
import net.minecraft.server.level.ChunkTrackingView;
|
||||||
import net.minecraft.server.level.GeneratingChunkMap;
|
import net.minecraft.server.level.GeneratingChunkMap;
|
||||||
import net.minecraft.server.level.GenerationChunkHolder;
|
import net.minecraft.server.level.GenerationChunkHolder;
|
||||||
@@ -31,8 +28,8 @@ import net.minecraft.server.level.progress.ChunkProgressListener;
|
|||||||
import net.minecraft.util.Mth;
|
import net.minecraft.util.Mth;
|
||||||
import net.minecraft.util.StaticCache2D;
|
import net.minecraft.util.StaticCache2D;
|
||||||
import net.minecraft.util.thread.BlockableEventLoop;
|
import net.minecraft.util.thread.BlockableEventLoop;
|
||||||
|
import net.minecraft.util.thread.ProcessorHandle;
|
||||||
import net.minecraft.world.level.ChunkPos;
|
import net.minecraft.world.level.ChunkPos;
|
||||||
import net.minecraft.world.level.TicketStorage;
|
|
||||||
import net.minecraft.world.level.chunk.ChunkAccess;
|
import net.minecraft.world.level.chunk.ChunkAccess;
|
||||||
import net.minecraft.world.level.chunk.ChunkGenerator;
|
import net.minecraft.world.level.chunk.ChunkGenerator;
|
||||||
import net.minecraft.world.level.chunk.LevelChunk;
|
import net.minecraft.world.level.chunk.LevelChunk;
|
||||||
@@ -44,6 +41,7 @@ import net.minecraft.world.level.chunk.storage.IOWorker;
|
|||||||
import net.minecraft.world.level.chunk.storage.RegionStorageInfo;
|
import net.minecraft.world.level.chunk.storage.RegionStorageInfo;
|
||||||
import net.minecraft.world.level.entity.ChunkStatusUpdateListener;
|
import net.minecraft.world.level.entity.ChunkStatusUpdateListener;
|
||||||
import net.minecraft.world.level.levelgen.structure.templatesystem.StructureTemplateManager;
|
import net.minecraft.world.level.levelgen.structure.templatesystem.StructureTemplateManager;
|
||||||
|
import net.minecraft.world.level.storage.DimensionDataStorage;
|
||||||
import net.minecraft.world.level.storage.LevelStorageSource;
|
import net.minecraft.world.level.storage.LevelStorageSource;
|
||||||
import org.jetbrains.annotations.Nullable;
|
import org.jetbrains.annotations.Nullable;
|
||||||
import org.spongepowered.asm.mixin.Final;
|
import org.spongepowered.asm.mixin.Final;
|
||||||
@@ -80,10 +78,13 @@ abstract class ChunkMapMixin extends ChunkStorage implements ChunkSystemChunkMap
|
|||||||
private volatile Long2ObjectLinkedOpenHashMap<ChunkHolder> visibleChunkMap;
|
private volatile Long2ObjectLinkedOpenHashMap<ChunkHolder> visibleChunkMap;
|
||||||
|
|
||||||
@Shadow
|
@Shadow
|
||||||
private ChunkTaskDispatcher worldgenTaskDispatcher;
|
private ChunkTaskPriorityQueueSorter queueSorter;
|
||||||
|
|
||||||
@Shadow
|
@Shadow
|
||||||
private ChunkTaskDispatcher lightTaskDispatcher;
|
private ProcessorHandle<ChunkTaskPriorityQueueSorter.Message<Runnable>> worldgenMailbox;
|
||||||
|
|
||||||
|
@Shadow
|
||||||
|
private ProcessorHandle<ChunkTaskPriorityQueueSorter.Message<Runnable>> mainThreadMailbox;
|
||||||
|
|
||||||
@Shadow
|
@Shadow
|
||||||
private int serverViewDistance;
|
private int serverViewDistance;
|
||||||
@@ -97,12 +98,6 @@ abstract class ChunkMapMixin extends ChunkStorage implements ChunkSystemChunkMap
|
|||||||
@Shadow
|
@Shadow
|
||||||
private Queue<Runnable> unloadQueue;
|
private Queue<Runnable> unloadQueue;
|
||||||
|
|
||||||
@Shadow
|
|
||||||
private LongSet chunksToEagerlySave;
|
|
||||||
|
|
||||||
@Shadow
|
|
||||||
private AtomicInteger activeChunkWrites;
|
|
||||||
|
|
||||||
public ChunkMapMixin(RegionStorageInfo regionStorageInfo, Path path, DataFixer dataFixer, boolean bl) {
|
public ChunkMapMixin(RegionStorageInfo regionStorageInfo, Path path, DataFixer dataFixer, boolean bl) {
|
||||||
super(regionStorageInfo, path, dataFixer, bl);
|
super(regionStorageInfo, path, dataFixer, bl);
|
||||||
}
|
}
|
||||||
@@ -124,27 +119,25 @@ abstract class ChunkMapMixin extends ChunkStorage implements ChunkSystemChunkMap
|
|||||||
)
|
)
|
||||||
)
|
)
|
||||||
private void constructor(
|
private void constructor(
|
||||||
ServerLevel p_214836_, LevelStorageSource.LevelStorageAccess p_214837_, DataFixer p_214838_,
|
ServerLevel arg, LevelStorageSource.LevelStorageAccess arg2, DataFixer dataFixer,
|
||||||
StructureTemplateManager p_214839_, Executor p_214840_, BlockableEventLoop p_214841_,
|
StructureTemplateManager arg3, Executor executor, BlockableEventLoop<Runnable> arg4,
|
||||||
LightChunkGetter p_214842_, ChunkGenerator p_214843_, ChunkProgressListener p_214844_,
|
LightChunkGetter arg5, ChunkGenerator arg6, ChunkProgressListener arg7,
|
||||||
ChunkStatusUpdateListener p_214845_, Supplier p_214846_, TicketStorage p_394462_, int p_214847_,
|
ChunkStatusUpdateListener arg8, Supplier<DimensionDataStorage> supplier, int j, boolean bl,
|
||||||
boolean p_214848_, CallbackInfo ci) {
|
final CallbackInfo ci) {
|
||||||
// intentionally destroy old chunk system hooks
|
// intentionally destroy old chunk system hooks
|
||||||
this.updatingChunkMap = null;
|
this.updatingChunkMap = null;
|
||||||
this.visibleChunkMap = null;
|
this.visibleChunkMap = null;
|
||||||
this.pendingUnloads = null;
|
this.pendingUnloads = null;
|
||||||
this.worldgenTaskDispatcher = null;
|
this.queueSorter = null;
|
||||||
this.lightTaskDispatcher = null;
|
this.worldgenMailbox = null;
|
||||||
|
this.mainThreadMailbox = null;
|
||||||
this.pendingGenerationTasks = null;
|
this.pendingGenerationTasks = null;
|
||||||
this.unloadQueue = null;
|
this.unloadQueue = null;
|
||||||
this.chunksToEagerlySave = null;
|
|
||||||
this.activeChunkWrites = null;
|
|
||||||
|
|
||||||
// Dummy impl for mods that try to loadAsync directly
|
// Dummy impl for mods that try to loadAsync directly
|
||||||
this.worker = new IOWorker(
|
this.worker = new IOWorker(
|
||||||
// copied from super call
|
// copied from super call
|
||||||
new RegionStorageInfo(p_214837_.getLevelId(), p_214836_.dimension(), "chunk"),
|
new RegionStorageInfo(arg2.getLevelId(), arg.dimension(), "chunk"), arg2.getDimensionPath(arg.dimension()).resolve("region"), bl
|
||||||
p_214837_.getDimensionPath(p_214836_.dimension()).resolve("region"), p_214848_
|
|
||||||
) {
|
) {
|
||||||
@Override
|
@Override
|
||||||
public boolean isOldChunkAround(final ChunkPos chunkPos, final int i) {
|
public boolean isOldChunkAround(final ChunkPos chunkPos, final int i) {
|
||||||
@@ -159,7 +152,7 @@ abstract class ChunkMapMixin extends ChunkStorage implements ChunkSystemChunkMap
|
|||||||
@Override
|
@Override
|
||||||
public CompletableFuture<Optional<CompoundTag>> loadAsync(final ChunkPos chunkPos) {
|
public CompletableFuture<Optional<CompoundTag>> loadAsync(final ChunkPos chunkPos) {
|
||||||
final CompletableFuture<Optional<CompoundTag>> future = new CompletableFuture<>();
|
final CompletableFuture<Optional<CompoundTag>> future = new CompletableFuture<>();
|
||||||
MoonriseRegionFileIO.loadDataAsync(ChunkMapMixin.this.level, chunkPos.x, chunkPos.z, MoonriseRegionFileIO.RegionFileType.CHUNK_DATA, (final CompoundTag tag, final Throwable throwable) -> {
|
MoonriseRegionFileIO.loadDataAsync(ChunkMapMixin.this.level, chunkPos.x, chunkPos.z, MoonriseRegionFileIO.RegionFileType.CHUNK_DATA, (tag, throwable) -> {
|
||||||
if (throwable != null) {
|
if (throwable != null) {
|
||||||
future.completeExceptionally(throwable);
|
future.completeExceptionally(throwable);
|
||||||
} else {
|
} else {
|
||||||
@@ -191,15 +184,6 @@ abstract class ChunkMapMixin extends ChunkStorage implements ChunkSystemChunkMap
|
|||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
|
||||||
* @reason This map is not needed, we maintain our own ordered set of chunks to autosave.
|
|
||||||
* @author Spottedleaf
|
|
||||||
*/
|
|
||||||
@Overwrite
|
|
||||||
public void setChunkUnsaved(final ChunkPos pos) {
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @reason Route to new chunk system hooks
|
* @reason Route to new chunk system hooks
|
||||||
* @author Spottedleaf
|
* @author Spottedleaf
|
||||||
@@ -277,15 +261,6 @@ abstract class ChunkMapMixin extends ChunkStorage implements ChunkSystemChunkMap
|
|||||||
throw new UnsupportedOperationException();
|
throw new UnsupportedOperationException();
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
|
||||||
* @reason Destroy old chunk system hooks
|
|
||||||
* @author Spottedleaf
|
|
||||||
*/
|
|
||||||
@Overwrite
|
|
||||||
public void onLevelChange(final ChunkPos chunkPos, final IntSupplier intSupplier, final int i, final IntConsumer intConsumer) {
|
|
||||||
throw new UnsupportedOperationException();
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @reason Destroy old chunk system hooks
|
* @reason Destroy old chunk system hooks
|
||||||
* @author Spottedleaf
|
* @author Spottedleaf
|
||||||
@@ -334,15 +309,6 @@ abstract class ChunkMapMixin extends ChunkStorage implements ChunkSystemChunkMap
|
|||||||
((ChunkSystemServerLevel)this.level).moonrise$getChunkTaskScheduler().chunkHolderManager.autoSave();
|
((ChunkSystemServerLevel)this.level).moonrise$getChunkTaskScheduler().chunkHolderManager.autoSave();
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
|
||||||
* @reason Destroy old chunk system hooks
|
|
||||||
* @author Spottedleaf
|
|
||||||
*/
|
|
||||||
@Overwrite
|
|
||||||
public void saveChunksEagerly(final BooleanSupplier hasTime) {
|
|
||||||
throw new UnsupportedOperationException();
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @reason Destroy old chunk system hooks
|
* @reason Destroy old chunk system hooks
|
||||||
* @author Spottedleaf
|
* @author Spottedleaf
|
||||||
@@ -437,7 +403,7 @@ abstract class ChunkMapMixin extends ChunkStorage implements ChunkSystemChunkMap
|
|||||||
* @author Spottedleaf
|
* @author Spottedleaf
|
||||||
*/
|
*/
|
||||||
@Overwrite
|
@Overwrite
|
||||||
public void onChunkReadyToSend(final ChunkHolder holder, final LevelChunk chunk) {
|
public void onChunkReadyToSend(final LevelChunk chunk) {
|
||||||
throw new UnsupportedOperationException();
|
throw new UnsupportedOperationException();
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -456,7 +422,7 @@ abstract class ChunkMapMixin extends ChunkStorage implements ChunkSystemChunkMap
|
|||||||
* @see NewChunkHolder#save(boolean)
|
* @see NewChunkHolder#save(boolean)
|
||||||
*/
|
*/
|
||||||
@Overwrite
|
@Overwrite
|
||||||
public boolean saveChunkIfNeeded(final ChunkHolder chunkHolder, final long time) {
|
public boolean saveChunkIfNeeded(final ChunkHolder chunkHolder) {
|
||||||
throw new UnsupportedOperationException();
|
throw new UnsupportedOperationException();
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -500,7 +466,7 @@ abstract class ChunkMapMixin extends ChunkStorage implements ChunkSystemChunkMap
|
|||||||
*/
|
*/
|
||||||
@Overwrite
|
@Overwrite
|
||||||
public int getPlayerViewDistance(final ServerPlayer player) {
|
public int getPlayerViewDistance(final ServerPlayer player) {
|
||||||
return PlatformHooks.get().getSendViewDistance(player);
|
return ChunkSystem.getSendViewDistance(player);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -539,39 +505,6 @@ abstract class ChunkMapMixin extends ChunkStorage implements ChunkSystemChunkMap
|
|||||||
throw new UnsupportedOperationException();
|
throw new UnsupportedOperationException();
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
|
||||||
* @reason Route to new chunk system
|
|
||||||
* @author Spottedleaf
|
|
||||||
*/
|
|
||||||
@Redirect(
|
|
||||||
method = "collectSpawningChunks",
|
|
||||||
at = @At(
|
|
||||||
value = "INVOKE",
|
|
||||||
target = "Lit/unimi/dsi/fastutil/longs/Long2ObjectLinkedOpenHashMap;get(J)Ljava/lang/Object;"
|
|
||||||
)
|
|
||||||
)
|
|
||||||
private <V> V redirectChunkHolderGetForSpawning(final Long2ObjectLinkedOpenHashMap<V> instance, final long key) {
|
|
||||||
return (V)this.getVisibleChunkIfPresent(key);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @reason Route to new chunk system
|
|
||||||
* @author Spottedleaf
|
|
||||||
*/
|
|
||||||
@Redirect(
|
|
||||||
method = {
|
|
||||||
"method_67499",
|
|
||||||
"lambda$forEachBlockTickingChunk$36"
|
|
||||||
},
|
|
||||||
at = @At(
|
|
||||||
value = "INVOKE",
|
|
||||||
target = "Lit/unimi/dsi/fastutil/longs/Long2ObjectLinkedOpenHashMap;get(J)Ljava/lang/Object;"
|
|
||||||
)
|
|
||||||
)
|
|
||||||
private <V> V redirectChunkHolderGetForBlockTicking(final Long2ObjectLinkedOpenHashMap<V> instance, final long key) {
|
|
||||||
return (V)this.getVisibleChunkIfPresent(key);
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public CompletableFuture<Optional<CompoundTag>> read(final ChunkPos pos) {
|
public CompletableFuture<Optional<CompoundTag>> read(final ChunkPos pos) {
|
||||||
final CompletableFuture<Optional<CompoundTag>> ret = new CompletableFuture<>();
|
final CompletableFuture<Optional<CompoundTag>> ret = new CompletableFuture<>();
|
||||||
@@ -591,11 +524,10 @@ abstract class ChunkMapMixin extends ChunkStorage implements ChunkSystemChunkMap
|
|||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public CompletableFuture<Void> write(final ChunkPos pos, final Supplier<CompoundTag> tag) {
|
public CompletableFuture<Void> write(final ChunkPos pos, final CompoundTag tag) {
|
||||||
MoonriseRegionFileIO.scheduleSave(
|
MoonriseRegionFileIO.scheduleSave(
|
||||||
this.level, pos.x, pos.z, tag.get(),
|
this.level, pos.x, pos.z, tag,
|
||||||
MoonriseRegionFileIO.RegionFileType.CHUNK_DATA
|
MoonriseRegionFileIO.RegionFileType.CHUNK_DATA);
|
||||||
);
|
|
||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -616,7 +548,7 @@ abstract class ChunkMapMixin extends ChunkStorage implements ChunkSystemChunkMap
|
|||||||
)
|
)
|
||||||
)
|
)
|
||||||
private void avoidUpdateChunkTrackingInUpdate(final ChunkMap instance, final ServerPlayer serverPlayer) {
|
private void avoidUpdateChunkTrackingInUpdate(final ChunkMap instance, final ServerPlayer serverPlayer) {
|
||||||
PlatformHooks.get().addPlayerToDistanceMaps(this.level, serverPlayer);
|
ChunkSystem.addPlayerToDistanceMaps(this.level, serverPlayer);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -646,7 +578,7 @@ abstract class ChunkMapMixin extends ChunkStorage implements ChunkSystemChunkMap
|
|||||||
)
|
)
|
||||||
private void avoidApplyChunkTrackingViewInUpdate(final ChunkMap instance, final ServerPlayer serverPlayer,
|
private void avoidApplyChunkTrackingViewInUpdate(final ChunkMap instance, final ServerPlayer serverPlayer,
|
||||||
final ChunkTrackingView chunkTrackingView) {
|
final ChunkTrackingView chunkTrackingView) {
|
||||||
PlatformHooks.get().removePlayerFromDistanceMaps(this.level, serverPlayer);
|
ChunkSystem.removePlayerFromDistanceMaps(this.level, serverPlayer);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -660,7 +592,7 @@ abstract class ChunkMapMixin extends ChunkStorage implements ChunkSystemChunkMap
|
|||||||
)
|
)
|
||||||
)
|
)
|
||||||
private void updateMapsHook(final ServerPlayer player, final CallbackInfo ci) {
|
private void updateMapsHook(final ServerPlayer player, final CallbackInfo ci) {
|
||||||
PlatformHooks.get().updateMaps(this.level, player);
|
ChunkSystem.updateMaps(this.level, player);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
|||||||
@@ -3,13 +3,13 @@ package ca.spottedleaf.moonrise.mixin.chunk_system;
|
|||||||
import net.minecraft.core.SectionPos;
|
import net.minecraft.core.SectionPos;
|
||||||
import net.minecraft.world.entity.ai.village.poi.PoiManager;
|
import net.minecraft.world.entity.ai.village.poi.PoiManager;
|
||||||
import net.minecraft.world.level.chunk.LevelChunkSection;
|
import net.minecraft.world.level.chunk.LevelChunkSection;
|
||||||
import net.minecraft.world.level.chunk.storage.SerializableChunkData;
|
import net.minecraft.world.level.chunk.storage.ChunkSerializer;
|
||||||
import org.spongepowered.asm.mixin.Mixin;
|
import org.spongepowered.asm.mixin.Mixin;
|
||||||
import org.spongepowered.asm.mixin.injection.At;
|
import org.spongepowered.asm.mixin.injection.At;
|
||||||
import org.spongepowered.asm.mixin.injection.Redirect;
|
import org.spongepowered.asm.mixin.injection.Redirect;
|
||||||
|
|
||||||
@Mixin(SerializableChunkData.class)
|
@Mixin(ChunkSerializer.class)
|
||||||
abstract class SerializableChunkDataMixin {
|
abstract class ChunkSerializerMixin {
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @reason Chunk system handles this during full transition
|
* @reason Chunk system handles this during full transition
|
||||||
@@ -23,6 +23,6 @@ abstract class SerializableChunkDataMixin {
|
|||||||
target = "Lnet/minecraft/world/entity/ai/village/poi/PoiManager;checkConsistencyWithBlocks(Lnet/minecraft/core/SectionPos;Lnet/minecraft/world/level/chunk/LevelChunkSection;)V"
|
target = "Lnet/minecraft/world/entity/ai/village/poi/PoiManager;checkConsistencyWithBlocks(Lnet/minecraft/core/SectionPos;Lnet/minecraft/world/level/chunk/LevelChunkSection;)V"
|
||||||
)
|
)
|
||||||
)
|
)
|
||||||
private void skipConsistencyCheck(final PoiManager instance, final SectionPos sectionPos, final LevelChunkSection levelChunkSection) {}
|
private static void skipConsistencyCheck(PoiManager instance, SectionPos sectionPos, LevelChunkSection levelChunkSection) {}
|
||||||
|
|
||||||
}
|
}
|
||||||
@@ -22,13 +22,12 @@ import org.spongepowered.asm.mixin.injection.callback.CallbackInfo;
|
|||||||
import java.io.IOException;
|
import java.io.IOException;
|
||||||
import java.util.Optional;
|
import java.util.Optional;
|
||||||
import java.util.concurrent.CompletableFuture;
|
import java.util.concurrent.CompletableFuture;
|
||||||
import java.util.function.Supplier;
|
|
||||||
|
|
||||||
@Mixin(ChunkStorage.class)
|
@Mixin(ChunkStorage.class)
|
||||||
abstract class ChunkStorageMixin implements ChunkSystemChunkStorage, AutoCloseable {
|
abstract class ChunkStorageMixin implements ChunkSystemChunkStorage, AutoCloseable {
|
||||||
|
|
||||||
@Shadow
|
@Shadow
|
||||||
public IOWorker worker;
|
private IOWorker worker;
|
||||||
|
|
||||||
@Unique
|
@Unique
|
||||||
private static final Logger LOGGER = LogUtils.getLogger();
|
private static final Logger LOGGER = LogUtils.getLogger();
|
||||||
@@ -119,13 +118,13 @@ abstract class ChunkStorageMixin implements ChunkSystemChunkStorage, AutoCloseab
|
|||||||
method = "write",
|
method = "write",
|
||||||
at = @At(
|
at = @At(
|
||||||
value = "INVOKE",
|
value = "INVOKE",
|
||||||
target = "Lnet/minecraft/world/level/chunk/storage/IOWorker;store(Lnet/minecraft/world/level/ChunkPos;Ljava/util/function/Supplier;)Ljava/util/concurrent/CompletableFuture;"
|
target = "Lnet/minecraft/world/level/chunk/storage/IOWorker;store(Lnet/minecraft/world/level/ChunkPos;Lnet/minecraft/nbt/CompoundTag;)Ljava/util/concurrent/CompletableFuture;"
|
||||||
)
|
)
|
||||||
)
|
)
|
||||||
private CompletableFuture<Void> redirectWrite(final IOWorker instance, final ChunkPos chunkPos,
|
private CompletableFuture<Void> redirectWrite(final IOWorker instance, final ChunkPos chunkPos,
|
||||||
final Supplier<CompoundTag> compoundTag) {
|
final CompoundTag compoundTag) {
|
||||||
try {
|
try {
|
||||||
this.storage.write(chunkPos, compoundTag.get());
|
this.storage.write(chunkPos, compoundTag);
|
||||||
return CompletableFuture.completedFuture(null);
|
return CompletableFuture.completedFuture(null);
|
||||||
} catch (final Throwable throwable) {
|
} catch (final Throwable throwable) {
|
||||||
return CompletableFuture.failedFuture(throwable);
|
return CompletableFuture.failedFuture(throwable);
|
||||||
|
|||||||
@@ -41,8 +41,8 @@ abstract class ClientLevelMixin extends Level implements ChunkSystemLevel {
|
|||||||
@Final
|
@Final
|
||||||
private ClientChunkCache chunkSource;
|
private ClientChunkCache chunkSource;
|
||||||
|
|
||||||
protected ClientLevelMixin(final WritableLevelData writableLevelData, final ResourceKey<Level> resourceKey, final RegistryAccess registryAccess, final Holder<DimensionType> holder, final boolean bl, final boolean bl2, final long l, final int i) {
|
protected ClientLevelMixin(WritableLevelData writableLevelData, ResourceKey<Level> resourceKey, RegistryAccess registryAccess, Holder<DimensionType> holder, Supplier<ProfilerFiller> supplier, boolean bl, boolean bl2, long l, int i) {
|
||||||
super(writableLevelData, resourceKey, registryAccess, holder, bl, bl2, l, i);
|
super(writableLevelData, resourceKey, registryAccess, holder, supplier, bl, bl2, l, i);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -55,8 +55,9 @@ abstract class ClientLevelMixin extends Level implements ChunkSystemLevel {
|
|||||||
value = "RETURN"
|
value = "RETURN"
|
||||||
)
|
)
|
||||||
)
|
)
|
||||||
private void init(ClientPacketListener clientPacketListener, ClientLevel.ClientLevelData clientLevelData, ResourceKey<Level> resourceKey,
|
private void init(ClientPacketListener clientPacketListener, ClientLevel.ClientLevelData clientLevelData,
|
||||||
Holder<DimensionType> holder, int i, int j, LevelRenderer levelRenderer, boolean bl, long l, int k, CallbackInfo ci) {
|
ResourceKey<Level> resourceKey, Holder<DimensionType> holder, int i, int j, Supplier<ProfilerFiller> supplier,
|
||||||
|
LevelRenderer levelRenderer, boolean bl, long l, CallbackInfo ci) {
|
||||||
this.entityStorage = null;
|
this.entityStorage = null;
|
||||||
|
|
||||||
this.moonrise$setEntityLookup(new ClientEntityLookup(this, ((ClientLevel)(Object)this).new EntityCallbacks()));
|
this.moonrise$setEntityLookup(new ClientEntityLookup(this, ((ClientLevel)(Object)this).new EntityCallbacks()));
|
||||||
|
|||||||
@@ -1,27 +1,21 @@
|
|||||||
package ca.spottedleaf.moonrise.mixin.chunk_system;
|
package ca.spottedleaf.moonrise.mixin.chunk_system;
|
||||||
|
|
||||||
import ca.spottedleaf.moonrise.common.list.ReferenceList;
|
|
||||||
import ca.spottedleaf.moonrise.common.util.CoordinateUtils;
|
|
||||||
import ca.spottedleaf.moonrise.common.util.MoonriseConstants;
|
|
||||||
import ca.spottedleaf.moonrise.patches.chunk_system.level.ChunkSystemServerLevel;
|
import ca.spottedleaf.moonrise.patches.chunk_system.level.ChunkSystemServerLevel;
|
||||||
import ca.spottedleaf.moonrise.patches.chunk_system.level.chunk.ChunkSystemDistanceManager;
|
import ca.spottedleaf.moonrise.patches.chunk_system.level.chunk.ChunkSystemDistanceManager;
|
||||||
import ca.spottedleaf.moonrise.patches.chunk_system.scheduling.ChunkHolderManager;
|
import ca.spottedleaf.moonrise.patches.chunk_system.scheduling.ChunkHolderManager;
|
||||||
import ca.spottedleaf.moonrise.patches.chunk_system.scheduling.NewChunkHolder;
|
import ca.spottedleaf.moonrise.patches.chunk_system.scheduling.NewChunkHolder;
|
||||||
import ca.spottedleaf.moonrise.patches.chunk_system.ticket.ChunkSystemTicketStorage;
|
import it.unimi.dsi.fastutil.longs.Long2ObjectOpenHashMap;
|
||||||
import it.unimi.dsi.fastutil.longs.LongConsumer;
|
|
||||||
import it.unimi.dsi.fastutil.longs.LongSet;
|
import it.unimi.dsi.fastutil.longs.LongSet;
|
||||||
import net.minecraft.server.level.ChunkHolder;
|
import net.minecraft.server.level.ChunkHolder;
|
||||||
import net.minecraft.server.level.ChunkMap;
|
import net.minecraft.server.level.ChunkMap;
|
||||||
|
import net.minecraft.server.level.ChunkTaskPriorityQueueSorter;
|
||||||
import net.minecraft.server.level.DistanceManager;
|
import net.minecraft.server.level.DistanceManager;
|
||||||
import net.minecraft.server.level.LoadingChunkTracker;
|
|
||||||
import net.minecraft.server.level.SimulationChunkTracker;
|
|
||||||
import net.minecraft.server.level.ThrottlingChunkTaskDispatcher;
|
|
||||||
import net.minecraft.server.level.Ticket;
|
import net.minecraft.server.level.Ticket;
|
||||||
import net.minecraft.util.Mth;
|
import net.minecraft.server.level.TicketType;
|
||||||
|
import net.minecraft.server.level.TickingTracker;
|
||||||
|
import net.minecraft.util.SortedArraySet;
|
||||||
|
import net.minecraft.util.thread.ProcessorHandle;
|
||||||
import net.minecraft.world.level.ChunkPos;
|
import net.minecraft.world.level.ChunkPos;
|
||||||
import net.minecraft.world.level.TicketStorage;
|
|
||||||
import net.minecraft.world.level.chunk.LevelChunk;
|
|
||||||
import org.spongepowered.asm.mixin.Final;
|
|
||||||
import org.spongepowered.asm.mixin.Mixin;
|
import org.spongepowered.asm.mixin.Mixin;
|
||||||
import org.spongepowered.asm.mixin.Overwrite;
|
import org.spongepowered.asm.mixin.Overwrite;
|
||||||
import org.spongepowered.asm.mixin.Shadow;
|
import org.spongepowered.asm.mixin.Shadow;
|
||||||
@@ -29,7 +23,6 @@ import org.spongepowered.asm.mixin.injection.At;
|
|||||||
import org.spongepowered.asm.mixin.injection.Inject;
|
import org.spongepowered.asm.mixin.injection.Inject;
|
||||||
import org.spongepowered.asm.mixin.injection.Redirect;
|
import org.spongepowered.asm.mixin.injection.Redirect;
|
||||||
import org.spongepowered.asm.mixin.injection.callback.CallbackInfo;
|
import org.spongepowered.asm.mixin.injection.callback.CallbackInfo;
|
||||||
import java.util.Objects;
|
|
||||||
import java.util.Set;
|
import java.util.Set;
|
||||||
import java.util.concurrent.Executor;
|
import java.util.concurrent.Executor;
|
||||||
|
|
||||||
@@ -37,14 +30,13 @@ import java.util.concurrent.Executor;
|
|||||||
abstract class DistanceManagerMixin implements ChunkSystemDistanceManager {
|
abstract class DistanceManagerMixin implements ChunkSystemDistanceManager {
|
||||||
|
|
||||||
@Shadow
|
@Shadow
|
||||||
public LoadingChunkTracker loadingChunkTracker;
|
Long2ObjectOpenHashMap<SortedArraySet<Ticket<?>>> tickets;
|
||||||
|
|
||||||
@Shadow
|
@Shadow
|
||||||
public SimulationChunkTracker simulationChunkTracker;
|
private DistanceManager.ChunkTicketTracker ticketTracker;
|
||||||
|
|
||||||
@Shadow
|
@Shadow
|
||||||
@Final
|
private TickingTracker tickingTicketsTracker;
|
||||||
private TicketStorage ticketStorage;
|
|
||||||
|
|
||||||
@Shadow
|
@Shadow
|
||||||
private DistanceManager.PlayerTicketTracker playerTicketManager;
|
private DistanceManager.PlayerTicketTracker playerTicketManager;
|
||||||
@@ -53,7 +45,13 @@ abstract class DistanceManagerMixin implements ChunkSystemDistanceManager {
|
|||||||
Set<ChunkHolder> chunksToUpdateFutures;
|
Set<ChunkHolder> chunksToUpdateFutures;
|
||||||
|
|
||||||
@Shadow
|
@Shadow
|
||||||
ThrottlingChunkTaskDispatcher ticketDispatcher;
|
ChunkTaskPriorityQueueSorter ticketThrottler;
|
||||||
|
|
||||||
|
@Shadow
|
||||||
|
ProcessorHandle<ChunkTaskPriorityQueueSorter.Message<Runnable>> ticketThrottlerInput;
|
||||||
|
|
||||||
|
@Shadow
|
||||||
|
ProcessorHandle<ChunkTaskPriorityQueueSorter.Release> ticketThrottlerReleaser;
|
||||||
|
|
||||||
@Shadow
|
@Shadow
|
||||||
LongSet ticketsToRelease;
|
LongSet ticketsToRelease;
|
||||||
@@ -62,8 +60,10 @@ abstract class DistanceManagerMixin implements ChunkSystemDistanceManager {
|
|||||||
Executor mainThreadExecutor;
|
Executor mainThreadExecutor;
|
||||||
|
|
||||||
@Shadow
|
@Shadow
|
||||||
private int simulationDistance;
|
private DistanceManager.FixedPlayerDistanceChunkTracker naturalSpawnChunkCounter;
|
||||||
|
|
||||||
|
@Shadow
|
||||||
|
private int simulationDistance;
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public ChunkMap moonrise$getChunkMap() {
|
public ChunkMap moonrise$getChunkMap() {
|
||||||
@@ -71,8 +71,7 @@ abstract class DistanceManagerMixin implements ChunkSystemDistanceManager {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @reason Destroy old chunk system state to prevent it from being used, and set the chunk map
|
* @reason Destroy old chunk system state to prevent it from being used
|
||||||
* for the ticket storage
|
|
||||||
* @author Spottedleaf
|
* @author Spottedleaf
|
||||||
*/
|
*/
|
||||||
@Inject(
|
@Inject(
|
||||||
@@ -82,23 +81,33 @@ abstract class DistanceManagerMixin implements ChunkSystemDistanceManager {
|
|||||||
)
|
)
|
||||||
)
|
)
|
||||||
private void destroyFields(final CallbackInfo ci) {
|
private void destroyFields(final CallbackInfo ci) {
|
||||||
this.loadingChunkTracker = null;
|
this.tickets = null;
|
||||||
this.simulationChunkTracker = null;
|
this.ticketTracker = null;
|
||||||
|
this.tickingTicketsTracker = null;
|
||||||
this.playerTicketManager = null;
|
this.playerTicketManager = null;
|
||||||
this.chunksToUpdateFutures = null;
|
this.chunksToUpdateFutures = null;
|
||||||
this.ticketDispatcher = null;
|
this.ticketThrottler = null;
|
||||||
|
this.ticketThrottlerInput = null;
|
||||||
|
this.ticketThrottlerReleaser = null;
|
||||||
this.ticketsToRelease = null;
|
this.ticketsToRelease = null;
|
||||||
this.mainThreadExecutor = null;
|
this.mainThreadExecutor = null;
|
||||||
this.simulationDistance = -1;
|
this.simulationDistance = -1;
|
||||||
|
|
||||||
((ChunkSystemTicketStorage)this.ticketStorage).moonrise$setChunkMap(this.moonrise$getChunkMap());
|
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public final ChunkHolderManager moonrise$getChunkHolderManager() {
|
public ChunkHolderManager moonrise$getChunkHolderManager() {
|
||||||
return ((ChunkSystemServerLevel)this.moonrise$getChunkMap().level).moonrise$getChunkTaskScheduler().chunkHolderManager;
|
return ((ChunkSystemServerLevel)this.moonrise$getChunkMap().level).moonrise$getChunkTaskScheduler().chunkHolderManager;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @reason Route to new chunk system
|
||||||
|
* @author Spottedleaf
|
||||||
|
*/
|
||||||
|
@Overwrite
|
||||||
|
public void purgeStaleTickets() {
|
||||||
|
this.moonrise$getChunkHolderManager().tick();
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @reason Route to new chunk system
|
* @reason Route to new chunk system
|
||||||
* @author Spottedleaf
|
* @author Spottedleaf
|
||||||
@@ -108,6 +117,46 @@ abstract class DistanceManagerMixin implements ChunkSystemDistanceManager {
|
|||||||
return this.moonrise$getChunkHolderManager().processTicketUpdates();
|
return this.moonrise$getChunkHolderManager().processTicketUpdates();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @reason Route to new chunk system
|
||||||
|
* @author Spottedleaf
|
||||||
|
*/
|
||||||
|
@Overwrite
|
||||||
|
public void addTicket(final long pos, final Ticket<?> ticket) {
|
||||||
|
this.moonrise$getChunkHolderManager().addTicketAtLevel((TicketType)ticket.getType(), pos, ticket.getTicketLevel(), ticket.key);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @reason Route to new chunk system
|
||||||
|
* @author Spottedleaf
|
||||||
|
*/
|
||||||
|
@Overwrite
|
||||||
|
public void removeTicket(final long pos, final Ticket<?> ticket) {
|
||||||
|
this.moonrise$getChunkHolderManager().removeTicketAtLevel((TicketType)ticket.getType(), pos, ticket.getTicketLevel(), ticket.key);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @reason Remove old chunk system hooks
|
||||||
|
* @author Spottedleaf
|
||||||
|
*/
|
||||||
|
@Overwrite
|
||||||
|
public SortedArraySet<Ticket<?>> getTickets(final long pos) {
|
||||||
|
throw new UnsupportedOperationException();
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @reason Route to new chunk system
|
||||||
|
* @author Spottedleaf
|
||||||
|
*/
|
||||||
|
@Overwrite
|
||||||
|
public void updateChunkForced(final ChunkPos pos, final boolean forced) {
|
||||||
|
if (forced) {
|
||||||
|
this.moonrise$getChunkHolderManager().addTicketAtLevel(TicketType.FORCED, pos, ChunkMap.FORCED_TICKET_LEVEL, pos);
|
||||||
|
} else {
|
||||||
|
this.moonrise$getChunkHolderManager().removeTicketAtLevel(TicketType.FORCED, pos, ChunkMap.FORCED_TICKET_LEVEL, pos);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @reason Remove old chunk system hooks
|
* @reason Remove old chunk system hooks
|
||||||
* @author Spottedleaf
|
* @author Spottedleaf
|
||||||
@@ -130,10 +179,11 @@ abstract class DistanceManagerMixin implements ChunkSystemDistanceManager {
|
|||||||
method = "addPlayer",
|
method = "addPlayer",
|
||||||
at = @At(
|
at = @At(
|
||||||
value = "INVOKE",
|
value = "INVOKE",
|
||||||
target = "Lnet/minecraft/world/level/TicketStorage;addTicket(Lnet/minecraft/server/level/Ticket;Lnet/minecraft/world/level/ChunkPos;)V"
|
target = "Lnet/minecraft/server/level/TickingTracker;addTicket(Lnet/minecraft/server/level/TicketType;Lnet/minecraft/world/level/ChunkPos;ILjava/lang/Object;)V"
|
||||||
)
|
)
|
||||||
)
|
)
|
||||||
private void skipTickingTicketTrackerAdd(final TicketStorage instance, final Ticket ticket, final ChunkPos pos) {}
|
private <T> void skipTickingTicketTrackerAdd(final TickingTracker instance, final TicketType<T> ticketType,
|
||||||
|
final ChunkPos chunkPos, final int i, final T object) {}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @reason Remove old chunk system hooks
|
* @reason Remove old chunk system hooks
|
||||||
@@ -172,10 +222,11 @@ abstract class DistanceManagerMixin implements ChunkSystemDistanceManager {
|
|||||||
method = "removePlayer",
|
method = "removePlayer",
|
||||||
at = @At(
|
at = @At(
|
||||||
value = "INVOKE",
|
value = "INVOKE",
|
||||||
target = "Lnet/minecraft/world/level/TicketStorage;removeTicket(Lnet/minecraft/server/level/Ticket;Lnet/minecraft/world/level/ChunkPos;)V"
|
target = "Lnet/minecraft/server/level/TickingTracker;removeTicket(Lnet/minecraft/server/level/TicketType;Lnet/minecraft/world/level/ChunkPos;ILjava/lang/Object;)V"
|
||||||
)
|
)
|
||||||
)
|
)
|
||||||
private void skipTickingTicketTrackerRemove(final TicketStorage instance, final Ticket ticket, final ChunkPos pos) {}
|
private <T> void skipTickingTicketTrackerRemove(final TickingTracker instance, final TicketType<T> ticketType,
|
||||||
|
final ChunkPos chunkPos, final int i, final T object) {}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @reason Remove old chunk system hooks
|
* @reason Remove old chunk system hooks
|
||||||
@@ -226,9 +277,8 @@ abstract class DistanceManagerMixin implements ChunkSystemDistanceManager {
|
|||||||
* @author Spottedleaf
|
* @author Spottedleaf
|
||||||
*/
|
*/
|
||||||
@Overwrite
|
@Overwrite
|
||||||
public int getChunkLevel(final long pos, final boolean simulation) {
|
public String getTicketDebugString(final long pos) {
|
||||||
final NewChunkHolder chunkHolder = this.moonrise$getChunkHolderManager().getChunkHolder(pos);
|
return this.moonrise$getChunkHolderManager().getTicketDebugString(pos);
|
||||||
return chunkHolder == null ? ChunkHolderManager.MAX_TICKET_LEVEL + 1 : chunkHolder.getTicketLevel();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -246,28 +296,7 @@ abstract class DistanceManagerMixin implements ChunkSystemDistanceManager {
|
|||||||
*/
|
*/
|
||||||
@Overwrite
|
@Overwrite
|
||||||
public void updateSimulationDistance(final int simulationDistance) {
|
public void updateSimulationDistance(final int simulationDistance) {
|
||||||
// note: vanilla does not clamp to 0, but we do simply because we need a min of 0
|
((ChunkSystemServerLevel)this.moonrise$getChunkMap().level).moonrise$getPlayerChunkLoader().setTickDistance(simulationDistance);
|
||||||
final int clamped = Mth.clamp(simulationDistance, 0, MoonriseConstants.MAX_VIEW_DISTANCE);
|
|
||||||
|
|
||||||
((ChunkSystemServerLevel)this.moonrise$getChunkMap().level).moonrise$getPlayerChunkLoader().setTickDistance(clamped);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @reason Route to new chunk system
|
|
||||||
* @author Spottedleaf
|
|
||||||
*/
|
|
||||||
@Overwrite
|
|
||||||
public void forEachEntityTickingChunk(final LongConsumer consumer) {
|
|
||||||
final ReferenceList<LevelChunk> chunks = ((ChunkSystemServerLevel)this.moonrise$getChunkMap().level).moonrise$getEntityTickingChunks();
|
|
||||||
final LevelChunk[] raw = chunks.getRawDataUnchecked();
|
|
||||||
final int size = chunks.size();
|
|
||||||
|
|
||||||
Objects.checkFromToIndex(0, size, raw.length);
|
|
||||||
for (int i = 0; i < size; ++i) {
|
|
||||||
final LevelChunk chunk = raw[i];
|
|
||||||
|
|
||||||
consumer.accept(CoordinateUtils.getChunkKey(chunk.getPos()));
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -276,6 +305,40 @@ abstract class DistanceManagerMixin implements ChunkSystemDistanceManager {
|
|||||||
*/
|
*/
|
||||||
@Overwrite
|
@Overwrite
|
||||||
public String getDebugStatus() {
|
public String getDebugStatus() {
|
||||||
return "N/A";
|
return "No DistanceManager stats available";
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @reason Remove old chunk system hooks
|
||||||
|
* @author Spottedleaf
|
||||||
|
*/
|
||||||
|
@Overwrite
|
||||||
|
public void dumpTickets(final String file) {
|
||||||
|
throw new UnsupportedOperationException();
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @reason Remove old chunk system hooks
|
||||||
|
* @author Spottedleaf
|
||||||
|
*/
|
||||||
|
@Overwrite
|
||||||
|
public TickingTracker tickingTracker() {
|
||||||
|
throw new UnsupportedOperationException();
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @reason This hack is not required anymore, see {@link MinecraftServerMixin}
|
||||||
|
* @author Spottedleaf
|
||||||
|
*/
|
||||||
|
@Overwrite
|
||||||
|
public void removeTicketsOnClosing() {}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @reason This hack is not required anymore, see {@link MinecraftServerMixin}
|
||||||
|
* @author Spottedleaf
|
||||||
|
*/
|
||||||
|
@Overwrite
|
||||||
|
public boolean hasTickets() {
|
||||||
|
throw new UnsupportedOperationException();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,30 +0,0 @@
|
|||||||
package ca.spottedleaf.moonrise.mixin.chunk_system;
|
|
||||||
|
|
||||||
import net.minecraft.core.SectionPos;
|
|
||||||
import net.minecraft.world.level.gameevent.DynamicGameEventListener;
|
|
||||||
import org.jetbrains.annotations.Nullable;
|
|
||||||
import org.spongepowered.asm.mixin.Mixin;
|
|
||||||
import org.spongepowered.asm.mixin.Shadow;
|
|
||||||
import org.spongepowered.asm.mixin.injection.At;
|
|
||||||
import org.spongepowered.asm.mixin.injection.Inject;
|
|
||||||
import org.spongepowered.asm.mixin.injection.callback.CallbackInfo;
|
|
||||||
|
|
||||||
@Mixin(DynamicGameEventListener.class)
|
|
||||||
abstract class DynamicGameEventListenerMixin {
|
|
||||||
@Shadow
|
|
||||||
@Nullable
|
|
||||||
private SectionPos lastSection;
|
|
||||||
|
|
||||||
@Inject(
|
|
||||||
method = "remove",
|
|
||||||
at = @At("RETURN")
|
|
||||||
)
|
|
||||||
private void onRemove(final CallbackInfo ci) {
|
|
||||||
// We need to unset the last section when removed, otherwise if the same instance is re-added at the same position it
|
|
||||||
// will assume there was no change and fail to re-register.
|
|
||||||
// In vanilla, chunks rarely unload and re-load quickly enough to trigger this issue. However, our chunk system has a
|
|
||||||
// quirk where fast chunk reload cycles will often occur on player login (see PR #22).
|
|
||||||
// So we fix this vanilla oversight as our changes cause it to manifest in bugs much more often (see issue #87).
|
|
||||||
this.lastSection = null;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -4,7 +4,6 @@ import ca.spottedleaf.moonrise.patches.chunk_system.level.chunk.ChunkData;
|
|||||||
import ca.spottedleaf.moonrise.patches.chunk_system.entity.ChunkSystemEntity;
|
import ca.spottedleaf.moonrise.patches.chunk_system.entity.ChunkSystemEntity;
|
||||||
import ca.spottedleaf.moonrise.patches.chunk_system.level.ChunkSystemLevel;
|
import ca.spottedleaf.moonrise.patches.chunk_system.level.ChunkSystemLevel;
|
||||||
import com.google.common.collect.ImmutableList;
|
import com.google.common.collect.ImmutableList;
|
||||||
import com.mojang.logging.LogUtils;
|
|
||||||
import net.minecraft.server.level.FullChunkStatus;
|
import net.minecraft.server.level.FullChunkStatus;
|
||||||
import net.minecraft.world.entity.Entity;
|
import net.minecraft.world.entity.Entity;
|
||||||
import net.minecraft.world.entity.player.Player;
|
import net.minecraft.world.entity.player.Player;
|
||||||
@@ -33,6 +32,10 @@ abstract class EntityMixin implements ChunkSystemEntity {
|
|||||||
@Shadow
|
@Shadow
|
||||||
protected abstract Stream<Entity> getIndirectPassengersStream();
|
protected abstract Stream<Entity> getIndirectPassengersStream();
|
||||||
|
|
||||||
|
@Shadow
|
||||||
|
@Final
|
||||||
|
private static Logger LOGGER;
|
||||||
|
|
||||||
@Shadow
|
@Shadow
|
||||||
private Level level;
|
private Level level;
|
||||||
|
|
||||||
@@ -40,8 +43,6 @@ abstract class EntityMixin implements ChunkSystemEntity {
|
|||||||
@Nullable
|
@Nullable
|
||||||
private Entity.RemovalReason removalReason;
|
private Entity.RemovalReason removalReason;
|
||||||
|
|
||||||
@Unique
|
|
||||||
private static final Logger LOGGER = LogUtils.getLogger();
|
|
||||||
|
|
||||||
@Unique
|
@Unique
|
||||||
private final boolean isHardColliding = this.moonrise$isHardCollidingUncached();
|
private final boolean isHardColliding = this.moonrise$isHardCollidingUncached();
|
||||||
|
|||||||
@@ -25,7 +25,7 @@ abstract class EntityTickListMixin {
|
|||||||
private Int2ObjectMap<Entity> passive;
|
private Int2ObjectMap<Entity> passive;
|
||||||
|
|
||||||
@Unique
|
@Unique
|
||||||
private final IteratorSafeOrderedReferenceSet<Entity> entities = new IteratorSafeOrderedReferenceSet<>(Entity.class);
|
private final IteratorSafeOrderedReferenceSet<Entity> entities = new IteratorSafeOrderedReferenceSet<>();
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @reason Initialise new fields and destroy old state
|
* @reason Initialise new fields and destroy old state
|
||||||
|
|||||||
@@ -195,6 +195,15 @@ abstract class GenerationChunkHolderMixin {
|
|||||||
throw new UnsupportedOperationException();
|
throw new UnsupportedOperationException();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @reason Chunk system is not built on futures anymore
|
||||||
|
* @author Spottedleaf
|
||||||
|
*/
|
||||||
|
@Overwrite
|
||||||
|
public int getGenerationRefCount() {
|
||||||
|
throw new UnsupportedOperationException();
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @reason Route to new chunk holder
|
* @reason Route to new chunk holder
|
||||||
* @author Spottedleaf
|
* @author Spottedleaf
|
||||||
|
|||||||
@@ -1,7 +1,6 @@
|
|||||||
package ca.spottedleaf.moonrise.mixin.chunk_system;
|
package ca.spottedleaf.moonrise.mixin.chunk_system;
|
||||||
|
|
||||||
import ca.spottedleaf.moonrise.patches.chunk_system.level.chunk.ChunkSystemLevelChunk;
|
import ca.spottedleaf.moonrise.patches.chunk_system.level.chunk.ChunkSystemLevelChunk;
|
||||||
import ca.spottedleaf.moonrise.patches.chunk_system.scheduling.NewChunkHolder;
|
|
||||||
import ca.spottedleaf.moonrise.patches.chunk_system.ticks.ChunkSystemLevelChunkTicks;
|
import ca.spottedleaf.moonrise.patches.chunk_system.ticks.ChunkSystemLevelChunkTicks;
|
||||||
import net.minecraft.core.Registry;
|
import net.minecraft.core.Registry;
|
||||||
import net.minecraft.server.level.ServerChunkCache;
|
import net.minecraft.server.level.ServerChunkCache;
|
||||||
@@ -49,7 +48,7 @@ abstract class LevelChunkMixin extends ChunkAccess implements ChunkSystemLevelCh
|
|||||||
private boolean postProcessingDone;
|
private boolean postProcessingDone;
|
||||||
|
|
||||||
@Unique
|
@Unique
|
||||||
private NewChunkHolder chunkAndHolder;
|
private ServerChunkCache.ChunkAndHolder chunkAndHolder;
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public final boolean moonrise$isPostProcessingDone() {
|
public final boolean moonrise$isPostProcessingDone() {
|
||||||
@@ -57,12 +56,12 @@ abstract class LevelChunkMixin extends ChunkAccess implements ChunkSystemLevelCh
|
|||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public final NewChunkHolder moonrise$getChunkHolder() {
|
public final ServerChunkCache.ChunkAndHolder moonrise$getChunkAndHolder() {
|
||||||
return this.chunkAndHolder;
|
return this.chunkAndHolder;
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public final void moonrise$setChunkHolder(final NewChunkHolder holder) {
|
public final void moonrise$setChunkAndHolder(final ServerChunkCache.ChunkAndHolder holder) {
|
||||||
this.chunkAndHolder = holder;
|
this.chunkAndHolder = holder;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -94,15 +93,11 @@ abstract class LevelChunkMixin extends ChunkAccess implements ChunkSystemLevelCh
|
|||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public boolean tryMarkSaved() {
|
public void setUnsaved(final boolean needsSaving) {
|
||||||
if (!this.isUnsaved()) {
|
if (!needsSaving) {
|
||||||
return false;
|
|
||||||
}
|
|
||||||
((ChunkSystemLevelChunkTicks)this.blockTicks).moonrise$clearDirty();
|
((ChunkSystemLevelChunkTicks)this.blockTicks).moonrise$clearDirty();
|
||||||
((ChunkSystemLevelChunkTicks)this.fluidTicks).moonrise$clearDirty();
|
((ChunkSystemLevelChunkTicks)this.fluidTicks).moonrise$clearDirty();
|
||||||
|
}
|
||||||
super.tryMarkSaved();
|
super.setUnsaved(needsSaving);
|
||||||
|
|
||||||
return true;
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -105,12 +105,12 @@ abstract class LevelChunkTicksMixin<T> implements ChunkSystemLevelChunkTicks, Se
|
|||||||
* @author Spottedleaf
|
* @author Spottedleaf
|
||||||
*/
|
*/
|
||||||
@Inject(
|
@Inject(
|
||||||
method = "pack",
|
method = "save(JLjava/util/function/Function;)Lnet/minecraft/nbt/ListTag;",
|
||||||
at = @At(
|
at = @At(
|
||||||
value = "HEAD"
|
value = "HEAD"
|
||||||
)
|
)
|
||||||
)
|
)
|
||||||
private void saveHook(final long time, final CallbackInfoReturnable<ListTag> cir) {
|
private void saveHook(final long time, final Function<T, String> idFunction, final CallbackInfoReturnable<ListTag> cir) {
|
||||||
this.lastSaved = time;
|
this.lastSaved = time;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -10,7 +10,6 @@ import ca.spottedleaf.moonrise.patches.chunk_system.level.entity.dfl.DefaultEnti
|
|||||||
import ca.spottedleaf.moonrise.patches.chunk_system.world.ChunkSystemEntityGetter;
|
import ca.spottedleaf.moonrise.patches.chunk_system.world.ChunkSystemEntityGetter;
|
||||||
import net.minecraft.core.BlockPos;
|
import net.minecraft.core.BlockPos;
|
||||||
import net.minecraft.server.level.FullChunkStatus;
|
import net.minecraft.server.level.FullChunkStatus;
|
||||||
import net.minecraft.util.profiling.Profiler;
|
|
||||||
import net.minecraft.util.profiling.ProfilerFiller;
|
import net.minecraft.util.profiling.ProfilerFiller;
|
||||||
import net.minecraft.world.entity.Entity;
|
import net.minecraft.world.entity.Entity;
|
||||||
import net.minecraft.world.entity.EntityType;
|
import net.minecraft.world.entity.EntityType;
|
||||||
@@ -38,6 +37,9 @@ import java.util.function.Predicate;
|
|||||||
@Mixin(Level.class)
|
@Mixin(Level.class)
|
||||||
abstract class LevelMixin implements ChunkSystemLevel, ChunkSystemEntityGetter, LevelAccessor, AutoCloseable {
|
abstract class LevelMixin implements ChunkSystemLevel, ChunkSystemEntityGetter, LevelAccessor, AutoCloseable {
|
||||||
|
|
||||||
|
@Shadow
|
||||||
|
public abstract ProfilerFiller getProfiler();
|
||||||
|
|
||||||
@Shadow
|
@Shadow
|
||||||
public abstract LevelChunk getChunk(int i, int j);
|
public abstract LevelChunk getChunk(int i, int j);
|
||||||
|
|
||||||
@@ -57,7 +59,7 @@ abstract class LevelMixin implements ChunkSystemLevel, ChunkSystemEntityGetter,
|
|||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public final void moonrise$setEntityLookup(final EntityLookup entityLookup) {
|
public void moonrise$setEntityLookup(final EntityLookup entityLookup) {
|
||||||
if (this.entityLookup != null && !(this.entityLookup instanceof DefaultEntityLookup)) {
|
if (this.entityLookup != null && !(this.entityLookup instanceof DefaultEntityLookup)) {
|
||||||
throw new IllegalStateException("Entity lookup already initialised");
|
throw new IllegalStateException("Entity lookup already initialised");
|
||||||
}
|
}
|
||||||
@@ -85,7 +87,7 @@ abstract class LevelMixin implements ChunkSystemLevel, ChunkSystemEntityGetter,
|
|||||||
@Overwrite
|
@Overwrite
|
||||||
@Override
|
@Override
|
||||||
public List<Entity> getEntities(final Entity entity, final AABB boundingBox, final Predicate<? super Entity> predicate) {
|
public List<Entity> getEntities(final Entity entity, final AABB boundingBox, final Predicate<? super Entity> predicate) {
|
||||||
Profiler.get().incrementCounter("getEntities");
|
this.getProfiler().incrementCounter("getEntities");
|
||||||
final List<Entity> ret = new ArrayList<>();
|
final List<Entity> ret = new ArrayList<>();
|
||||||
|
|
||||||
((ChunkSystemLevel)this).moonrise$getEntityLookup().getEntities(entity, boundingBox, ret, predicate);
|
((ChunkSystemLevel)this).moonrise$getEntityLookup().getEntities(entity, boundingBox, ret, predicate);
|
||||||
@@ -103,7 +105,7 @@ abstract class LevelMixin implements ChunkSystemLevel, ChunkSystemEntityGetter,
|
|||||||
public <T extends Entity> void getEntities(final EntityTypeTest<Entity, T> entityTypeTest,
|
public <T extends Entity> void getEntities(final EntityTypeTest<Entity, T> entityTypeTest,
|
||||||
final AABB boundingBox, final Predicate<? super T> predicate,
|
final AABB boundingBox, final Predicate<? super T> predicate,
|
||||||
final List<? super T> into, final int maxCount) {
|
final List<? super T> into, final int maxCount) {
|
||||||
Profiler.get().incrementCounter("getEntities");
|
this.getProfiler().incrementCounter("getEntities");
|
||||||
|
|
||||||
if (entityTypeTest instanceof EntityType<T> byType) {
|
if (entityTypeTest instanceof EntityType<T> byType) {
|
||||||
if (maxCount != Integer.MAX_VALUE) {
|
if (maxCount != Integer.MAX_VALUE) {
|
||||||
@@ -176,7 +178,7 @@ abstract class LevelMixin implements ChunkSystemLevel, ChunkSystemEntityGetter,
|
|||||||
*/
|
*/
|
||||||
@Override
|
@Override
|
||||||
public final <T extends Entity> List<T> getEntitiesOfClass(final Class<T> entityClass, final AABB boundingBox, final Predicate<? super T> predicate) {
|
public final <T extends Entity> List<T> getEntitiesOfClass(final Class<T> entityClass, final AABB boundingBox, final Predicate<? super T> predicate) {
|
||||||
Profiler.get().incrementCounter("getEntities");
|
this.getProfiler().incrementCounter("getEntities");
|
||||||
final List<T> ret = new ArrayList<>();
|
final List<T> ret = new ArrayList<>();
|
||||||
|
|
||||||
((ChunkSystemLevel)this).moonrise$getEntityLookup().getEntities(entityClass, null, boundingBox, ret, predicate);
|
((ChunkSystemLevel)this).moonrise$getEntityLookup().getEntities(entityClass, null, boundingBox, ret, predicate);
|
||||||
@@ -194,7 +196,7 @@ abstract class LevelMixin implements ChunkSystemLevel, ChunkSystemEntityGetter,
|
|||||||
*/
|
*/
|
||||||
@Override
|
@Override
|
||||||
public final List<Entity> moonrise$getHardCollidingEntities(final Entity entity, final AABB box, final Predicate<? super Entity> predicate) {
|
public final List<Entity> moonrise$getHardCollidingEntities(final Entity entity, final AABB box, final Predicate<? super Entity> predicate) {
|
||||||
Profiler.get().incrementCounter("getEntities");
|
this.getProfiler().incrementCounter("getEntities");
|
||||||
final List<Entity> ret = new ArrayList<>();
|
final List<Entity> ret = new ArrayList<>();
|
||||||
|
|
||||||
((ChunkSystemLevel)this).moonrise$getEntityLookup().getHardCollidingEntities(entity, box, ret, predicate);
|
((ChunkSystemLevel)this).moonrise$getEntityLookup().getHardCollidingEntities(entity, box, ret, predicate);
|
||||||
|
|||||||
@@ -190,11 +190,6 @@ abstract class MinecraftServerMixin extends ReentrantBlockableEventLoop<TickTask
|
|||||||
)
|
)
|
||||||
)
|
)
|
||||||
private boolean doNotWaitChunkSystemShutdown(final Stream<ServerLevel> instance, final Predicate<? super ServerLevel> predicate) {
|
private boolean doNotWaitChunkSystemShutdown(final Stream<ServerLevel> instance, final Predicate<? super ServerLevel> predicate) {
|
||||||
// note: make sure we call deactivateTicketsOnClosing
|
|
||||||
for (final ServerLevel world : this.getAllLevels()) {
|
|
||||||
world.getChunkSource().deactivateTicketsOnClosing();
|
|
||||||
}
|
|
||||||
|
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -243,9 +238,8 @@ abstract class MinecraftServerMixin extends ReentrantBlockableEventLoop<TickTask
|
|||||||
)
|
)
|
||||||
)
|
)
|
||||||
private void closeIOThreads(final CallbackInfo ci) {
|
private void closeIOThreads(final CallbackInfo ci) {
|
||||||
LOGGER.info("Waiting for all RegionFile I/O tasks to complete...");
|
LOGGER.info("Waiting for I/O tasks to complete...");
|
||||||
MoonriseRegionFileIO.flush((MinecraftServer)(Object)this);
|
MoonriseRegionFileIO.flush((MinecraftServer)(Object)this);
|
||||||
LOGGER.info("All RegionFile I/O tasks to complete");
|
|
||||||
if ((Object)this instanceof DedicatedServer) {
|
if ((Object)this instanceof DedicatedServer) {
|
||||||
MoonriseCommon.haltExecutors();
|
MoonriseCommon.haltExecutors();
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -43,17 +43,4 @@ abstract class PlayerListMixin {
|
|||||||
)
|
)
|
||||||
private void doNotAdjustVD(final PlayerList instance, final Packet<?> packet) {}
|
private void doNotAdjustVD(final PlayerList instance, final Packet<?> packet) {}
|
||||||
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @reason The RegionizedPlayerChunkLoader will handle the SD packet
|
|
||||||
* @author Spottedleaf
|
|
||||||
*/
|
|
||||||
@Redirect(
|
|
||||||
method = "setSimulationDistance",
|
|
||||||
at = @At(
|
|
||||||
value = "INVOKE",
|
|
||||||
target = "Lnet/minecraft/server/players/PlayerList;broadcastAll(Lnet/minecraft/network/protocol/Packet;)V"
|
|
||||||
)
|
|
||||||
)
|
|
||||||
private void doNotAdjustSD(final PlayerList instance, final Packet<?> packet) {}
|
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -36,7 +36,6 @@ import org.spongepowered.asm.mixin.injection.callback.CallbackInfo;
|
|||||||
import java.io.IOException;
|
import java.io.IOException;
|
||||||
import java.nio.file.Path;
|
import java.nio.file.Path;
|
||||||
import java.util.Optional;
|
import java.util.Optional;
|
||||||
import java.util.function.BiFunction;
|
|
||||||
import java.util.function.BooleanSupplier;
|
import java.util.function.BooleanSupplier;
|
||||||
import java.util.function.Function;
|
import java.util.function.Function;
|
||||||
import java.util.function.Predicate;
|
import java.util.function.Predicate;
|
||||||
@@ -45,14 +44,7 @@ import java.util.stream.Stream;
|
|||||||
@Mixin(PoiManager.class)
|
@Mixin(PoiManager.class)
|
||||||
// Declare the generic type as Object so that our Overrides match the method signature of the superclass
|
// Declare the generic type as Object so that our Overrides match the method signature of the superclass
|
||||||
// Specifically, getOrCreate must return Object so that existing invokes do not route to the superclass
|
// Specifically, getOrCreate must return Object so that existing invokes do not route to the superclass
|
||||||
public abstract class PoiManagerMixin extends SectionStorage<Object, Object> implements ChunkSystemPoiManager {
|
public abstract class PoiManagerMixin extends SectionStorage<Object> implements ChunkSystemPoiManager {
|
||||||
|
|
||||||
public PoiManagerMixin(final SimpleRegionStorage simpleRegionStorage, final Codec<Object> codec, final Function<Object, Object> function,
|
|
||||||
final BiFunction<Object, Runnable, Object> biFunction, final Function<Runnable, Object> function2,
|
|
||||||
final RegistryAccess registryAccess, final ChunkIOErrorReporter chunkIOErrorReporter,
|
|
||||||
final LevelHeightAccessor levelHeightAccessor) {
|
|
||||||
super(simpleRegionStorage, codec, function, biFunction, function2, registryAccess, chunkIOErrorReporter, levelHeightAccessor);
|
|
||||||
}
|
|
||||||
|
|
||||||
@Shadow
|
@Shadow
|
||||||
abstract boolean isVillageCenter(long l);
|
abstract boolean isVillageCenter(long l);
|
||||||
@@ -60,6 +52,10 @@ public abstract class PoiManagerMixin extends SectionStorage<Object, Object> imp
|
|||||||
@Shadow
|
@Shadow
|
||||||
public abstract void checkConsistencyWithBlocks(SectionPos sectionPos, LevelChunkSection levelChunkSection);
|
public abstract void checkConsistencyWithBlocks(SectionPos sectionPos, LevelChunkSection levelChunkSection);
|
||||||
|
|
||||||
|
public PoiManagerMixin(SimpleRegionStorage simpleRegionStorage, Function<Runnable, Codec<Object>> function, Function<Runnable, Object> function2, RegistryAccess registryAccess, ChunkIOErrorReporter chunkIOErrorReporter, LevelHeightAccessor levelHeightAccessor) {
|
||||||
|
super(simpleRegionStorage, function, function2, registryAccess, chunkIOErrorReporter, levelHeightAccessor);
|
||||||
|
}
|
||||||
|
|
||||||
@Unique
|
@Unique
|
||||||
private ServerLevel world;
|
private ServerLevel world;
|
||||||
|
|
||||||
@@ -155,8 +151,8 @@ public abstract class PoiManagerMixin extends SectionStorage<Object, Object> imp
|
|||||||
|
|
||||||
TickThread.ensureTickThread(this.world, chunkX, chunkZ, "Accessing poi chunk off-main");
|
TickThread.ensureTickThread(this.world, chunkX, chunkZ, "Accessing poi chunk off-main");
|
||||||
|
|
||||||
final PoiChunk ret = ((ChunkSystemServerLevel)this.world).moonrise$getChunkTaskScheduler().chunkHolderManager
|
final ChunkHolderManager manager = ((ChunkSystemServerLevel)this.world).moonrise$getChunkTaskScheduler().chunkHolderManager;
|
||||||
.getPoiChunkIfLoaded(chunkX, chunkZ, true);
|
final PoiChunk ret = manager.getPoiChunkIfLoaded(chunkX, chunkZ, true);
|
||||||
|
|
||||||
return ret == null ? Optional.empty() : (Optional)ret.getSectionForVanilla(chunkY);
|
return ret == null ? Optional.empty() : (Optional)ret.getSectionForVanilla(chunkY);
|
||||||
}
|
}
|
||||||
@@ -210,13 +206,9 @@ public abstract class PoiManagerMixin extends SectionStorage<Object, Object> imp
|
|||||||
public final void moonrise$onUnload(final long coordinate) { // Paper - rewrite chunk system
|
public final void moonrise$onUnload(final long coordinate) { // Paper - rewrite chunk system
|
||||||
final int chunkX = CoordinateUtils.getChunkX(coordinate);
|
final int chunkX = CoordinateUtils.getChunkX(coordinate);
|
||||||
final int chunkZ = CoordinateUtils.getChunkZ(coordinate);
|
final int chunkZ = CoordinateUtils.getChunkZ(coordinate);
|
||||||
|
|
||||||
final int minY = WorldUtil.getMinSection(this.world);
|
|
||||||
final int maxY = WorldUtil.getMaxSection(this.world);
|
|
||||||
|
|
||||||
TickThread.ensureTickThread(this.world, chunkX, chunkZ, "Unloading poi chunk off-main");
|
TickThread.ensureTickThread(this.world, chunkX, chunkZ, "Unloading poi chunk off-main");
|
||||||
for (int sectionY = minY; sectionY <= maxY; ++sectionY) {
|
for (int section = this.levelHeightAccessor.getMinSection(); section < this.levelHeightAccessor.getMaxSection(); ++section) {
|
||||||
final long sectionPos = SectionPos.asLong(chunkX, sectionY, chunkZ);
|
final long sectionPos = SectionPos.asLong(chunkX, section, chunkZ);
|
||||||
this.updateDistanceTracking(sectionPos);
|
this.updateDistanceTracking(sectionPos);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -225,12 +217,8 @@ public abstract class PoiManagerMixin extends SectionStorage<Object, Object> imp
|
|||||||
public final void moonrise$loadInPoiChunk(final PoiChunk poiChunk) {
|
public final void moonrise$loadInPoiChunk(final PoiChunk poiChunk) {
|
||||||
final int chunkX = poiChunk.chunkX;
|
final int chunkX = poiChunk.chunkX;
|
||||||
final int chunkZ = poiChunk.chunkZ;
|
final int chunkZ = poiChunk.chunkZ;
|
||||||
|
|
||||||
final int minY = WorldUtil.getMinSection(this.world);
|
|
||||||
final int maxY = WorldUtil.getMaxSection(this.world);
|
|
||||||
|
|
||||||
TickThread.ensureTickThread(this.world, chunkX, chunkZ, "Loading poi chunk off-main");
|
TickThread.ensureTickThread(this.world, chunkX, chunkZ, "Loading poi chunk off-main");
|
||||||
for (int sectionY = minY; sectionY <= maxY; ++sectionY) {
|
for (int sectionY = this.levelHeightAccessor.getMinSection(); sectionY < this.levelHeightAccessor.getMaxSection(); ++sectionY) {
|
||||||
final PoiSection section = poiChunk.getSection(sectionY);
|
final PoiSection section = poiChunk.getSection(sectionY);
|
||||||
if (section != null && !((ChunkSystemPoiSection)section).moonrise$isEmpty()) {
|
if (section != null && !((ChunkSystemPoiSection)section).moonrise$isEmpty()) {
|
||||||
this.onSectionLoad(SectionPos.asLong(chunkX, sectionY, chunkZ));
|
this.onSectionLoad(SectionPos.asLong(chunkX, sectionY, chunkZ));
|
||||||
@@ -266,4 +254,20 @@ public abstract class PoiManagerMixin extends SectionStorage<Object, Object> imp
|
|||||||
private <T> Stream<T> skipLoadedSet(final Stream<T> instance, final Predicate<? super T> predicate) {
|
private <T> Stream<T> skipLoadedSet(final Stream<T> instance, final Predicate<? super T> predicate) {
|
||||||
return instance;
|
return instance;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public final void moonrise$close() throws IOException {}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public final CompoundTag moonrise$read(final int chunkX, final int chunkZ) throws IOException {
|
||||||
|
return MoonriseRegionFileIO.loadData(
|
||||||
|
this.world, chunkX, chunkZ, MoonriseRegionFileIO.RegionFileType.POI_DATA,
|
||||||
|
MoonriseRegionFileIO.getIOBlockingPriorityForCurrentThread()
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public final void moonrise$write(final int chunkX, final int chunkZ, final CompoundTag data) throws IOException {
|
||||||
|
MoonriseRegionFileIO.scheduleSave(this.world, chunkX, chunkZ, data, MoonriseRegionFileIO.RegionFileType.POI_DATA);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -30,7 +30,7 @@ abstract class PoiSectionMixin implements ChunkSystemPoiSection {
|
|||||||
|
|
||||||
|
|
||||||
@Unique
|
@Unique
|
||||||
private final Optional<PoiSection> noAllocOptional = Optional.of((PoiSection)(Object)this);
|
private final Optional<PoiSection> noAllocOptional = Optional.of((PoiSection)(Object)this);;
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public final boolean moonrise$isEmpty() {
|
public final boolean moonrise$isEmpty() {
|
||||||
|
|||||||
@@ -2,6 +2,8 @@ package ca.spottedleaf.moonrise.mixin.chunk_system;
|
|||||||
|
|
||||||
import ca.spottedleaf.moonrise.patches.chunk_system.level.storage.ChunkSystemSectionStorage;
|
import ca.spottedleaf.moonrise.patches.chunk_system.level.storage.ChunkSystemSectionStorage;
|
||||||
import net.minecraft.nbt.CompoundTag;
|
import net.minecraft.nbt.CompoundTag;
|
||||||
|
import net.minecraft.nbt.Tag;
|
||||||
|
import net.minecraft.resources.RegistryOps;
|
||||||
import net.minecraft.world.level.ChunkPos;
|
import net.minecraft.world.level.ChunkPos;
|
||||||
import net.minecraft.world.level.chunk.storage.RegionFileStorage;
|
import net.minecraft.world.level.chunk.storage.RegionFileStorage;
|
||||||
import net.minecraft.world.level.chunk.storage.SectionStorage;
|
import net.minecraft.world.level.chunk.storage.SectionStorage;
|
||||||
@@ -21,11 +23,15 @@ import java.util.Optional;
|
|||||||
import java.util.concurrent.CompletableFuture;
|
import java.util.concurrent.CompletableFuture;
|
||||||
|
|
||||||
@Mixin(SectionStorage.class)
|
@Mixin(SectionStorage.class)
|
||||||
abstract class SectionStorageMixin<R, P> implements ChunkSystemSectionStorage, AutoCloseable {
|
abstract class SectionStorageMixin implements ChunkSystemSectionStorage, AutoCloseable {
|
||||||
|
|
||||||
@Shadow
|
@Shadow
|
||||||
private SimpleRegionStorage simpleRegionStorage;
|
private SimpleRegionStorage simpleRegionStorage;
|
||||||
|
|
||||||
|
@Shadow
|
||||||
|
@Final
|
||||||
|
private static Logger LOGGER;
|
||||||
|
|
||||||
|
|
||||||
@Unique
|
@Unique
|
||||||
private RegionFileStorage storage;
|
private RegionFileStorage storage;
|
||||||
@@ -35,9 +41,6 @@ abstract class SectionStorageMixin<R, P> implements ChunkSystemSectionStorage, A
|
|||||||
return this.storage;
|
return this.storage;
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
|
||||||
public void moonrise$close() throws IOException {}
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @reason Retrieve storage from IOWorker, and then nuke it
|
* @reason Retrieve storage from IOWorker, and then nuke it
|
||||||
* @author Spottedleaf
|
* @author Spottedleaf
|
||||||
@@ -58,8 +61,12 @@ abstract class SectionStorageMixin<R, P> implements ChunkSystemSectionStorage, A
|
|||||||
* @author Spottedleaf
|
* @author Spottedleaf
|
||||||
*/
|
*/
|
||||||
@Overwrite
|
@Overwrite
|
||||||
public final CompletableFuture<Optional<SectionStorage.PackedChunk<P>>> tryRead(final ChunkPos pos) {
|
public final CompletableFuture<Optional<CompoundTag>> tryRead(final ChunkPos pos) {
|
||||||
throw new IllegalStateException("Only chunk system can write state, offending class:" + this.getClass().getName());
|
try {
|
||||||
|
return CompletableFuture.completedFuture(Optional.ofNullable(this.moonrise$read(pos.x, pos.z)));
|
||||||
|
} catch (final Throwable thr) {
|
||||||
|
return CompletableFuture.failedFuture(thr);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -67,17 +74,29 @@ abstract class SectionStorageMixin<R, P> implements ChunkSystemSectionStorage, A
|
|||||||
* @author Spottedleaf
|
* @author Spottedleaf
|
||||||
*/
|
*/
|
||||||
@Overwrite
|
@Overwrite
|
||||||
public void unpackChunk(final ChunkPos chunkPos, final SectionStorage.PackedChunk<P> packedChunk) {
|
public void readColumn(final ChunkPos pos, final RegistryOps<Tag> ops, final CompoundTag data) {
|
||||||
throw new IllegalStateException("Only chunk system can load in state, offending class:" + this.getClass().getName());
|
throw new IllegalStateException("Only chunk system can load in state, offending class:" + this.getClass().getName());
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @reason Destroy old chunk system hook
|
* @reason Route to new chunk system hook
|
||||||
* @author Spottedleaf
|
* @author Spottedleaf
|
||||||
*/
|
*/
|
||||||
@Overwrite
|
@Redirect(
|
||||||
private void writeChunk(final ChunkPos chunkPos) {
|
method = "writeColumn(Lnet/minecraft/world/level/ChunkPos;)V",
|
||||||
throw new IllegalStateException("Only chunk system can write state, offending class:" + this.getClass().getName());
|
at = @At(
|
||||||
|
value = "INVOKE",
|
||||||
|
target = "Lnet/minecraft/world/level/chunk/storage/SimpleRegionStorage;write(Lnet/minecraft/world/level/ChunkPos;Lnet/minecraft/nbt/CompoundTag;)Ljava/util/concurrent/CompletableFuture;"
|
||||||
|
)
|
||||||
|
)
|
||||||
|
private CompletableFuture<Void> redirectWrite(final SimpleRegionStorage instance, final ChunkPos pos,
|
||||||
|
final CompoundTag tag) {
|
||||||
|
try {
|
||||||
|
this.moonrise$write(pos.x, pos.z, tag);
|
||||||
|
} catch (final IOException ex) {
|
||||||
|
LOGGER.error("Error writing poi chunk data to disk for chunk " + pos, ex);
|
||||||
|
}
|
||||||
|
return null;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
|||||||
@@ -3,17 +3,18 @@ package ca.spottedleaf.moonrise.mixin.chunk_system;
|
|||||||
import ca.spottedleaf.concurrentutil.map.ConcurrentLong2ReferenceChainedHashTable;
|
import ca.spottedleaf.concurrentutil.map.ConcurrentLong2ReferenceChainedHashTable;
|
||||||
import ca.spottedleaf.concurrentutil.util.Priority;
|
import ca.spottedleaf.concurrentutil.util.Priority;
|
||||||
import ca.spottedleaf.moonrise.common.PlatformHooks;
|
import ca.spottedleaf.moonrise.common.PlatformHooks;
|
||||||
|
import ca.spottedleaf.moonrise.common.list.ReferenceList;
|
||||||
import ca.spottedleaf.moonrise.common.util.CoordinateUtils;
|
import ca.spottedleaf.moonrise.common.util.CoordinateUtils;
|
||||||
import ca.spottedleaf.moonrise.common.util.TickThread;
|
import ca.spottedleaf.moonrise.common.util.TickThread;
|
||||||
import ca.spottedleaf.moonrise.patches.chunk_system.level.ChunkSystemServerLevel;
|
import ca.spottedleaf.moonrise.patches.chunk_system.level.ChunkSystemServerLevel;
|
||||||
import ca.spottedleaf.moonrise.patches.chunk_system.scheduling.ChunkHolderManager;
|
import ca.spottedleaf.moonrise.patches.chunk_system.scheduling.ChunkHolderManager;
|
||||||
import ca.spottedleaf.moonrise.patches.chunk_system.scheduling.ChunkTaskScheduler;
|
import ca.spottedleaf.moonrise.patches.chunk_system.scheduling.ChunkTaskScheduler;
|
||||||
import ca.spottedleaf.moonrise.patches.chunk_system.scheduling.NewChunkHolder;
|
import ca.spottedleaf.moonrise.patches.chunk_system.scheduling.NewChunkHolder;
|
||||||
|
import ca.spottedleaf.moonrise.patches.chunk_system.server.ChunkSystemMinecraftServer;
|
||||||
import ca.spottedleaf.moonrise.patches.chunk_system.world.ChunkSystemServerChunkCache;
|
import ca.spottedleaf.moonrise.patches.chunk_system.world.ChunkSystemServerChunkCache;
|
||||||
import net.minecraft.server.level.ChunkHolder;
|
import net.minecraft.server.level.ChunkHolder;
|
||||||
import net.minecraft.server.level.ChunkLevel;
|
import net.minecraft.server.level.ChunkLevel;
|
||||||
import net.minecraft.server.level.ChunkResult;
|
import net.minecraft.server.level.ChunkResult;
|
||||||
import net.minecraft.server.level.DistanceManager;
|
|
||||||
import net.minecraft.server.level.FullChunkStatus;
|
import net.minecraft.server.level.FullChunkStatus;
|
||||||
import net.minecraft.server.level.ServerChunkCache;
|
import net.minecraft.server.level.ServerChunkCache;
|
||||||
import net.minecraft.server.level.ServerLevel;
|
import net.minecraft.server.level.ServerLevel;
|
||||||
@@ -23,7 +24,6 @@ import net.minecraft.world.level.chunk.ChunkSource;
|
|||||||
import net.minecraft.world.level.chunk.LevelChunk;
|
import net.minecraft.world.level.chunk.LevelChunk;
|
||||||
import net.minecraft.world.level.chunk.LightChunk;
|
import net.minecraft.world.level.chunk.LightChunk;
|
||||||
import net.minecraft.world.level.chunk.status.ChunkStatus;
|
import net.minecraft.world.level.chunk.status.ChunkStatus;
|
||||||
import net.minecraft.world.level.storage.DimensionDataStorage;
|
|
||||||
import org.spongepowered.asm.mixin.Final;
|
import org.spongepowered.asm.mixin.Final;
|
||||||
import org.spongepowered.asm.mixin.Mixin;
|
import org.spongepowered.asm.mixin.Mixin;
|
||||||
import org.spongepowered.asm.mixin.Overwrite;
|
import org.spongepowered.asm.mixin.Overwrite;
|
||||||
@@ -34,6 +34,8 @@ import org.spongepowered.asm.mixin.injection.Inject;
|
|||||||
import org.spongepowered.asm.mixin.injection.Redirect;
|
import org.spongepowered.asm.mixin.injection.Redirect;
|
||||||
import org.spongepowered.asm.mixin.injection.callback.CallbackInfo;
|
import org.spongepowered.asm.mixin.injection.callback.CallbackInfo;
|
||||||
import java.io.IOException;
|
import java.io.IOException;
|
||||||
|
import java.util.List;
|
||||||
|
import java.util.Objects;
|
||||||
import java.util.concurrent.CompletableFuture;
|
import java.util.concurrent.CompletableFuture;
|
||||||
import java.util.function.Consumer;
|
import java.util.function.Consumer;
|
||||||
|
|
||||||
@@ -49,13 +51,12 @@ abstract class ServerChunkCacheMixin extends ChunkSource implements ChunkSystemS
|
|||||||
@Final
|
@Final
|
||||||
public ServerLevel level;
|
public ServerLevel level;
|
||||||
|
|
||||||
@Shadow
|
|
||||||
@Final
|
|
||||||
private DimensionDataStorage dataStorage;
|
|
||||||
|
|
||||||
@Unique
|
@Unique
|
||||||
private final ConcurrentLong2ReferenceChainedHashTable<LevelChunk> fullChunks = new ConcurrentLong2ReferenceChainedHashTable<>();
|
private final ConcurrentLong2ReferenceChainedHashTable<LevelChunk> fullChunks = new ConcurrentLong2ReferenceChainedHashTable<>();
|
||||||
|
|
||||||
|
@Unique
|
||||||
|
private long chunksTicked;
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public final void moonrise$setFullChunk(final int chunkX, final int chunkZ, final LevelChunk chunk) {
|
public final void moonrise$setFullChunk(final int chunkX, final int chunkZ, final LevelChunk chunk) {
|
||||||
final long key = CoordinateUtils.getChunkKey(chunkX, chunkZ);
|
final long key = CoordinateUtils.getChunkKey(chunkX, chunkZ);
|
||||||
@@ -80,13 +81,6 @@ abstract class ServerChunkCacheMixin extends ChunkSource implements ChunkSystemS
|
|||||||
completable::complete
|
completable::complete
|
||||||
);
|
);
|
||||||
|
|
||||||
if (!completable.isDone() && chunkTaskScheduler.hasShutdown()) {
|
|
||||||
throw new IllegalStateException(
|
|
||||||
"Chunk system has shut down, cannot process chunk requests in world '" + ca.spottedleaf.moonrise.common.util.WorldUtil.getWorldName(this.level) + "' at "
|
|
||||||
+ "(" + chunkX + "," + chunkZ + ") status: " + toStatus
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (TickThread.isTickThreadFor(this.level, chunkX, chunkZ)) {
|
if (TickThread.isTickThreadFor(this.level, chunkX, chunkZ)) {
|
||||||
ChunkTaskScheduler.pushChunkWait(this.level, chunkX, chunkZ);
|
ChunkTaskScheduler.pushChunkWait(this.level, chunkX, chunkZ);
|
||||||
this.mainThreadProcessor.managedBlock(completable::isDone);
|
this.mainThreadProcessor.managedBlock(completable::isDone);
|
||||||
@@ -267,7 +261,6 @@ abstract class ServerChunkCacheMixin extends ChunkSource implements ChunkSystemS
|
|||||||
@Override
|
@Override
|
||||||
@Overwrite
|
@Overwrite
|
||||||
public void close() throws IOException {
|
public void close() throws IOException {
|
||||||
this.dataStorage.close();
|
|
||||||
((ChunkSystemServerLevel)this.level).moonrise$getChunkTaskScheduler().chunkHolderManager.close(true, true);
|
((ChunkSystemServerLevel)this.level).moonrise$getChunkTaskScheduler().chunkHolderManager.close(true, true);
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -316,51 +309,86 @@ abstract class ServerChunkCacheMixin extends ChunkSource implements ChunkSystemS
|
|||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @reason We need to use {@link ChunkHolder#getChunkToSend()} as the new chunk system will not bring every chunk
|
* @reason Perform mid-tick chunk task processing during chunk tick
|
||||||
* sent to players up to block ticking.
|
|
||||||
* @author Spottedleaf
|
* @author Spottedleaf
|
||||||
*/
|
*/
|
||||||
@Redirect(
|
@Inject(
|
||||||
method = "broadcastChangedChunks",
|
method = "tickChunks",
|
||||||
at = @At(
|
at = @At(
|
||||||
value = "INVOKE",
|
value = "INVOKE",
|
||||||
target = "Lnet/minecraft/server/level/ChunkHolder;getTickingChunk()Lnet/minecraft/world/level/chunk/LevelChunk;")
|
shift = At.Shift.AFTER,
|
||||||
|
target = "Lnet/minecraft/server/level/ServerLevel;tickChunk(Lnet/minecraft/world/level/chunk/LevelChunk;I)V"
|
||||||
)
|
)
|
||||||
private LevelChunk redirectTickingChunk(final ChunkHolder instance) {
|
)
|
||||||
return instance.getChunkToSend();
|
private void midTickChunks(final CallbackInfo ci) {
|
||||||
|
if ((++this.chunksTicked & 7L) != 0L) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
((ChunkSystemMinecraftServer)this.level.getServer()).moonrise$executeMidTickTasks();
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @reason In the chunk system, spawn chunks will return only entity ticking chunks - we can elide the
|
* @reason In the chunk system, ticking chunks always have loaded entities. Of course, they are also always
|
||||||
* entity ticking range check.
|
* marked to be as ticking as well.
|
||||||
* @author Spottedleaf
|
* @author Spottedleaf
|
||||||
*/
|
*/
|
||||||
@Redirect(
|
@Redirect(
|
||||||
method = "tickSpawningChunk",
|
method = "tickChunks",
|
||||||
at = @At(
|
at = @At(
|
||||||
value = "INVOKE",
|
value = "INVOKE",
|
||||||
target = "Lnet/minecraft/server/level/DistanceManager;inEntityTickingRange(J)Z"
|
target = "Lnet/minecraft/server/level/ServerLevel;isNaturalSpawningAllowed(Lnet/minecraft/world/level/ChunkPos;)Z"
|
||||||
)
|
)
|
||||||
)
|
)
|
||||||
private boolean shortTickThunder(final DistanceManager instance, final long pos) {
|
private boolean shortNaturalSpawning(final ServerLevel instance, final ChunkPos chunkPos) {
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @reason In the chunk system, spawn chunks will return only entity ticking chunks - we can elide the
|
* @reason In the chunk system, ticking chunks always have loaded entities. Of course, they are also always
|
||||||
* entity ticking check.
|
* marked to be as ticking as well.
|
||||||
* @author Spottedleaf
|
* @author Spottedleaf
|
||||||
*/
|
*/
|
||||||
@Redirect(
|
@Redirect(
|
||||||
method = "tickSpawningChunk",
|
method = "tickChunks",
|
||||||
at = @At(
|
at = @At(
|
||||||
value = "INVOKE",
|
value = "INVOKE",
|
||||||
target = "Lnet/minecraft/server/level/ServerLevel;canSpawnEntitiesInChunk(Lnet/minecraft/world/level/ChunkPos;)Z"
|
target = "Lnet/minecraft/server/level/ServerLevel;shouldTickBlocksAt(J)Z"
|
||||||
)
|
)
|
||||||
)
|
)
|
||||||
private boolean onlyCheckWBForSpawning(final ServerLevel instance, final ChunkPos pos) {
|
private boolean shortShouldTickBlocks(final ServerLevel instance, final long pos) {
|
||||||
return instance.getWorldBorder().isWithinBounds(pos);
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @reason Since chunks in non-simulation range are only brought up to FULL status, not TICKING,
|
||||||
|
* those chunks may not be present in the ticking list and as a result we need to use our own list
|
||||||
|
* to ensure these chunks broadcast changes
|
||||||
|
* @author Spottedleaf
|
||||||
|
*/
|
||||||
|
@Redirect(
|
||||||
|
method = "tickChunks",
|
||||||
|
at = @At(
|
||||||
|
value = "INVOKE",
|
||||||
|
target = "Ljava/util/List;forEach(Ljava/util/function/Consumer;)V"
|
||||||
|
)
|
||||||
|
)
|
||||||
|
private void fixBroadcastChanges(final List<ServerChunkCache.ChunkAndHolder> instance,
|
||||||
|
final Consumer<ServerChunkCache.ChunkAndHolder> consumer) {
|
||||||
|
final ReferenceList<ChunkHolder> unsyncedChunks = ((ChunkSystemServerLevel)this.level).moonrise$getUnsyncedChunks();
|
||||||
|
final ChunkHolder[] chunkHolders = unsyncedChunks.getRawDataUnchecked();
|
||||||
|
final int totalUnsyncedChunks = unsyncedChunks.size();
|
||||||
|
|
||||||
|
Objects.checkFromToIndex(0, totalUnsyncedChunks, chunkHolders.length);
|
||||||
|
for (int i = 0; i < totalUnsyncedChunks; ++i) {
|
||||||
|
final ChunkHolder chunkHolder = chunkHolders[i];
|
||||||
|
final LevelChunk chunk = chunkHolder.getChunkToSend();
|
||||||
|
if (chunk != null) {
|
||||||
|
chunkHolder.broadcastChanges(chunk);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
((ChunkSystemServerLevel)this.level).moonrise$clearUnsyncedChunks();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
Some files were not shown because too many files have changed in this diff Show More
Reference in New Issue
Block a user