Java

脑袋抽筋了的我非要调试OpenJdk,且看他怎么虐我

2020-10-24  本文已影响0人  马小莫QAQ

前言

说来话长,最近又闲的无聊,在看JDK的源码,但是很多关键的地方都是native方法,这就导致需要在往深处看,也就是需要看openjdk源码了,但是c++代码又谈何容易,况且也不怎么会。

但是想来想去,决定还是要研究一下的,在以前的文章中已经编译过了openjdk11,虽然过程坎坷,但也是成功了,那么接下来就是导入openjdk源码到ide中,在下面会以clion作为示例,clion下载安装就不说了。

激动的心,颤抖的手,Deepin下成功编译OpenJdk11!!!

光这导入就花了我1天,毕竟openjdk很复杂,而且参考的文章有的说只导入hotspot目录,有的说导入openjdk\src目录,还有的说要导入根目录,导来导去,差点没放弃,但是放弃这1天就浪费了,所以瞎搞了一顿也不知道算不算成功,反正能正常调试了。

后面会使用到编译好的java程序(指的是bin下面的java),所以必须先编译openjdk。

导入openjdk

然后在"选择"对话框中选择openjdk的根目录,没错就是根目录,然后见到ok点ok,见到finish点finish。

导入后大概这个样子。

然后按照下面gif进行配置,注意Executable是选择编译好的java可执行程序,在bin目录下,也就你编译出来的,不能选在官网下载的。

记住如果在Before launch下有东西的话全部移除。

点击运行就能出来版本信息,这就是上步配置的-version参数起的作用。

运行java程序

但是需要运行我们写好的程序才行啊,不然图了个啥,所以先简单写一个:

publicclassMain{

    publicstaticvoidmain(String[] args){

        inta=12;

        intb=34;       

        System.out.println("a+b="+(a+b));

    }

    publicstaticvoidtest(String[] args){       

        System.out.println("test");

    }

}

我们知道java的程序是从main方法开始的,但是能不能不从main开始,当然可以,就需要我们修改openjdk源码,上面的test方法稍后会充当main方法运行。

先进行编译。

然后重新配置一下启动参数。

再次运行。

上面说了,我们能不能不从main开始,当然可以,回到openjdk源码,找到java.c文件,定位到JavaMain方法下,里面有一段是获取main方法的方法id,接着后面会调用这个方法,我们只需要修改GetStaticMethodID参数中的"main"即可,然后重新编译,这次编译只需要执行make就行,而且速度很快。

关于([Ljava/lang/String;)V的意思在以前分析class文件格式的时候也说过了,代表是一个String数组,并且方法是无返回值的。

恕我直言,这可能是你见过最详细的class文件结构分析

int JNICALL

JavaMain(void * _args){

mainID=(*env)->GetStaticMethodID(env, mainClass,"main","([Ljava/lang/String;)V"); 

(*env)->CallStaticVoidMethod(env, mainClass, mainID, mainArgs);

}

再次运行,可以看到,这次没执行main方法,而是执行了test方法。

剩下就是打断点进行调试,比如线程的start0方法,最终会调用操作系统的api启动线程,我们可以从中进行断点调试。

上一篇 下一篇

猜你喜欢

热点阅读