1 /*
  2  * Copyright (c) 1998, 2025, Oracle and/or its affiliates. All rights reserved.
  3  * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
  4  *
  5  * This code is free software; you can redistribute it and/or modify it
  6  * under the terms of the GNU General Public License version 2 only, as
  7  * published by the Free Software Foundation.
  8  *
  9  * This code is distributed in the hope that it will be useful, but WITHOUT
 10  * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
 11  * FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General Public License
 12  * version 2 for more details (a copy is included in the LICENSE file that
 13  * accompanied this code).
 14  *
 15  * You should have received a copy of the GNU General Public License version
 16  * 2 along with this work; if not, write to the Free Software Foundation,
 17  * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
 18  *
 19  * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
 20  * or visit www.oracle.com if you need additional information or have any
 21  * questions.
 22  *
 23  */
 24 
 25 #include "cds/cdsConfig.hpp"
 26 #include "compiler/compilerDefinitions.hpp"
 27 #include "jvm_io.h"
 28 #include "runtime/arguments.hpp"
 29 #include "runtime/vm_version.hpp"
 30 #include "utilities/globalDefinitions.hpp"
 31 
 32 const char* Abstract_VM_Version::_s_vm_release = Abstract_VM_Version::vm_release();
 33 const char* Abstract_VM_Version::_s_internal_vm_info_string = Abstract_VM_Version::internal_vm_info_string();
 34 
 35 uint64_t Abstract_VM_Version::_features = 0;
 36 const char* Abstract_VM_Version::_features_string = "";
 37 const char* Abstract_VM_Version::_cpu_info_string = "";
 38 uint64_t Abstract_VM_Version::_cpu_features = 0;
 39 
 40 #ifndef SUPPORTS_NATIVE_CX8
 41 bool Abstract_VM_Version::_supports_cx8 = false;
 42 #endif
 43 bool Abstract_VM_Version::_supports_atomic_getset4 = false;
 44 bool Abstract_VM_Version::_supports_atomic_getset8 = false;
 45 bool Abstract_VM_Version::_supports_atomic_getadd4 = false;
 46 bool Abstract_VM_Version::_supports_atomic_getadd8 = false;
 47 unsigned int Abstract_VM_Version::_logical_processors_per_package = 1U;
 48 unsigned int Abstract_VM_Version::_L1_data_cache_line_size = 0;
 49 unsigned int Abstract_VM_Version::_data_cache_line_flush_size = 0;
 50 
 51 VirtualizationType Abstract_VM_Version::_detected_virtualization = NoDetectedVirtualization;
 52 
 53 #ifndef HOTSPOT_VERSION_STRING
 54   #error HOTSPOT_VERSION_STRING must be defined
 55 #endif
 56 
 57 #ifndef VERSION_FEATURE
 58   #error VERSION_FEATURE must be defined
 59 #endif
 60 #ifndef VERSION_INTERIM
 61   #error VERSION_INTERIM must be defined
 62 #endif
 63 #ifndef VERSION_UPDATE
 64   #error VERSION_UPDATE must be defined
 65 #endif
 66 #ifndef VERSION_PATCH
 67   #error VERSION_PATCH must be defined
 68 #endif
 69 #ifndef VERSION_BUILD
 70   #error VERSION_BUILD must be defined
 71 #endif
 72 
 73 #ifndef VERSION_STRING
 74   #error VERSION_STRING must be defined
 75 #endif
 76 
 77 #ifndef DEBUG_LEVEL
 78   #error DEBUG_LEVEL must be defined
 79 #endif
 80 
 81 #ifndef HOTSPOT_BUILD_TIME
 82   #error HOTSPOT_BUILD_TIME must be defined
 83 #endif
 84 
 85 #ifndef JVM_VARIANT
 86   #error JVM_VARIANT must be defined
 87 #endif
 88 
 89 #define VM_RELEASE HOTSPOT_VERSION_STRING
 90 
 91 // HOTSPOT_VERSION_STRING equals the JDK VERSION_STRING (unless overridden
 92 // in a standalone build).
 93 int Abstract_VM_Version::_vm_major_version = VERSION_FEATURE;
 94 int Abstract_VM_Version::_vm_minor_version = VERSION_INTERIM;
 95 int Abstract_VM_Version::_vm_security_version = VERSION_UPDATE;
 96 int Abstract_VM_Version::_vm_patch_version = VERSION_PATCH;
 97 int Abstract_VM_Version::_vm_build_number = VERSION_BUILD;
 98 
 99 #if defined(_LP64)
100   #define VMLP "64-Bit "
101 #else
102   #define VMLP ""
103 #endif
104 
105 #ifndef VMTYPE
106   #if COMPILER1_AND_COMPILER2
107     #define VMTYPE "Server"
108   #else // COMPILER1_AND_COMPILER2
109   #ifdef ZERO
110     #define VMTYPE "Zero"
111   #else // ZERO
112      #define VMTYPE COMPILER1_PRESENT("Client")   \
113                     COMPILER2_PRESENT("Server")
114   #endif // ZERO
115   #endif // COMPILER1_AND_COMPILER2
116 #endif
117 
118 #ifndef HOTSPOT_VM_DISTRO
119   #error HOTSPOT_VM_DISTRO must be defined
120 #endif
121 #define VMNAME HOTSPOT_VM_DISTRO " " VMLP VMTYPE " VM"
122 
123 const char* Abstract_VM_Version::vm_name() {
124   return VMNAME;
125 }
126 
127 #ifndef VENDOR_PADDING
128 # define VENDOR_PADDING 64
129 #endif
130 #ifndef VENDOR
131 # define VENDOR  "Oracle Corporation"
132 #endif
133 
134 static const char vm_vendor_string[sizeof(VENDOR) < VENDOR_PADDING ? VENDOR_PADDING : sizeof(VENDOR)] = VENDOR;
135 
136 const char* Abstract_VM_Version::vm_vendor() {
137   return vm_vendor_string;
138 }
139 
140 
141 // The VM info string should be a constant, but its value cannot be finalized until after VM arguments
142 // have been fully processed. And we want to avoid dynamic memory allocation which will cause ASAN
143 // report error, so we enumerate all the cases by static const string value.
144 const char* Abstract_VM_Version::vm_info_string() {
145   switch (Arguments::mode()) {
146     case Arguments::_int:
147       if (is_vm_statically_linked()) {
148         return CDSConfig::is_using_archive() ? "interpreted mode, static, sharing" : "interpreted mode, static";
149       } else {
150         return CDSConfig::is_using_archive() ? "interpreted mode, sharing" : "interpreted mode";
151       }
152     case Arguments::_mixed:
153       if (is_vm_statically_linked()) {
154         if (CompilationModeFlag::quick_only()) {
155           return CDSConfig::is_using_archive() ? "mixed mode, emulated-client, static, sharing" : "mixed mode, emulated-client, static";
156         } else {
157           return CDSConfig::is_using_archive() ? "mixed mode, static, sharing" : "mixed mode, static";
158          }
159       } else {
160         if (CompilationModeFlag::quick_only()) {
161           return CDSConfig::is_using_archive() ? "mixed mode, emulated-client, sharing" : "mixed mode, emulated-client";
162         } else {
163           return CDSConfig::is_using_archive() ? "mixed mode, sharing" : "mixed mode";
164         }
165       }
166     case Arguments::_comp:
167       if (is_vm_statically_linked()) {
168         if (CompilationModeFlag::quick_only()) {
169           return CDSConfig::is_using_archive() ? "compiled mode, emulated-client, static, sharing" : "compiled mode, emulated-client, static";
170         }
171         return CDSConfig::is_using_archive() ? "compiled mode, static, sharing" : "compiled mode, static";
172       } else {
173         if (CompilationModeFlag::quick_only()) {
174           return CDSConfig::is_using_archive() ? "compiled mode, emulated-client, sharing" : "compiled mode, emulated-client";
175         }
176         return CDSConfig::is_using_archive() ? "compiled mode, sharing" : "compiled mode";
177       }
178   }
179   ShouldNotReachHere();
180   return "";
181 }
182 
183 // NOTE: do *not* use stringStream. this function is called by
184 //       fatal error handler. if the crash is in native thread,
185 //       stringStream cannot get resource allocated and will SEGV.
186 const char* Abstract_VM_Version::vm_release() {
187   return VM_RELEASE;
188 }
189 
190 #define OS       LINUX_ONLY("linux")             \
191                  WINDOWS_ONLY("windows")         \
192                  AIX_ONLY("aix")                 \
193                  BSD_ONLY("bsd")
194 
195 #ifndef CPU
196 #ifdef ZERO
197 #define CPU      ZERO_LIBARCH
198 #elif defined(PPC64)
199 #if defined(VM_LITTLE_ENDIAN)
200 #define CPU      "ppc64le"
201 #else
202 #define CPU      "ppc64"
203 #endif // PPC64
204 #else
205 #define CPU      AARCH64_ONLY("aarch64")         \
206                  AMD64_ONLY("amd64")             \
207                  IA32_ONLY("x86")                \
208                  S390_ONLY("s390")               \
209                  RISCV64_ONLY("riscv64")
210 #endif // !ZERO
211 #endif // !CPU
212 
213 const char *Abstract_VM_Version::vm_platform_string() {
214   return OS "-" CPU;
215 }
216 
217 const char* Abstract_VM_Version::vm_variant() {
218   return JVM_VARIANT;
219 }
220 
221 const char* Abstract_VM_Version::internal_vm_info_string() {
222   #ifndef HOTSPOT_BUILD_COMPILER
223     #ifdef _MSC_VER
224       #if _MSC_VER == 1911
225         #define HOTSPOT_BUILD_COMPILER "MS VC++ 15.3 (VS2017)"
226       #elif _MSC_VER == 1912
227         #define HOTSPOT_BUILD_COMPILER "MS VC++ 15.5 (VS2017)"
228       #elif _MSC_VER == 1913
229         #define HOTSPOT_BUILD_COMPILER "MS VC++ 15.6 (VS2017)"
230       #elif _MSC_VER == 1914
231         #define HOTSPOT_BUILD_COMPILER "MS VC++ 15.7 (VS2017)"
232       #elif _MSC_VER == 1915
233         #define HOTSPOT_BUILD_COMPILER "MS VC++ 15.8 (VS2017)"
234       #elif _MSC_VER == 1916
235         #define HOTSPOT_BUILD_COMPILER "MS VC++ 15.9 (VS2017)"
236       #elif _MSC_VER == 1920
237         #define HOTSPOT_BUILD_COMPILER "MS VC++ 16.0 (VS2019)"
238       #elif _MSC_VER == 1921
239         #define HOTSPOT_BUILD_COMPILER "MS VC++ 16.1 (VS2019)"
240       #elif _MSC_VER == 1922
241         #define HOTSPOT_BUILD_COMPILER "MS VC++ 16.2 (VS2019)"
242       #elif _MSC_VER == 1923
243         #define HOTSPOT_BUILD_COMPILER "MS VC++ 16.3 (VS2019)"
244       #elif _MSC_VER == 1924
245         #define HOTSPOT_BUILD_COMPILER "MS VC++ 16.4 (VS2019)"
246       #elif _MSC_VER == 1925
247         #define HOTSPOT_BUILD_COMPILER "MS VC++ 16.5 (VS2019)"
248       #elif _MSC_VER == 1926
249         #define HOTSPOT_BUILD_COMPILER "MS VC++ 16.6 (VS2019)"
250       #elif _MSC_VER == 1927
251         #define HOTSPOT_BUILD_COMPILER "MS VC++ 16.7 (VS2019)"
252       #elif _MSC_VER == 1928
253         #define HOTSPOT_BUILD_COMPILER "MS VC++ 16.8 / 16.9 (VS2019)"
254       #elif _MSC_VER == 1929
255         #define HOTSPOT_BUILD_COMPILER "MS VC++ 16.10 / 16.11 (VS2019)"
256       #elif _MSC_VER == 1930
257         #define HOTSPOT_BUILD_COMPILER "MS VC++ 17.0 (VS2022)"
258       #elif _MSC_VER == 1931
259         #define HOTSPOT_BUILD_COMPILER "MS VC++ 17.1 (VS2022)"
260       #elif _MSC_VER == 1932
261         #define HOTSPOT_BUILD_COMPILER "MS VC++ 17.2 (VS2022)"
262       #elif _MSC_VER == 1933
263         #define HOTSPOT_BUILD_COMPILER "MS VC++ 17.3 (VS2022)"
264       #elif _MSC_VER == 1934
265         #define HOTSPOT_BUILD_COMPILER "MS VC++ 17.4 (VS2022)"
266       #elif _MSC_VER == 1935
267         #define HOTSPOT_BUILD_COMPILER "MS VC++ 17.5 (VS2022)"
268       #elif _MSC_VER == 1936
269         #define HOTSPOT_BUILD_COMPILER "MS VC++ 17.6 (VS2022)"
270       #elif _MSC_VER == 1937
271         #define HOTSPOT_BUILD_COMPILER "MS VC++ 17.7 (VS2022)"
272       #elif _MSC_VER == 1938
273         #define HOTSPOT_BUILD_COMPILER "MS VC++ 17.8 (VS2022)"
274       #else
275         #define HOTSPOT_BUILD_COMPILER "unknown MS VC++:" XSTR(_MSC_VER)
276       #endif
277     #elif defined(__clang_version__)
278         #define HOTSPOT_BUILD_COMPILER "clang " __VERSION__
279     #elif defined(__GNUC__)
280         #define HOTSPOT_BUILD_COMPILER "gcc " __VERSION__
281     #else
282       #define HOTSPOT_BUILD_COMPILER "unknown compiler"
283     #endif
284   #endif
285 
286   #ifndef FLOAT_ARCH
287     #if defined(__SOFTFP__)
288       #define FLOAT_ARCH_STR "-sflt"
289     #else
290       #define FLOAT_ARCH_STR ""
291     #endif
292   #else
293     #define FLOAT_ARCH_STR XSTR(FLOAT_ARCH)
294   #endif
295 
296   #ifdef MUSL_LIBC
297     #define LIBC_STR "-" XSTR(LIBC)
298   #else
299     #define LIBC_STR ""
300   #endif
301 
302   #define INTERNAL_VERSION_SUFFIX VM_RELEASE ")" \
303          " for " OS "-" CPU FLOAT_ARCH_STR LIBC_STR \
304          " JRE (" VERSION_STRING "), built on " HOTSPOT_BUILD_TIME \
305          " with " HOTSPOT_BUILD_COMPILER
306 
307   return strcmp(DEBUG_LEVEL, "release") == 0
308       ? VMNAME " (" INTERNAL_VERSION_SUFFIX
309       : VMNAME " (" DEBUG_LEVEL " " INTERNAL_VERSION_SUFFIX;
310 }
311 
312 const char *Abstract_VM_Version::jdk_debug_level() {
313   return DEBUG_LEVEL;
314 }
315 
316 const char *Abstract_VM_Version::printable_jdk_debug_level() {
317   // Debug level is not printed for "release" builds
318   return strcmp(DEBUG_LEVEL, "release") == 0 ? "" : DEBUG_LEVEL " ";
319 }
320 
321 unsigned int Abstract_VM_Version::jvm_version() {
322   return ((Abstract_VM_Version::vm_major_version() & 0xFF) << 24) |
323          ((Abstract_VM_Version::vm_minor_version() & 0xFF) << 16) |
324          ((Abstract_VM_Version::vm_security_version() & 0xFF) << 8) |
325          (Abstract_VM_Version::vm_build_number() & 0xFF);
326 }
327 
328 bool Abstract_VM_Version::print_matching_lines_from_file(const char* filename, outputStream* st, const char* keywords_to_match[]) {
329   char line[500];
330   FILE* fp = os::fopen(filename, "r");
331   if (fp == nullptr) {
332     return false;
333   }
334 
335   st->print_cr("Virtualization information:");
336   while (fgets(line, sizeof(line), fp) != nullptr) {
337     int i = 0;
338     while (keywords_to_match[i] != nullptr) {
339       if (strncmp(line, keywords_to_match[i], strlen(keywords_to_match[i])) == 0) {
340         st->print("%s", line);
341         break;
342       }
343       i++;
344     }
345   }
346   fclose(fp);
347   return true;
348 }
349 
350 // Abstract_VM_Version statics
351 int   Abstract_VM_Version::_no_of_threads = 0;
352 int   Abstract_VM_Version::_no_of_cores = 0;
353 int   Abstract_VM_Version::_no_of_sockets = 0;
354 bool  Abstract_VM_Version::_initialized = false;
355 char  Abstract_VM_Version::_cpu_name[CPU_TYPE_DESC_BUF_SIZE] = {0};
356 char  Abstract_VM_Version::_cpu_desc[CPU_DETAILED_DESC_BUF_SIZE] = {0};
357 
358 int Abstract_VM_Version::number_of_threads(void) {
359   assert(_initialized, "should be initialized");
360   return _no_of_threads;
361 }
362 
363 int Abstract_VM_Version::number_of_cores(void) {
364   assert(_initialized, "should be initialized");
365   return _no_of_cores;
366 }
367 
368 int Abstract_VM_Version::number_of_sockets(void) {
369   assert(_initialized, "should be initialized");
370   return _no_of_sockets;
371 }
372 
373 const char* Abstract_VM_Version::cpu_name(void) {
374   assert(_initialized, "should be initialized");
375   char* tmp = NEW_C_HEAP_ARRAY_RETURN_NULL(char, CPU_TYPE_DESC_BUF_SIZE, mtTracing);
376   if (nullptr == tmp) {
377     return nullptr;
378   }
379   strncpy(tmp, _cpu_name, CPU_TYPE_DESC_BUF_SIZE);
380   return tmp;
381 }
382 
383 const char* Abstract_VM_Version::cpu_description(void) {
384   assert(_initialized, "should be initialized");
385   char* tmp = NEW_C_HEAP_ARRAY_RETURN_NULL(char, CPU_DETAILED_DESC_BUF_SIZE, mtTracing);
386   if (nullptr == tmp) {
387     return nullptr;
388   }
389   strncpy(tmp, _cpu_desc, CPU_DETAILED_DESC_BUF_SIZE);
390   return tmp;
391 }