/LiteLoader/tags/1.6.4_02/java/com/mumfrey/liteloader/launch/ClassPathInjector.java

https://gitlab.com/LiteLoader/LiteLoaderMirror · Java · 143 lines · 94 code · 17 blank · 32 comment · 10 complexity · dc00c01ab74f4f8571efe0c7fe7e7ac6 MD5 · raw file

  1. package com.mumfrey.liteloader.launch;
  2. import java.lang.reflect.Field;
  3. import java.net.URL;
  4. import java.net.URLClassLoader;
  5. import java.util.ArrayList;
  6. import java.util.Stack;
  7. import java.util.logging.Level;
  8. import java.util.logging.Logger;
  9. import net.minecraft.launchwrapper.LaunchClassLoader;
  10. import sun.misc.URLClassPath;
  11. /**
  12. * Nasty horrible reflection hack to inject a classpath entry at positons in classpath other than at the bottom
  13. *
  14. * @author Adam Mummery-Smith
  15. */
  16. public abstract class ClassPathInjector
  17. {
  18. private static Logger logger = Logger.getLogger("liteloader");
  19. /**
  20. * URLClassLoader::ucp -> instance of URLClassPath
  21. */
  22. private static Field ucp;
  23. /**
  24. * URLClassLoader::urls -> instance of Stack<URL>
  25. */
  26. private static Field classPathURLs;
  27. /**
  28. * URLClassLoader::path -> instance of ArrayList<URL>
  29. */
  30. private static Field classPathPath;
  31. private static boolean canInject;
  32. static
  33. {
  34. try
  35. {
  36. ClassPathInjector.ucp = URLClassLoader.class.getDeclaredField("ucp");
  37. ClassPathInjector.ucp.setAccessible(true);
  38. ClassPathInjector.classPathURLs = URLClassPath.class.getDeclaredField("urls");
  39. ClassPathInjector.classPathURLs.setAccessible(true);
  40. ClassPathInjector.classPathPath = URLClassPath.class.getDeclaredField("path");
  41. ClassPathInjector.classPathPath.setAccessible(true);
  42. ClassPathInjector.canInject = true;
  43. }
  44. catch (Throwable th)
  45. {
  46. ClassPathInjector.logger.log(Level.SEVERE, "ClassPathInjector: Error initialising ClassPathInjector, special class path injection disabled", th);
  47. th.printStackTrace();
  48. }
  49. }
  50. /**
  51. * Injects a URL into the classpath at the TOP of the stack
  52. *
  53. * @param classLoader
  54. * @param url
  55. */
  56. public static void injectIntoClassPath(URLClassLoader classLoader, URL url)
  57. {
  58. ClassPathInjector.injectIntoClassPath(classLoader, url, null);
  59. }
  60. /**
  61. * Injects a URL into the classpath at the TOP of the stack
  62. *
  63. * @param classLoader
  64. * @param url
  65. * @param above
  66. */
  67. @SuppressWarnings({ "unchecked" })
  68. public static void injectIntoClassPath(URLClassLoader classLoader, URL url, URL above)
  69. {
  70. if (ClassPathInjector.canInject)
  71. {
  72. ClassPathInjector.logger.info(String.format("ClassPathInjector: attempting to inject %s into %s", url, classLoader.getClass().getSimpleName()));
  73. try
  74. {
  75. URLClassPath classPath = (URLClassPath)ClassPathInjector.ucp.get(classLoader);
  76. Stack<URL> urls = (Stack<URL>)ClassPathInjector.classPathURLs.get(classPath);
  77. ArrayList<URL> path = (ArrayList<URL>)ClassPathInjector.classPathPath.get(classPath);
  78. synchronized (urls)
  79. {
  80. if (!path.contains(url))
  81. {
  82. urls.add(url);
  83. if (above == null)
  84. {
  85. path.add(0, url);
  86. }
  87. else
  88. {
  89. for (int pos = path.size() - 1; pos > 0; pos--)
  90. {
  91. if (above.equals(path.get(pos)))
  92. path.add(pos, url);
  93. }
  94. }
  95. }
  96. }
  97. }
  98. catch (Exception ex)
  99. {
  100. ClassPathInjector.logger.warning(String.format("ClassPathInjector: failed to inject %s", url));
  101. }
  102. }
  103. if (classLoader instanceof LaunchClassLoader)
  104. {
  105. ((LaunchClassLoader)classLoader).addURL(url);
  106. }
  107. }
  108. /**
  109. * @param classLoader
  110. * @param url
  111. * @param above
  112. */
  113. public static void injectIntoClassPath(LaunchClassLoader classLoader, URL url, String above)
  114. {
  115. above = above.trim().toLowerCase();
  116. if (above.length() < 1) return;
  117. for (URL classPathUrl : classLoader.getURLs())
  118. {
  119. if (classPathUrl.toString().toLowerCase().contains(above))
  120. {
  121. ClassPathInjector.injectIntoClassPath(classLoader, url, classPathUrl);
  122. return;
  123. }
  124. }
  125. }
  126. }