Создав собственный профилировщик, я использую API JVMTI для создания собственного агента библиотеки. Этот агент можно запустить вместе с JVM с помощью параметра добавления -agentlib. Кроме того, существует Attach API, который позволяет внедрить агент в запущенную JVM. Я хотел реализовать эту функцию для своего профилировщика, используя следующий код:
try {
String pid = VirtualMachine.list().get(0).id();
VirtualMachine vm = VirtualMachine.attach(pid);
vm.loadAgentLibrary("agent");
} catch (AgentLoadException e1) {
e1.printStackTrace();
} catch (AgentInitializationException e1) {
e1.printStackTrace();
} catch (IOException e1) {
e1.printStackTrace();
} catch (AttachNotSupportedException e) {
e.printStackTrace();
}
Что он делает? Из всех доступных виртуальных машин (VirtualMachine.list()
) я выбираю первый, присоединяюсь к нему и пытаюсь загрузить в него свой агент. Агент, в системах UNIX с именем libagent.so, можно найти, но при попытке загрузить агент вызывается следующее исключение:
com.sun.tools.attach.AttachNotSupportedException:
Unable to open socket file:
target process not responding or HotSpot VM not loaded.
Заглядывая в исходный код, это исключение вызывается, потому что он не может найти файл с именем .java_pid<pid>
. Я не нашел много информации об этом виде файла в документации. Я часто слышал, что этот файл больше не используется, но я запускаю Java 1.6.
Я также пытался подключиться к другим JVM, на самом деле я сохранил этот процесс присоединения динамическим, по причинам тестирования я просто пытаюсь подключиться к любой JVM.
Это код, который приводит к исключению, взятому из sun.tools.attach: LinuxVirtualMachine.java:
// Return the socket file for the given process.
// Checks working directory of process for .java_pid<pid>. If not
// found it looks in /tmp.
private String findSocketFile(int pid) {
// First check for a .java_pid<pid> file in the working directory
// of the target process
String fn = ".java_pid" + pid;
String path = "/proc/" + pid + "/cwd/" + fn;
File f = new File(path);
if (!f.exists()) {
// Not found, so try /tmp
path = "/tmp/" + fn;
f = new File(path);
if (!f.exists()) {
return null; // not found
}
}
return path;
}
В нем говорится, что он ищет корень в каталоге /proc/<pid>
. Глядя на набор изменений JDK7, кажется, что они вносят изменения в код JDK7 Изменения в LinuxVirtualMachine