前言
一般情况下,JDK中提供了Client和Server两种类型的JVM,那JDK是在运行时是如何选择的呢。本文主要探讨默认情况下Windows和Solaris(Linux)是如何选择JVM种类的。
JVM.cfg文件
在JRE_HOME/lib/<arch>(i386)/下存在这个一个jvm.cfg文件,用于配置JVM的种类。
/OpenJDK7/hotspot/src/share/tools/launcher/java.c 下的ReadKnownVMs函数注释详细的说明了该文件的主要格式:
/*
* jvmcfg := { vmLine }
1820 * vmLine := knownLine
1821 * | aliasLine
1822 * | warnLine
1823 * | ignoreLine
1824 * | errorLine
1825 * | predicateLine
1826 * | commentLine
1827 * knownLine := flag "KNOWN" EOL
1828 * warnLine := flag "WARN" EOL
1829 * ignoreLine := flag "IGNORE" EOL
1830 * errorLine := flag "ERROR" EOL
1831 * aliasLine := flag "ALIASED_TO" flag EOL
1832 * predicateLine := flag "IF_SERVER_CLASS" flag EOL
1833 * commentLine := "#" text EOL
1834 * flag := "-" identifier
*/
1.假如flag出现在knowLine行,那么identifier将作为存放JVM 库文件的路径名,去加载相应的VM;
2.假如flag出现在aliasLine行的第一个位置上,那么改行的第二个flag的identifier将作为JVM的名称;
3.假如flag出现在warnLine行的位置,那么该flag的identifier作为JVM的名称,但是会产生一条警告信息;
C:\Users\xingjl.fnst>D:\JDK\Oracle\jdk1.6.0_45\bin\java -classic -version
Warning: classic VM not supported; client VM will be used
java version "1.6.0_45"
Java(TM) SE Runtime Environment (build 1.6.0_45-b06)
Java HotSpot(TM) Client VM (build 20.45-b01, mixed mode)
4.假如flag出现在ignoreLine行的位置,那么该identifier将被忽略使用默认的VM。
5.假如flag出现在errorLine行的位置,那么错误消息将会生成。
C:\Users\xingjl.fnst>D:\JDK\Oracle\jdk1.6.0_45\bin\java -error -version
Unrecognized option: -error
Could not create the Java virtual machine.
6.假如flag出现在predicateLine行,并且通过了机器上的判断(是否为服务器),那么predicateLine行的第二个flag的identifier将作为vm的名称;否则使用第一个。
7.如果在命令行中没有flag指定,那么jvm.cfg的文件的第一行将作为VM的名称。
8.PredicateLines这一行只可以作为jvm,cfg文件的第一行使用。
下面一个Windows JDK6u45的jvm.cfg文件的内容
-client KNOWN
-server KNOWN
-hotspot ALIASED_TO -client
-classic WARN
-native ERROR
-green ERROR
Lanuger启动流程
首先java.exe的CreateExecutionEnvironment函数调用GetJREPath去分析JRE的的安装路径,然后调用ReadKnownVMs函数去加载jvm.cfg文件,将文件中内容加载到如下的结构体中:
/* Values for vmdesc.flag */
#define VM_UNKNOWN -1
#define VM_KNOWN 0
#define VM_ALIASED_TO 1
#define VM_WARN 2
#define VM_ERROR 3
#define VM_IF_SERVER_CLASS 4
#define VM_IGNORE 5
struct vmdesc {
char *name;
int flag;
char *alias;
char *server_class;
};
其中flag为上面宏定义中定义的内容,name为jvm.cfg 文件中每行的第一个flag,当一行是aliasLine行时,aliasLine 则记录第二个falg的值;当一行为predicateLine 行时,server_class则记录第二flag的值。
加载完成后,调用CheckJvmType函数来确定当前所使用的VM种类;
命令行指定VM类型
1.命令行中,使用的是VM_KNOWN类型的(-client或-server),那么JVM将按照相应的路径加载jvm.dll
-client |
${ JAVA_HOME } \jre\bin\client\jvm.dll |
-server |
${ JAVA_HOME } \jre\bin\server\jvm.dll |
2.在命令行中,使用的是VM_ALIASED_TO(别名)类型的第一个flag时(-hotspot),那么使用VM_ALIASED_TO行的第二个flag作为vm的种类
while (knownVMs[jvmidx].flag == VM_ALIASED_TO) {
int nextIdx = KnownVMIndex(knownVMs[jvmidx].alias);
–中略-
jvmidx = nextIdx;
jvmtype = knownVMs[jvmidx].name+1;
loopCount++;
}
3.在命令行中,使用的是VM_WARN行内容,则先输出一条警告消息,然后使用jvm.cfg文件中的第一行falg作为VM类型
4.在命令行中,使用的是VM_IGNORE行内容,使用jvm.cfg文件中的第一行falg作为VM类型
5.在命令行中,使用的是VM_ERROR行内容,则会打印Error消息。
以下是3-5条的代码:
switch (knownVMs[jvmidx].flag) {
case VM_WARN:
if (!speculative) {
fprintf(stderr, "Warning: %s VM not supported; %s VM will be used\n",
jvmtype, knownVMs[0].name + 1);
}
/* fall through */
case VM_IGNORE:
jvmtype = knownVMs[jvmidx=0].name + 1;
/* fall through */
case VM_KNOWN:
break;
case VM_ERROR:
if (!speculative) {
ReportErrorMessage2("Error: %s VM not supported", jvmtype, JNI_TRUE);
exit(1);
} else {
return "ERROR";
}
}
命令行未指定VM类型
在CheckJvmType函数中有如下代码:
if (jvmtype == NULL) {
char* result = knownVMs[0].name+1;
/* Use a different VM type if we are on a server class machine? */
if ((knownVMs[0].flag == VM_IF_SERVER_CLASS) &&
(ServerClassMachine() == JNI_TRUE)) {
result = knownVMs[0].server_class+1;
}
if (_launcher_debug) {
printf("Default VM: %s\n", result);
}
return result;
}
由此可知,VM类型默认使用jvm.cfg第一行的flag。最终由该行的类型(是否为VM_IF_SERVER_CLASS)以及是否为服务器决定VM的类型。
Windows下:
JDK6u45的jvm.cfg文件参照上面提供的。
第一行非VM_IF_SERVER_CLASS类型,如果人为的修改为VM_IF_SERVER_CLASS类型,进行第二步的判断:是否为服务器级别的机器,Windows环境下的ServerClassMachine函数如下:
824jboolean
825ServerClassMachine() {
826 jboolean result = JNI_FALSE;
827#if defined(NEVER_ACT_AS_SERVER_CLASS_MACHINE)
828 result = JNI_FALSE;
829#elif defined(ALWAYS_ACT_AS_SERVER_CLASS_MACHINE)
830 result = JNI_TRUE;
831#endif
832 return result;
833}
一般情况下总是返回false,所以,即使修改了cfg文件,VM种类也不会根据机器的配置进行自动搭配。
Solaris(Linux)
JDK6u45的jvm.cfg文件如下:
-client IF_SERVER_CLASS -server
-server KNOWN
-hotspot ALIASED_TO -client
-classic WARN
-native ERROR
-green ERROR
ServerClassMachine函数如下:
jboolean
1457ServerClassMachine(void) {
1458 jboolean result = JNI_FALSE;
1459#if defined(NEVER_ACT_AS_SERVER_CLASS_MACHINE)
1460 result = JNI_FALSE;
1461#elif defined(ALWAYS_ACT_AS_SERVER_CLASS_MACHINE)
1462 result = JNI_TRUE;
1463#elif defined(__sun) && defined(__sparc)
1464 result = solaris_sparc_ServerClassMachine();
1465#elif defined(__sun) && defined(i586)
1466 result = solaris_i586_ServerClassMachine();
1467#elif defined(__linux__) && defined(i586)
1468 result = linux_i586_ServerClassMachine();
1469#else
1470 if (_launcher_debug) {
1471 printf("ServerClassMachine: returns default value of %s\n",
1472 (result == JNI_TRUE ? "true" : "false"));
1473 }
1474#endif
1475 return result;
1476}
我们以sun sparc平台的JDK为例,solaris_sparc_ServerClassMachine函数代码如下:
1102jboolean
1103solaris_sparc_ServerClassMachine(void) {
1104 jboolean result = JNI_FALSE;
1105 /* How big is a server class machine? */
1106 const unsigned long server_processors = 2UL;
1107 const uint64_t server_memory = 2UL * GB;
1108 const uint64_t actual_memory = physical_memory();
1109
1110 /* Is this a server class machine? */
1111 if (actual_memory >= server_memory) {
1112 const unsigned long actual_processors = physical_processors();
1113 if (actual_processors >= server_processors) {
1114 result = JNI_TRUE;
1115 }
1116 }
1117 if (_launcher_debug) {
1118 printf("solaris_" LIBARCHNAME "_ServerClassMachine: %s\n",
1119 (result == JNI_TRUE ? "JNI_TRUE" : "JNI_FALSE"));
1120 }
1121 return result;
1122}
如果机器的内存大于等于2GB,并且机器的CPU为2核或以上时,即判定为服务器级别的机器。
综合Widows和Solaris的分析,默认的VM种类选在如下:
1. 32位的JDK在Windows平台上,总是使用-client模式(即jvm.cfg文件中的第一个flag)
默认第一个flag
-client KNOWN
2. 32位的JDK在Solaris(Linux)机器上,根据机器是否为服务器级别的选择相应的VM;服务器级别的使用-sever,否则使用-client
jvm.cfg默认的第一个falg
-client IF_SERVER_CLASS -server
3. 64位的JDK在Windows,Solaris平台上只有-server一种VM类型,可以选择-client但是 会被忽略。
-client IGNORE
4. 64位的JDK在Linux平台上只有-server一种VM类型,选择-client会报错
-client ERROR
-以上-