aboutsummaryrefslogtreecommitdiff
path: root/src/main/java/gregtech/api/threads/GT_Runnable_MachineBlockUpdate.java
diff options
context:
space:
mode:
Diffstat (limited to 'src/main/java/gregtech/api/threads/GT_Runnable_MachineBlockUpdate.java')
-rw-r--r--src/main/java/gregtech/api/threads/GT_Runnable_MachineBlockUpdate.java152
1 files changed, 126 insertions, 26 deletions
diff --git a/src/main/java/gregtech/api/threads/GT_Runnable_MachineBlockUpdate.java b/src/main/java/gregtech/api/threads/GT_Runnable_MachineBlockUpdate.java
index 92944d3e32..3d28a94de9 100644
--- a/src/main/java/gregtech/api/threads/GT_Runnable_MachineBlockUpdate.java
+++ b/src/main/java/gregtech/api/threads/GT_Runnable_MachineBlockUpdate.java
@@ -1,43 +1,143 @@
package gregtech.api.threads;
+import gregtech.GT_Mod;
import gregtech.api.GregTech_API;
import gregtech.api.interfaces.tileentity.IMachineBlockUpdateable;
import net.minecraft.tileentity.TileEntity;
import net.minecraft.world.ChunkPosition;
import net.minecraft.world.World;
-import java.util.ArrayList;
+import java.util.HashSet;
+import java.util.Set;
+import java.util.concurrent.ExecutorService;
+import java.util.concurrent.Executors;
+import java.util.concurrent.ThreadFactory;
+import java.util.concurrent.TimeUnit;
public class GT_Runnable_MachineBlockUpdate implements Runnable {
- private final int mX, mY, mZ;
- private final World mWorld;
-
- public GT_Runnable_MachineBlockUpdate(World aWorld, int aX, int aY, int aZ) {
- mWorld = aWorld;
- mX = aX;
- mY = aY;
- mZ = aZ;
- }
-
- private static void stepToUpdateMachine(World aWorld, int aX, int aY, int aZ, ArrayList<ChunkPosition> aList) {
- aList.add(new ChunkPosition(aX, aY, aZ));
- TileEntity tTileEntity = aWorld.getTileEntity(aX, aY, aZ);
- if (tTileEntity instanceof IMachineBlockUpdateable)
- ((IMachineBlockUpdateable) tTileEntity).onMachineBlockUpdate();
- if (aList.size() < 5 || (tTileEntity instanceof IMachineBlockUpdateable) || GregTech_API.isMachineBlock(aWorld.getBlock(aX, aY, aZ), aWorld.getBlockMetadata(aX, aY, aZ))) {
- if (!aList.contains(new ChunkPosition(aX + 1, aY, aZ))) stepToUpdateMachine(aWorld, aX + 1, aY, aZ, aList);
- if (!aList.contains(new ChunkPosition(aX - 1, aY, aZ))) stepToUpdateMachine(aWorld, aX - 1, aY, aZ, aList);
- if (!aList.contains(new ChunkPosition(aX, aY + 1, aZ))) stepToUpdateMachine(aWorld, aX, aY + 1, aZ, aList);
- if (!aList.contains(new ChunkPosition(aX, aY - 1, aZ))) stepToUpdateMachine(aWorld, aX, aY - 1, aZ, aList);
- if (!aList.contains(new ChunkPosition(aX, aY, aZ + 1))) stepToUpdateMachine(aWorld, aX, aY, aZ + 1, aList);
- if (!aList.contains(new ChunkPosition(aX, aY, aZ - 1))) stepToUpdateMachine(aWorld, aX, aY, aZ - 1, aList);
+ //used by runner thread
+ private final int x, y, z;
+ private final World world;
+ private final Set<ChunkPosition> visited = new HashSet<>(80);
+
+ //Threading
+ private static final ThreadFactory THREAD_FACTORY = r -> {
+ Thread thread = new Thread(r);
+ thread.setName("GT_MachineBlockUpdate");
+ return thread;
+ };
+ private static ExecutorService EXECUTOR_SERVICE;
+
+ //This class should never be initiated outside of this class!
+ private GT_Runnable_MachineBlockUpdate(World aWorld, int aX, int aY, int aZ) {
+ this.world = aWorld;
+ this.x = aX;
+ this.y = aY;
+ this.z = aZ;
+ }
+
+ public static void setMachineUpdateValues(World aWorld, int aX, int aY, int aZ) {
+ EXECUTOR_SERVICE.submit(new GT_Runnable_MachineBlockUpdate(aWorld, aX, aY, aZ));
+ }
+
+ public static void initExecutorService() {
+ EXECUTOR_SERVICE = Executors.newFixedThreadPool(Runtime.getRuntime().availableProcessors(), THREAD_FACTORY);
+ //Executors.newSingleThreadExecutor(THREAD_FACTORY);
+ //Executors.newCachedThreadPool(THREAD_FACTORY);
+ //Executors.newFixedThreadPool(Runtime.getRuntime().availableProcessors(),THREAD_FACTORY);
+ }
+
+ public static void shutdownExecutorService() {
+ try {
+ GT_Mod.GT_FML_LOGGER.info("Shutting down Machine block update executor service");
+ EXECUTOR_SERVICE.shutdown(); // Disable new tasks from being submitted
+ // Wait a while for existing tasks to terminate
+ if (!EXECUTOR_SERVICE.awaitTermination(60, TimeUnit.SECONDS)) {
+ EXECUTOR_SERVICE.shutdownNow(); // Cancel currently executing tasks
+ // Wait a while for tasks to respond to being cancelled
+ if (!EXECUTOR_SERVICE.awaitTermination(60, TimeUnit.SECONDS)) {
+ GT_Mod.GT_FML_LOGGER.error("Well this didn't terminated well... GT_Runnable_MachineBlockUpdate.shutdownExecutorService");
+ }
+ }
+ } catch (InterruptedException ie) {
+ GT_Mod.GT_FML_LOGGER.error("Well this interruption got interrupted...", ie);
+ // (Re-)Cancel if current thread also interrupted
+ EXECUTOR_SERVICE.shutdownNow();
+ // Preserve interrupt status
+ Thread.currentThread().interrupt();
+ }catch (Exception e){
+ GT_Mod.GT_FML_LOGGER.error("Well this didn't terminated well...",e);
+ // (Re-)Cancel in case
+ EXECUTOR_SERVICE.shutdownNow();
+ }finally {
+ GT_Mod.GT_FML_LOGGER.info("Leaving... GT_Runnable_MachineBlockUpdate.shutdownExecutorService");
+ }
+ }
+
+ private boolean shouldRecurse(TileEntity aTileEntity, int aX, int aY, int aZ) {
+ //no check on IGregTechTileEntity as it should call the underlying meta tile isMachineBlockUpdateRecursive
+ //if (aTileEntity instanceof IGregTechTileEntity) {
+ // return ((IGregTechTileEntity) aTileEntity).isMachineBlockUpdateRecursive();
+ //}
+ return (aTileEntity instanceof IMachineBlockUpdateable &&
+ ((IMachineBlockUpdateable) aTileEntity).isMachineBlockUpdateRecursive()) ||
+ GregTech_API.isMachineBlock(world.getBlock(aX, aY, aZ), world.getBlockMetadata(aX, aY, aZ));
+ }
+
+ private void causeUpdate(TileEntity tileEntity) {
+ //no check for IGregTechTileEntity as it should call the underlying meta tile onMachineBlockUpdate
+ if (tileEntity instanceof IMachineBlockUpdateable) {
+ ((IMachineBlockUpdateable) tileEntity).onMachineBlockUpdate();
+ }
+ }
+
+ private void stepToUpdateMachine(int aX, int aY, int aZ) {
+ if (!visited.add(new ChunkPosition(aX, aY, aZ)))
+ return;
+ TileEntity tTileEntity = world.getTileEntity(aX, aY, aZ);
+
+ causeUpdate(tTileEntity);
+
+ if (visited.size() < 5 || shouldRecurse(tTileEntity, aX, aY, aZ)) {
+ stepToUpdateMachine(aX + 1, aY, aZ);
+ stepToUpdateMachine(aX - 1, aY, aZ);
+ stepToUpdateMachine(aX, aY + 1, aZ);
+ stepToUpdateMachine(aX, aY - 1, aZ);
+ stepToUpdateMachine(aX, aY, aZ + 1);
+ stepToUpdateMachine(aX, aY, aZ - 1);
}
}
@Override
public void run() {
try {
- stepToUpdateMachine(mWorld, mX, mY, mZ, new ArrayList<ChunkPosition>());
- } catch (Throwable e) {/**/}
+ stepToUpdateMachine(x, y, z);
+ } catch (Exception e) {
+ GT_Mod.GT_FML_LOGGER.error("Well this update was broken... " + new Coordinates(x, y, z, world), e);
+ }
+ }
+
+ public static class Coordinates {
+ public final int mX;
+ public final int mY;
+ public final int mZ;
+ public final World mWorld;
+
+ public Coordinates(int mX, int mY, int mZ, World mWorld) {
+ this.mX = mX;
+ this.mY = mY;
+ this.mZ = mZ;
+ this.mWorld = mWorld;
+ }
+
+ @Override
+ public String toString() {
+ return "Coordinates{" +
+ "mX=" + mX +
+ ", mY=" + mY +
+ ", mZ=" + mZ +
+ ", mWorld=" + mWorld.getProviderName() + " @dimId " + mWorld.provider.dimensionId +
+ '}';
+ }
}
-} \ No newline at end of file
+}