-
Notifications
You must be signed in to change notification settings - Fork 35
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
- Loading branch information
Showing
9 changed files
with
135 additions
and
150 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file was deleted.
Oops, something went wrong.
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
38 changes: 0 additions & 38 deletions
38
junit/src/main/java/net/neoforged/fml/junit/LaunchWrapper.java
This file was deleted.
Oops, something went wrong.
28 changes: 0 additions & 28 deletions
28
loader/src/main/java/net/neoforged/fml/loading/targets/JUnitDevLaunchTarget.java
This file was deleted.
Oops, something went wrong.
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
80 changes: 80 additions & 0 deletions
80
loader/src/main/java/net/neoforged/fml/startup/FmlInstrumentation.java
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,80 @@ | ||
/* | ||
* Copyright (c) NeoForged and contributors | ||
* SPDX-License-Identifier: LGPL-2.1-only | ||
*/ | ||
|
||
package net.neoforged.fml.startup; | ||
|
||
import java.lang.instrument.Instrumentation; | ||
import java.util.ArrayList; | ||
|
||
public final class FmlInstrumentation { | ||
private FmlInstrumentation() {} | ||
|
||
public static Instrumentation obtainInstrumentation() { | ||
var stackWalker = StackWalker.getInstance(StackWalker.Option.RETAIN_CLASS_REFERENCE); | ||
var callingPackage = stackWalker.getCallerClass().getPackageName(); | ||
if (!callingPackage.equals(DevAgent.class.getPackage().getName()) | ||
&& !callingPackage.equals("net.neoforged.fml.junit")) { | ||
throw new IllegalStateException("This method may only be called by FML"); | ||
} | ||
|
||
var storedExceptions = new ArrayList<Exception>(); | ||
|
||
// Obtain instrumentation as early as possible. We use reflection here since we want to make sure that even if | ||
// we are loaded through other means, we get the agent class from the system CL. | ||
try { | ||
return getFromOurOwnAgent(); | ||
} catch (Exception e) { | ||
storedExceptions.add(e); | ||
} | ||
|
||
// Still don't have it? Try self-attach! | ||
// This most likely will go away in the next Java LTS, but until then, it's convenient for unit tests. | ||
try { | ||
var classpathItem = SelfAttach.getClassPathItem(); | ||
var command = ProcessHandle.current().info().command().orElseThrow(() -> new RuntimeException("Could not self-attach: failed to determine our own commandline")); | ||
var process = new ProcessBuilder(command, "-cp", classpathItem, SelfAttach.class.getName(), DevAgent.class.getName()) | ||
.redirectError(ProcessBuilder.Redirect.DISCARD) | ||
.redirectOutput(ProcessBuilder.Redirect.DISCARD) | ||
.inheritIO() | ||
.start(); | ||
process.getOutputStream().close(); | ||
var result = process.waitFor(); | ||
if (result != 0) { | ||
throw new RuntimeException("Could not self-attach agent: " + result); | ||
} | ||
return getFromOurOwnAgent(); | ||
} catch (Exception e) { | ||
storedExceptions.add(e); | ||
} | ||
|
||
// If our own self-attach fails due to a user using an unexpected JVM, we support that they add ByteBuddy | ||
// which has a plethora of self-attach options. | ||
try { | ||
var byteBuddyAgent = Class.forName("net.bytebuddy.agent.ByteBuddyAgent", true, ClassLoader.getSystemClassLoader()); | ||
var instrumentation = (Instrumentation) byteBuddyAgent.getMethod("install").invoke(null); | ||
StartupLog.info("Using byte-buddy fallback"); | ||
return instrumentation; | ||
} catch (Exception e) { | ||
storedExceptions.add(e); | ||
} | ||
|
||
var e = new IllegalStateException("Failed to obtain instrumentation."); | ||
storedExceptions.forEach(e::addSuppressed); | ||
throw e; | ||
} | ||
|
||
private static Instrumentation getFromOurOwnAgent() throws Exception { | ||
// This code may be surprising, but the DevAgent is *always* loaded on the system classloader. | ||
// If we have been loaded somewhere beneath, our copy of DevAgent may not be the same. To ensure we actually | ||
// get the "real" agent, we specifically grab the class from the system CL. | ||
var devAgent = Class.forName("net.neoforged.fml.startup.DevAgent", true, ClassLoader.getSystemClassLoader()); | ||
var instrumentation = (Instrumentation) devAgent.getMethod("getInstrumentation").invoke(null); | ||
StartupLog.info("Using our own agent"); | ||
if (instrumentation == null) { | ||
throw new IllegalStateException("Our DevAgent was not attached. Pass an appropriate -javaagent parameter."); | ||
} | ||
return instrumentation; | ||
} | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters