源码学习准备

前言

java的跨平台是如何实现的,为了提升性能又做了哪些工作,所以学习jdk源码很有必要。我们知道hotspot是C++写的,其实里面有大量的汇编直接操作CPU寄存器,C++的很多特性和具体的汇编指令我们不必要全部了解,大概知道有栈创建对象、堆创建对象,方法调用,宏定义等基本的就可以。通过学习hotspot源码会对加深计算机的操作系统、CPU寄存器、内存以及指针压缩、线程栈内存分配、大页内存、NUMA架构等了解,下面就让我们进入hotspot源码的世界。

源码下载

下载openjdk8源码压缩包
OpenJDK™ Source Releases (java.net)

IDE工具安装

1、CLion下载安装
CLoin download
因同是JetBrains产品,激活同IDEAJ,使用ja-netfilter

导入项目

image.png

熟悉目录结构

jdk以及hotspot的源码目录结构有相同点,就是基础相同的代码在share目录,操作系统及CPU有区别的分别放置各自目录,比如jdk中除了share目录有windows、solaris(solaris、linux等unix派系),这个就是操作系统相关的代码。Hotspot虚拟机的源码除了share目录还有os(操作系统相关)、cpu(cpu架构指令相关)、os_cpu(操作系统和CPU架构相关)目录。

源码阅读小技巧

hotspot源码是c、c++写的,阅读起来比较困难,经常有类 或 方法不知道哪里来的,eclipse也跟踪不进去,我们一般看include了哪些头文件,按住CTRL点不进去的代码先在include的头文件找,有的是多层include,实在找不着的就全局搜索了。比如jdk的share/bin/main.c中方法JLI_Launch不知道哪里来的,我们看导main.c里面有#include "defines.h", 就去defines.h里面找,发现defines.h里面有#include "java.h",那我们基本知道是java.h或java.c里面定义的JLI_Launch。

C C++小知识

new对象

栈上分配对象 ObjectClass obj;
堆上分配对象 ObjectClass *obj = new ObjectClass(); ,调用方法用->

方法调用

1、::类静态方法调用 Lable p = Label::create();
2、->类对象方法指针调用 p.create();
3、.类对象方法实例调用 (
p).create();

引用与指针

&标识引用
标识指针 (p) 标识指针指向的值
指针和引用都可以做为参数,func(int *i) 、 func(int &i)

正式开始

我们都知道java程序运行过程,由main方法或者jar的manifest文件定义的启动类的main方法开始,这个过程又是怎样的呢,就从JVM启动开始一步一步探究。
java Test.class
则调jdk的src/share/bin/main.c中JLI_Launch方法


image.png

JLI_Launch我通过#include "defines.h"引入,defines.h中又引入了java.h


image.png

因此我们找到了java.c中定义的JLI_Launch方法


image.png

这个方法里面前面都是校验和准备工作,判断CPU架构、找到jre路径、解析java启动命令带的参数、装载动态库jvm.all或jvm.so(LoadJavaVM)等等,最后一步是ContinueInNewThread方法,启动了一个线程,也就是我们的main主线程。
ContinueInNewThread方法中调用ContinueInNewThread0方法,
image.png

ContinueInNewThread0方法我们只能在java.h中有定义,而java.h中有引入#include "java_md.h" ,这个根据操作系统不同实现不一样,我们看一下unix派系操作系统的实现,代码在jdk的src/solaris/bin/java_md.c。


image.png

意思是根据线程栈大小创建线程,然后从continuation地址开始,等待CPU调度,也就是CPU执行的是continuation地址对应的代码,我们再回过头看下java.c中的调用传入参数,rslt = ContinueInNewThread0(JavaMain, threadStackSize, (void*)&args); 因此实际调用的是java.c中的JavaMain方法。

//java.c
int JNICALL
JavaMain(void * _args)
{
   JavaMainArgs *args = (JavaMainArgs *)_args;
   int argc = args->argc;
   char **argv = args->argv;
   int mode = args->mode;
   char *what = args->what;
   InvocationFunctions ifn = args->ifn;

   JavaVM *vm = 0;
   JNIEnv *env = 0;
   jclass mainClass = NULL;
   jmethodID mainID;
   jobjectArray mainArgs;
   int ret = 0;
   jlong start, end;

   /* Initialize the virtual machine */
   start = CounterGet();
   if (!InitializeJVM(&vm, &env, &ifn)) {
       JLI_ReportErrorMessage(JVM_ERROR1);
       exit(1);
   }

   if (printVersion || showVersion) {
       PrintJavaVersion(env, showVersion);
       CHECK_EXCEPTION_LEAVE(0);
       if (printVersion) {
           LEAVE();
       }
   }

   if (showSettings != NULL) {
       ShowSettings(env, showSettings);
       CHECK_EXCEPTION_LEAVE(1);
   }
   /* If the user specified neither a class name nor a JAR file */
   if (printXUsage || printUsage || what == 0 || mode == LM_UNKNOWN) {
       PrintUsage(env, printXUsage);
       CHECK_EXCEPTION_LEAVE(1);
       LEAVE();
   }

   FreeKnownVMs();  /* after last possible PrintUsage() */

   if (JLI_IsTraceLauncher()) {
       end = CounterGet();
       JLI_TraceLauncher("%ld micro seconds to InitializeJVM\n",
              (long)(jint)Counter2Micros(end-start));
   }

   /* At this stage, argc/argv have the application's arguments */
   if (JLI_IsTraceLauncher()){
       int i;
       printf("%s is '%s'\n", launchModeNames[mode], what);
       printf("App's argc is %d\n", argc);
       for (i=0; i < argc; i++) {
           printf("    argv[%2d] = '%s'\n", i, argv[i]);
       }
   }

   ret = 1;

   /*
    * Get the application's main class.
    *
    * See bugid 5030265.  The Main-Class name has already been parsed
    * from the manifest, but not parsed properly for UTF-8 support.
    * Hence the code here ignores the value previously extracted and
    * uses the pre-existing code to reextract the value.  This is
    * possibly an end of release cycle expedient.  However, it has
    * also been discovered that passing some character sets through
    * the environment has "strange" behavior on some variants of
    * Windows.  Hence, maybe the manifest parsing code local to the
    * launcher should never be enhanced.
    *
    * Hence, future work should either:
    *     1)   Correct the local parsing code and verify that the
    *          Main-Class attribute gets properly passed through
    *          all environments,
    *     2)   Remove the vestages of maintaining main_class through
    *          the environment (and remove these comments).
    */
   mainClass = LoadMainClass(env, mode, what);
   CHECK_EXCEPTION_NULL_LEAVE(mainClass);

   /*
    * The LoadMainClass not only loads the main class, it will also ensure
    * that the main method's signature is correct, therefore further checking
    * is not required. The main method is invoked here so that extraneous java
    * stacks are not in the application stack trace.
    */
   mainID = (*env)->GetStaticMethodID(env, mainClass, "main",
                                      "([Ljava/lang/String;)V");
   CHECK_EXCEPTION_NULL_LEAVE(mainID);

   /* Build argument array */
   mainArgs = NewPlatformStringArray(env, argv, argc);
   CHECK_EXCEPTION_NULL_LEAVE(mainArgs);

   /* Invoke main method. */
   (*env)->CallStaticVoidMethod(env, mainClass, mainID, mainArgs);

   /*
    * The launcher's exit code (in the absence of calls to
    * System.exit) will be non-zero if main threw an exception.
    */
   ret = (*env)->ExceptionOccurred(env) == NULL ? 0 : 1;
   LEAVE();
}

准备阶段判断如果JVM没有初始化则初始化JVM(InitializeJVM),初始化JVM先创建JVM(CreateJavaVM),这个会通过JNI创建虚拟机,我们后面再看下JVM是如何创建的。
JVM准备好了再执行main方法,核心有4步,1:mainClass = LoadMainClass(env, mode, what); 加载主class文件; 2: mainID = (env)->GetStaticMethodID(env, mainClass, "main", "([Ljava/lang/String;)V"); 获取静态方法main方法;3:mainArgs = NewPlatformStringArray(env, argv, argc); 获取方法参数列表;4:(env)->CallStaticVoidMethod(env, mainClass, mainID, mainArgs);调用main方法。

后面章节将会详细讲解这个主方法,解开我们的疑惑1:class是如何加载的,2:main方法是怎么调用。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,417评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,921评论 3 387
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 158,850评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,945评论 1 285
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,069评论 6 385
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,188评论 1 291
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,239评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,994评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,409评论 1 304
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,735评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,898评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,578评论 4 336
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,205评论 3 317
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,916评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,156评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,722评论 2 363
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,781评论 2 351

推荐阅读更多精彩内容