How can I tell if I'm running in 64-bit JVM or 32-bit JVM (from within a program)?
Asked Answered
E

13

610

How can I tell if the JVM in which my application runs is 32 bit or 64-bit? Specifically, what functions or properties I can used to detect this within the program?

Emphasis answered 14/1, 2010 at 3:38 Comment(9)
Just out of curiosity, why would you need to know the natural size of the system? Details such as this are abstracted away in Java, so you shouldn't (in theory, at least) have to know them.Groundless
It lets me roughly estimate the memory requirements for objects due to pointers. Curiosity too -- seemed like there should be a way, but I'd never heard of it.Emphasis
This "detail" is not abstracted away when interacting with the Java Native Interface. 32-bit DLLs can't be loaded with a 64-bit JVM (and vice versa). So this is quite essential information for anyone using JNI. It's a pity that there seems to be no portable way to obtain this info. One way is to first try loading a 32-bit version of the DLL, and if it fails, try the 64-bit version, etc. Ugly!Nonoccurrence
Another situation where discerning between 32 or 64 bit JVMs is important is for mapped files. On 32 bit systems only 2GB can be mapped, so it's important to map and unmap file segments accordingly so that this limit is not exceeded, while on 64 bit jvms the limit is much, much, much higher.Swafford
@simonegianni wouldn't that make such a mapping the perfect test?Claimant
Note that there is a vast difference between a human being able to tell, and the program being able to tell. Humanly readable strings are rarely defined rigidly so relying on them makes you subtly vendor dependent.Claimant
It's really nice to be able to choose the numerical algorithm that will be fastest on the machine in question.Riotous
Also note that both the detection mechanisms and the object size is vendor dependent.Claimant
@PatrickNiedzielski Another situation where discerning between 32 or 64 bit JVMs is of paramount importance is when a Java program must start an external executable, for which both versions exist. Like writing Selenium tests, where the right browser driver must be started. And the same need exists for discerning the operating system. I'm sure one can think of several more cases where something that is supposed to be "abstracted away" is not.Kansas
M
337

You retrieve the system property that marks the bitness of this JVM with:

System.getProperty("sun.arch.data.model");

Possible results are:

  • "32" – 32-bit JVM
  • "64" – 64-bit JVM
  • "unknown" – Unknown JVM

As described in the HotSpot FAQ:

When writing Java code, how do I distinguish between 32 and 64-bit operation?

There's no public API that allows you to distinguish between 32 and 64-bit operation. Think of 64-bit as just another platform in the write once, run anywhere tradition. However, if you'd like to write code which is platform specific (shame on you), the system property sun.arch.data.model has the value "32", "64", or "unknown".

An example where this could be necessary is if your Java code depends on native libraries, and you need to determine whether to load the 32- or 64-bit version of the libraries on startup.

Maemaeander answered 14/1, 2010 at 3:44 Comment(9)
Is this defined by the Java spec to work under all JVMs, or is it Hotspot-specific?Emphasis
I wouldn't expect to find sun.* system properties with an IBM JVM. In other words, it's not portable.Nonrecognition
Is this still valid now that Java is owned by Oracle? I am running an environment-auditor that will run pre-deploy of our application. We need to verify that the 64-bit production boxes had 64-bit Java installed on them since we don't manage these machines. I need to make sure this still works and will continue to work moving forward with Oracle.Zachery
How can you tell from the command line? If you are running 32-bit or 64-bit? Just curious.Dying
Why is the accepted answer Sun-dependent? "os.arch" will accomplish the same thing without having to use proprietary sun packages.Kalk
@b1naryatr0phy, does os.arch report on the Operating System, or the JVM? I often run 32-bit JVM on my 64-bit workstation, for development purposes.Girardi
This property is supported on IBM JVMs, but not on GCJ. See #807763Therapist
For logging and benchmarking the type of VM is pretty important as well. So stating it is useful only for dependence on native libraries is incorrect, in my opinion.Armalda
As GCJ is as dead as a doornail, what it does and doesn't support is of no interest.Stillbirth
C
762

For certain versions of Java, you can check the bitness of the JVM from the command line with the flags -d32 and -d64.

$ java -help
...
    -d32          use a 32-bit data model if available
    -d64          use a 64-bit data model if available

To check for a 64-bit JVM, run:

$ java -d64 -version

If it's not a 64-bit JVM, you'll get this:

Error: This Java instance does not support a 64-bit JVM.
Please install the desired version.

Similarly, to check for a 32-bit JVM, run:

$ java -d32 -version

If it's not a 32-bit JVM, you'll get this:

Error: This Java instance does not support a 32-bit JVM.
Please install the desired version.

These flags were added in Java 7, deprecated in Java 9, removed in Java 10, and no longer available on modern versions of Java.

Cyclotron answered 14/1, 2010 at 5:0 Comment(15)
Although this is good to know, it's not useful b/c I need to run it from outside the program or use java options to start a new process.Emphasis
Sorry I, completely misread the question. Within Java itself, the correct manner would be as described by codadictCyclotron
Exactly what I was looking for. And you can run java -d32 -version to verify you are not running 32-bit. Both wish work on Win7.Dying
I am on Windows 7, and I get 'unrecognized option' error from java -d32 -version and also from java -d64 -version.Deictic
I am able to use this command to determine the bits of my VM, thanks! CGuo@CGUO-E8420 ~ $ java -d64 -version Error: This Java instance does not support a 64-bit JVM. Please install the desired version. CGuo@CGUO-E8420 ~ $ java -d32 -version java version "1.7.0_09" Java(TM) SE Runtime Environment (build 1.7.0_09-b05) Java HotSpot(TM) Client VM (build 23.5-b02, mixed mode)Pahang
I am on windows 7 and the lowercase -d64 appeared to work while the uppercase -D64 appeared to have no effectBogy
For me neither -D64 nor -D32 is showing `This Java instance does not support a 64-bit JVM. Please install the desired version.'.Nottinghamshire
Do not use "-D64", for that does something completely different. It defines a system property called "64". This is definitely not what is wanted here.Banc
The -d32 or -d64 flags will only work for a Java 7 or greater.Flier
Reading and trying again and again, and end up with sad news. I have JDK 6Genoa
No -version argument is required. You can use java -d64.Wilfredowilfrid
@Flier FYI, one of my old linux server running JDK 1.5.0_17, it works for both -d32 & -d64. The output is: "Running a 64-bit JVM is not supported on this platform." for d64, while the d32 option is regular version output.Checkrein
@Willmore No, it is not correct. If you have 64 bit JVM and run java -d32, it detects error of not supported. But, if you run java -d64, it complains about missing parameters and -version is needed.Badman
java -d64 -version. Worked on Windows 10, thank you.Recension
This argument was removed in Java 9, see JDK-8168010. Additionally it might not satisfy "from within a program" reliably, given that Solaris supported both pre JDK8 and therefore you could likely currently be running with a 32 bit data model, but java -d64 -version returned with exit code 0.Mizzen
M
337

You retrieve the system property that marks the bitness of this JVM with:

System.getProperty("sun.arch.data.model");

Possible results are:

  • "32" – 32-bit JVM
  • "64" – 64-bit JVM
  • "unknown" – Unknown JVM

As described in the HotSpot FAQ:

When writing Java code, how do I distinguish between 32 and 64-bit operation?

There's no public API that allows you to distinguish between 32 and 64-bit operation. Think of 64-bit as just another platform in the write once, run anywhere tradition. However, if you'd like to write code which is platform specific (shame on you), the system property sun.arch.data.model has the value "32", "64", or "unknown".

An example where this could be necessary is if your Java code depends on native libraries, and you need to determine whether to load the 32- or 64-bit version of the libraries on startup.

Maemaeander answered 14/1, 2010 at 3:44 Comment(9)
Is this defined by the Java spec to work under all JVMs, or is it Hotspot-specific?Emphasis
I wouldn't expect to find sun.* system properties with an IBM JVM. In other words, it's not portable.Nonrecognition
Is this still valid now that Java is owned by Oracle? I am running an environment-auditor that will run pre-deploy of our application. We need to verify that the 64-bit production boxes had 64-bit Java installed on them since we don't manage these machines. I need to make sure this still works and will continue to work moving forward with Oracle.Zachery
How can you tell from the command line? If you are running 32-bit or 64-bit? Just curious.Dying
Why is the accepted answer Sun-dependent? "os.arch" will accomplish the same thing without having to use proprietary sun packages.Kalk
@b1naryatr0phy, does os.arch report on the Operating System, or the JVM? I often run 32-bit JVM on my 64-bit workstation, for development purposes.Girardi
This property is supported on IBM JVMs, but not on GCJ. See #807763Therapist
For logging and benchmarking the type of VM is pretty important as well. So stating it is useful only for dependence on native libraries is incorrect, in my opinion.Armalda
As GCJ is as dead as a doornail, what it does and doesn't support is of no interest.Stillbirth
B
201

Just type java -version in your console.

If a 64 bit version is running, you'll get a message like:

java version "1.6.0_18"
Java(TM) SE Runtime Environment (build 1.6.0_18-b07)
Java HotSpot(TM) 64-Bit Server VM (build 16.0-b13, mixed mode)

A 32 bit version will show something similar to:

java version "1.6.0_41"
Java(TM) SE Runtime Environment (build 1.6.0_41-b02)
Java HotSpot(TM) Client VM (build 20.14-b01, mixed mode, sharing)

Note Client instead of 64-Bit Server in the third line. The Client/Server part is irrelevant, it's the absence of the 64-Bit that matters.

If multiple Java versions are installed on your system, navigate to the /bin folder of the Java version you want to check, and type java -version there.

Bolivia answered 3/6, 2011 at 20:58 Comment(2)
but in hp nonstop oss env I'm not getting 64bit or 32 bitFredrika
OP specifically says within the program.Postobit
T
37

I installed 32-bit JVM and retried it again, looks like the following does tell you JVM bitness, not OS arch:

System.getProperty("os.arch");
#
# on a 64-bit Linux box:
# "x86" when using 32-bit JVM
# "amd64" when using 64-bit JVM

This was tested against both SUN and IBM JVM (32 and 64-bit). Clearly, the system property is not just the operating system arch.

Trinh answered 14/1, 2010 at 3:47 Comment(7)
Yeah. It gives x86, etc. Not quite the same thing, since I'm not sure if it gives "x64" for x86-64 running in 64-bit rather than 32-bit.Emphasis
@codaddict, looks like it is JVM bitness indeed.Trinh
@Maemaeander That is completely false (and I have no idea why six ppl have voted that comment up.) "os.arch" is designed to return the JVM version. Test it out for yourself and god help you if you're actually relying on this for OS detection.Kalk
@b1naryatr0phy, never mind my comment to you on the other answer to this question. Thanks! :)Girardi
os.arch has many possible values, it's difficult to tell if it's 32 or 64 bits. See lopica.sourceforge.net/os.htmlTherapist
This is a string intended for human eyes and without a strict definition of valid values, relying on this is not a good idea - write code that checks actual functionality instead.Claimant
it's amd64 not xmd64Strenuous
W
15

Complementary info:

On a running process you may use (at least with some recent Sun JDK5/6 versions):

$ /opt/java1.5/bin/jinfo -sysprops 14680 | grep sun.arch.data.model
Attaching to process ID 14680, please wait...
Debugger attached successfully.
Server compiler detected.
JVM version is 1.5.0_16-b02
sun.arch.data.model = 32

where 14680 is PID of jvm running the application. "os.arch" works too.

Also other scenarios are supported:

jinfo [ option ] pid
jinfo [ option ] executable core
jinfo [ option ] [server-id@]remote-hostname-or-IP 

However consider also this note:

"NOTE - This utility is unsupported and may or may not be available in future versions of the JDK. In Windows Systems where dbgent.dll is not present, 'Debugging Tools for Windows' needs to be installed to have these tools working. Also the PATH environment variable should contain the location of jvm.dll used by the target process or the location from which the Crash Dump file was produced."

Writ answered 16/2, 2012 at 19:4 Comment(0)
S
7

On Linux, you can get ELF header information by using either of the following two commands:

file {YOUR_JRE_LOCATION_HERE}/bin/java

o/p: ELF 64-bit LSB executable, AMD x86-64, version 1 (SYSV), for GNU/Linux 2.4.0, dynamically linked (uses shared libs), for GNU/Linux 2.4.0, not stripped

or

readelf -h {YOUR_JRE_LOCATION_HERE}/bin/java | grep 'Class'

o/p: Class: ELF64

Sauerkraut answered 7/6, 2012 at 8:50 Comment(0)
K
7

If you are using JNA, you can check whether com.sun.jna.Native.POINTER_SIZE == 4 (32 bit) or com.sun.jna.Native.POINTER_SIZE == 8 (64 bit).

Korn answered 10/4, 2015 at 16:49 Comment(1)
This is clever but accessing the pointer size is significantly slower than the other solutions here (it needs some time to initialize).Burgin
B
5

If you're using JNA, you can do this

Platform.is64Bit()
Betweenwhiles answered 8/1, 2019 at 3:43 Comment(0)
L
1

This is the way JNA solves this with Platform.is64Bit() (https://github.com/java-native-access/jna/blob/master/src/com/sun/jna/Platform.java)

 public static final boolean is64Bit() {
        String model = System.getProperty("sun.arch.data.model",
                                          System.getProperty("com.ibm.vm.bitmode"));
        if (model != null) {
            return "64".equals(model);
        }
        if ("x86-64".equals(ARCH)
            || "ia64".equals(ARCH)
            || "ppc64".equals(ARCH) || "ppc64le".equals(ARCH)
            || "sparcv9".equals(ARCH)
            || "mips64".equals(ARCH) || "mips64el".equals(ARCH)
            || "amd64".equals(ARCH)
            || "aarch64".equals(ARCH)) {
            return true;
        }
        return Native.POINTER_SIZE == 8;
}

ARCH = getCanonicalArchitecture(System.getProperty("os.arch"), osType);

static String getCanonicalArchitecture(String arch, int platform) {
        arch = arch.toLowerCase().trim();
        if ("powerpc".equals(arch)) {
            arch = "ppc";
        }
        else if ("powerpc64".equals(arch)) {
            arch = "ppc64";
        }
        else if ("i386".equals(arch) || "i686".equals(arch)) {
            arch = "x86";
        }
        else if ("x86_64".equals(arch) || "amd64".equals(arch)) {
            arch = "x86-64";
        }
        // Work around OpenJDK mis-reporting os.arch
        // https://bugs.openjdk.java.net/browse/JDK-8073139
        if ("ppc64".equals(arch) && "little".equals(System.getProperty("sun.cpu.endian"))) {
            arch = "ppc64le";
        }
        // Map arm to armel if the binary is running as softfloat build
        if("arm".equals(arch) && platform == Platform.LINUX && isSoftFloat()) {
            arch = "armel";
        }

        return arch;
    }

static {
        String osName = System.getProperty("os.name");
        if (osName.startsWith("Linux")) {
            if ("dalvik".equals(System.getProperty("java.vm.name").toLowerCase())) {
                osType = ANDROID;
                // Native libraries on android must be bundled with the APK
                System.setProperty("jna.nounpack", "true");
            }
            else {
                osType = LINUX;
            }
        }
        else if (osName.startsWith("AIX")) {
            osType = AIX;
        }
        else if (osName.startsWith("Mac") || osName.startsWith("Darwin")) {
            osType = MAC;
        }
        else if (osName.startsWith("Windows CE")) {
            osType = WINDOWSCE;
        }
        else if (osName.startsWith("Windows")) {
            osType = WINDOWS;
        }
        else if (osName.startsWith("Solaris") || osName.startsWith("SunOS")) {
            osType = SOLARIS;
        }
        else if (osName.startsWith("FreeBSD")) {
            osType = FREEBSD;
        }
        else if (osName.startsWith("OpenBSD")) {
            osType = OPENBSD;
        }
        else if (osName.equalsIgnoreCase("gnu")) {
            osType = GNU;
        }
        else if (osName.equalsIgnoreCase("gnu/kfreebsd")) {
            osType = KFREEBSD;
        }
        else if (osName.equalsIgnoreCase("netbsd")) {
            osType = NETBSD;
        }
        else {
            osType = UNSPECIFIED;
        }
}
Layette answered 10/9, 2020 at 13:36 Comment(0)
E
0

You can use a JNI library. This will always work and is independent of the running JVM brand.

Java code:

package org.mytest;

public class NativeBinding
{
    public static native int getRegisterWidth(); // returns 32 or 64
}

And this is the C code:

#include <jni.h>

// will return register width (32 or 64)
extern "C" JNIEXPORT jint JNICALL
Java_org_mytest_NativeBinding_getRegisterWidth(JNIEnv*, jclass)
{
    return sizeof(void*) * 8;
}
Epidiascope answered 12/5, 2022 at 10:15 Comment(0)
M
-1

Under Windows 7 in the "Control Panel" under "Programs | Programs and Features" the 64-bit variants of JRE & JDK are listed with "64-bit" in parentheses (e.g. "Java SE Development Kit 7 Update 65 (64-Bit)"), while for the 32-bit variants the variant is not mentioned in parentheses (e.g. just "Java SE Development Kit 8 Update 60").

Myriammyriameter answered 6/9, 2015 at 20:54 Comment(1)
This does not answer the question in which JVM the app is running 32 bit or 64 bit.Rotunda
S
-3

To get the version of JVM currently running the program

System.out.println(Runtime.class.getPackage().getImplementationVersion());
Stucker answered 12/9, 2017 at 18:54 Comment(3)
Would that report for the JVM or the operating system? You can run a 32-bit JVM on a 64-bit operating system.Claimant
That doesn’t use JMX?Claimant
This returns something like 1.8.0_172 or null on Java 10 and doesn't answer the question anyway.Burgin
B
-3

For Windows, you can check the Java home location. If it contains (x86) it is 32-bit otherwise 64-bit:

public static boolean is32Bit()
{
    val javaHome = System.getProperty("java.home");
    return javaHome.contains("(x86)");
}

public static boolean is64Bit()
{
    return !is32Bit();
}

Example paths:

C:\Program Files (x86)\Java\jdk1.8.0_181\bin\java.exe # 32-bit
C:\Program Files\Java\jdk-10.0.2\bin\java.exe # 64-bit

Why care about a Windows only solution?

If you need to know which bit version you're running on, you're likely fiddling around with native code on Windows so platform-independence is out of the window anyway.

Burgin answered 7/8, 2018 at 21:6 Comment(3)
This is not a good idea, since the path can be chosen individually when installing Java. The absence of "(x86)" in the installation path says absolutely nothing about whether Java is available in 32/64bit version.Tutelary
@André: Fair enough but 99% of all users would install to the default locations anyway. Maybe instead query the java or javac utility and see if certain 32-Bit exclusive command line switches work or the java -version output includes something like 64-BitBurgin
It's very unreliable.Rotunda

© 2022 - 2024 — McMap. All rights reserved.