JPDA 架构研究2 - JVMTI代理

引入:

我们先从JVMTI讲起。JVMTI的主要作用是提供一组接口来检测VM的状态和控制VM中运行的JAVA程序。JVMTI是个双向接口:

JVMTI的客户端叫Agent,它会在VM发生变化时通过事件机制被通知到变化。

JVMTI的服务端是许多函数,它们会和VM实际打交道并把结果告知Agent.


实践:

我们这里先来看下Agent.


Agent的方法定义在哪里呢?它们定义在$JAVA_HOME/include/jvmti.h文件中。

JNIEXPORT jint JNICALL
Agent_OnLoad(JavaVM *vm, char *options, void *reserved);

JNIEXPORT void JNICAL。
Agent_OnUnload(JavaVM *vm);


Agent_OnLoad方法:

当代理被VM加载时,会由VM调用Agent_Onload方法。此时VM有如下能力:

(1)VM的System Property已经被设置完毕。

(2)VM的Capabilities已经被设置完毕。

(3)任何字节码都没被执行。

(4)任何类都没有被加载。

(5)任何对象都没有被创建。


Agent_OnLoad实现过程中,最重要的事情之一就是调用GetEnv方法来获取JVMTI环境的指针。

jint GetEnv(JavaVM *vm, void **env, jint version);

这样,这个代理就可以知道被代理的环境的全部信息。



Agent_OnUnload方法:

当代理被VM卸载时,会由VM调用Agent_OnUnload方法。一般发生在终止VM的时候,它一般可以用来清理在Agent_OnLoad阶段创建的资源。


因为我们在远程调试时候启动JVM时候加了代理参数:

java -agentlib:<agentLibName> ,而agentLibName我们配置的是jdwp, 所以它就对应上jdwp.dll.

也就是远程调试时候自动会在启动target VM时候启用jdwp这个代理库.


当明白了代理的作用后,我们来找其的实现。在Sun的JDK中,我们找到了jdwp.dll (Linux环境则是jdwp.so) .  它位于 $JAVA_HOME/jre/bin目录下。我们用exeScope软件打开查看内容:

wKiom1SFSz7xFsInAAJpNug-rho655.jpg

显然,它是Oracle的Sun的JDK提供的Agent ,它提供了2个方法,一个是_Agent_OnLoad(),一个是_Agent_OnUnload()

wKiom1SFS37Ty0DhAAEzsbfIxTM606.jpg

和我们设想的一致。





总结:

从上过程我们似乎可以总结一些结论:

1. Agent 是在虚拟机启动之时加载的,这个加载处于虚拟机初始化的早期.在这个时间点上:

  所有的 Java 类都未被初始化;所有的对象实例都未被创建; 因而,没有任何 Java 代码被执行; 

(从这点上说,最明显的好处就是它能完成早期调试中用System.out.println()无法解决的问题,因为System.out.println()前提是代码行所在的类已经被初始化过了)

2.但在这个时候,我们已经可以:

操作 JVMTI 的 Capability 参数; 使用系统参数; 动态库被加载之后,虚拟机会先寻找一个 Agent 入口函数.


你可能感兴趣的:(jpda,JDWP)