/* SPDX-FileCopyrightText: 2026 Project Tick
* SPDX-FileContributor: Project Tick
* SPDX-License-Identifier: GPL-3.0-or-later
*
* MeshMC - A Custom Launcher for Minecraft
* Copyright (C) 2026 Project Tick
*
* This program is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program. If not, see .
*
* This file incorporates work covered by the following copyright and
* permission notice:
*
* Copyright 2013-2021 MultiMC Contributors
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include "java/JavaUtils.h"
#include "java/JavaInstallList.h"
#include "FileSystem.h"
#define IBUS "@im=ibus"
JavaUtils::JavaUtils() {}
QString JavaUtils::managedJavaRoot()
{
return FS::PathCombine(QDir::currentPath(), "java");
}
namespace
{
QString javaBinaryName()
{
#if defined(Q_OS_WIN)
return "javaw.exe";
#else
return "java";
#endif
}
void appendUniquePath(QList& paths, const QString& path)
{
if (!path.isEmpty() && !paths.contains(path)) {
paths.append(path);
}
}
void appendExecutablePath(QList& paths, const QString& path)
{
QFileInfo info(path);
if (info.exists() && info.isFile()) {
appendUniquePath(paths, info.absoluteFilePath());
}
}
void appendJavaHome(QList& paths, const QString& homePath)
{
if (homePath.isEmpty()) {
return;
}
QFileInfo info(homePath);
if (info.exists() && info.isFile()) {
appendExecutablePath(paths, info.absoluteFilePath());
return;
}
const auto binaryName = javaBinaryName();
const auto contentsHome = FS::PathCombine(homePath, "Contents", "Home");
appendExecutablePath(paths,
FS::PathCombine(homePath, "bin", binaryName));
appendExecutablePath(
paths, FS::PathCombine(homePath, "jre", "bin", binaryName));
appendExecutablePath(paths,
FS::PathCombine(contentsHome, "bin", binaryName));
appendExecutablePath(
paths, FS::PathCombine(FS::PathCombine(contentsHome, "jre"), "bin",
binaryName));
appendExecutablePath(
paths, FS::PathCombine(FS::PathCombine(homePath, "Contents"),
"Commands", binaryName));
appendExecutablePath(
paths,
FS::PathCombine(
FS::PathCombine(homePath, "libexec/openjdk.jdk/Contents/Home"),
"bin", binaryName));
}
[[maybe_unused]] void scanJavaHomes(QList& paths,
const QString& rootPath)
{
QDir root(rootPath);
if (!root.exists()) {
return;
}
const auto entries = root.entryInfoList(
QDir::Dirs | QDir::NoDotAndDotDot | QDir::Hidden | QDir::System);
for (const auto& entry : entries) {
appendJavaHome(paths, entry.absoluteFilePath());
}
}
[[maybe_unused]] void appendManagedJavaCandidates(QList& paths,
const QString& rootPath)
{
QDir managedDir(rootPath);
if (!managedDir.exists()) {
return;
}
QDirIterator it(rootPath, QStringList() << javaBinaryName(),
QDir::Files, QDirIterator::Subdirectories);
while (it.hasNext()) {
it.next();
const auto candidate = QDir::fromNativeSeparators(it.filePath());
if (candidate.contains("/bin/")) {
appendUniquePath(paths, candidate);
}
}
}
} // namespace
#if defined(Q_OS_LINUX) || defined(Q_OS_FREEBSD)
static QString processLD_LIBRARY_PATH(const QString& LD_LIBRARY_PATH)
{
QDir mmcBin(QCoreApplication::applicationDirPath());
auto items = LD_LIBRARY_PATH.split(':');
QStringList final;
for (auto& item : items) {
QDir test(item);
if (test == mmcBin) {
qDebug() << "Env:LD_LIBRARY_PATH ignoring path" << item;
continue;
}
final.append(item);
}
return final.join(':');
}
#endif
QProcessEnvironment CleanEnviroment()
{
// prepare the process environment
QProcessEnvironment rawenv = QProcessEnvironment::systemEnvironment();
QProcessEnvironment env;
QStringList ignored = {"JAVA_ARGS", "CLASSPATH", "CONFIGPATH",
"JAVA_HOME", "JRE_HOME", "_JAVA_OPTIONS",
"JAVA_OPTIONS", "JAVA_TOOL_OPTIONS"};
for (auto key : rawenv.keys()) {
auto value = rawenv.value(key);
// filter out dangerous java crap
if (ignored.contains(key)) {
qDebug() << "Env: ignoring" << key << value;
continue;
}
// filter MeshMC-related things
if (key.startsWith("QT_")) {
qDebug() << "Env: ignoring" << key << value;
continue;
}
#if defined(Q_OS_LINUX) || defined(Q_OS_FREEBSD)
// Do not pass LD_* variables to java. They were intended for MeshMC
if (key.startsWith("LD_")) {
qDebug() << "Env: ignoring" << key << value;
continue;
}
// Strip IBus
// IBus is a Linux IME framework. For some reason, it breaks MC?
if (key == "XMODIFIERS" && value.contains(IBUS)) {
QString save = value;
value.replace(IBUS, "");
qDebug() << "Env: stripped" << IBUS << "from" << save << ":"
<< value;
}
if (key == "GAME_PRELOAD") {
env.insert("LD_PRELOAD", value);
continue;
}
if (key == "GAME_LIBRARY_PATH") {
env.insert("LD_LIBRARY_PATH", processLD_LIBRARY_PATH(value));
continue;
}
#endif
// qDebug() << "Env: " << key << value;
env.insert(key, value);
}
#ifdef Q_OS_LINUX
// HACK: Workaround for QTBUG42500
if (!env.contains("LD_LIBRARY_PATH")) {
env.insert("LD_LIBRARY_PATH", "");
}
#endif
return env;
}
JavaInstallPtr JavaUtils::MakeJavaPtr(QString path, QString id, QString arch)
{
JavaInstallPtr javaVersion(new JavaInstall());
javaVersion->id = id;
javaVersion->arch = arch;
javaVersion->path = path;
return javaVersion;
}
JavaInstallPtr JavaUtils::GetDefaultJava()
{
JavaInstallPtr javaVersion(new JavaInstall());
javaVersion->id = "java";
javaVersion->arch = "unknown";
#if defined(Q_OS_WIN32)
javaVersion->path = "javaw";
#else
javaVersion->path = "java";
#endif
return javaVersion;
}
#if defined(Q_OS_WIN32)
QList JavaUtils::FindJavaFromRegistryKey(DWORD keyType,
QString keyName,
QString keyJavaDir,
QString subkeySuffix)
{
QList javas;
QString archType = "unknown";
if (keyType == KEY_WOW64_64KEY)
archType = "64";
else if (keyType == KEY_WOW64_32KEY)
archType = "32";
HKEY jreKey;
if (RegOpenKeyExA(HKEY_LOCAL_MACHINE, keyName.toStdString().c_str(), 0,
KEY_READ | keyType | KEY_ENUMERATE_SUB_KEYS,
&jreKey) == ERROR_SUCCESS) {
// Read the current type version from the registry.
// This will be used to find any key that contains the JavaHome value.
char* value = new char[0];
DWORD valueSz = 0;
if (RegQueryValueExA(jreKey, "CurrentVersion", nullptr, nullptr,
(BYTE*)value, &valueSz) == ERROR_MORE_DATA) {
value = new char[valueSz];
RegQueryValueExA(jreKey, "CurrentVersion", nullptr, nullptr,
(BYTE*)value, &valueSz);
}
char subKeyName[255];
DWORD subKeyNameSize, numSubKeys, retCode;
// Get the number of subkeys
RegQueryInfoKeyA(jreKey, nullptr, nullptr, nullptr, &numSubKeys,
nullptr, nullptr, nullptr, nullptr, nullptr, nullptr,
nullptr);
// Iterate until RegEnumKeyEx fails
for (DWORD i = 0; i < numSubKeys; i++) {
subKeyNameSize = 255;
retCode = RegEnumKeyExA(jreKey, i, subKeyName, &subKeyNameSize,
nullptr, nullptr, nullptr, nullptr);
if (retCode != ERROR_SUCCESS)
continue;
// Now open the registry key for the version that we just got.
QString newKeyName = keyName + "\\" + subKeyName + subkeySuffix;
HKEY newKey;
if (RegOpenKeyExA(
HKEY_LOCAL_MACHINE, newKeyName.toStdString().c_str(), 0,
KEY_READ | KEY_WOW64_64KEY, &newKey) != ERROR_SUCCESS)
continue;
// Read the JavaHome value to find where Java is installed.
value = new char[0];
valueSz = 0;
if (RegQueryValueExA(newKey, keyJavaDir.toStdString().c_str(),
nullptr, nullptr, (BYTE*)value,
&valueSz) == ERROR_MORE_DATA) {
value = new char[valueSz];
RegQueryValueExA(newKey, keyJavaDir.toStdString().c_str(),
nullptr, nullptr, (BYTE*)value, &valueSz);
// Now, we construct the version object and add it to the list.
JavaInstallPtr javaVersion(new JavaInstall());
javaVersion->id = subKeyName;
javaVersion->arch = archType;
javaVersion->path = QDir(FS::PathCombine(value, "bin"))
.absoluteFilePath("javaw.exe");
javas.append(javaVersion);
}
RegCloseKey(newKey);
}
RegCloseKey(jreKey);
}
return javas;
}
QList JavaUtils::FindJavaPaths()
{
QList java_candidates;
// Oracle
QList JRE64s = this->FindJavaFromRegistryKey(
KEY_WOW64_64KEY, "SOFTWARE\\JavaSoft\\Java Runtime Environment",
"JavaHome");
QList JDK64s = this->FindJavaFromRegistryKey(
KEY_WOW64_64KEY, "SOFTWARE\\JavaSoft\\Java Development Kit",
"JavaHome");
QList JRE32s = this->FindJavaFromRegistryKey(
KEY_WOW64_32KEY, "SOFTWARE\\JavaSoft\\Java Runtime Environment",
"JavaHome");
QList JDK32s = this->FindJavaFromRegistryKey(
KEY_WOW64_32KEY, "SOFTWARE\\JavaSoft\\Java Development Kit",
"JavaHome");
// Oracle for Java 9 and newer
QList NEWJRE64s = this->FindJavaFromRegistryKey(
KEY_WOW64_64KEY, "SOFTWARE\\JavaSoft\\JRE", "JavaHome");
QList NEWJDK64s = this->FindJavaFromRegistryKey(
KEY_WOW64_64KEY, "SOFTWARE\\JavaSoft\\JDK", "JavaHome");
QList NEWJRE32s = this->FindJavaFromRegistryKey(
KEY_WOW64_32KEY, "SOFTWARE\\JavaSoft\\JRE", "JavaHome");
QList NEWJDK32s = this->FindJavaFromRegistryKey(
KEY_WOW64_32KEY, "SOFTWARE\\JavaSoft\\JDK", "JavaHome");
// AdoptOpenJDK
QList ADOPTOPENJRE32s = this->FindJavaFromRegistryKey(
KEY_WOW64_32KEY, "SOFTWARE\\AdoptOpenJDK\\JRE", "Path",
"\\hotspot\\MSI");
QList ADOPTOPENJRE64s = this->FindJavaFromRegistryKey(
KEY_WOW64_64KEY, "SOFTWARE\\AdoptOpenJDK\\JRE", "Path",
"\\hotspot\\MSI");
QList ADOPTOPENJDK32s = this->FindJavaFromRegistryKey(
KEY_WOW64_32KEY, "SOFTWARE\\AdoptOpenJDK\\JDK", "Path",
"\\hotspot\\MSI");
QList ADOPTOPENJDK64s = this->FindJavaFromRegistryKey(
KEY_WOW64_64KEY, "SOFTWARE\\AdoptOpenJDK\\JDK", "Path",
"\\hotspot\\MSI");
// Eclipse Foundation
QList FOUNDATIONJDK32s = this->FindJavaFromRegistryKey(
KEY_WOW64_32KEY, "SOFTWARE\\Eclipse Foundation\\JDK", "Path",
"\\hotspot\\MSI");
QList FOUNDATIONJDK64s = this->FindJavaFromRegistryKey(
KEY_WOW64_64KEY, "SOFTWARE\\Eclipse Foundation\\JDK", "Path",
"\\hotspot\\MSI");
// Eclipse Adoptium
QList ADOPTIUMJRE32s = this->FindJavaFromRegistryKey(
KEY_WOW64_32KEY, "SOFTWARE\\Eclipse Adoptium\\JRE", "Path",
"\\hotspot\\MSI");
QList ADOPTIUMJRE64s = this->FindJavaFromRegistryKey(
KEY_WOW64_64KEY, "SOFTWARE\\Eclipse Adoptium\\JRE", "Path",
"\\hotspot\\MSI");
QList ADOPTIUMJDK32s = this->FindJavaFromRegistryKey(
KEY_WOW64_32KEY, "SOFTWARE\\Eclipse Adoptium\\JDK", "Path",
"\\hotspot\\MSI");
QList ADOPTIUMJDK64s = this->FindJavaFromRegistryKey(
KEY_WOW64_64KEY, "SOFTWARE\\Eclipse Adoptium\\JDK", "Path",
"\\hotspot\\MSI");
// Microsoft
QList MICROSOFTJDK64s = this->FindJavaFromRegistryKey(
KEY_WOW64_64KEY, "SOFTWARE\\Microsoft\\JDK", "Path", "\\hotspot\\MSI");
// Azul Zulu
QList ZULU64s = this->FindJavaFromRegistryKey(
KEY_WOW64_64KEY, "SOFTWARE\\Azul Systems\\Zulu", "InstallationPath");
QList ZULU32s = this->FindJavaFromRegistryKey(
KEY_WOW64_32KEY, "SOFTWARE\\Azul Systems\\Zulu", "InstallationPath");
// BellSoft Liberica
QList LIBERICA64s = this->FindJavaFromRegistryKey(
KEY_WOW64_64KEY, "SOFTWARE\\BellSoft\\Liberica", "InstallationPath");
QList LIBERICA32s = this->FindJavaFromRegistryKey(
KEY_WOW64_32KEY, "SOFTWARE\\BellSoft\\Liberica", "InstallationPath");
// List x64 before x86
java_candidates.append(JRE64s);
java_candidates.append(NEWJRE64s);
java_candidates.append(ADOPTOPENJRE64s);
java_candidates.append(ADOPTIUMJRE64s);
java_candidates.append(
MakeJavaPtr("C:/Program Files/Java/jre8/bin/javaw.exe"));
java_candidates.append(
MakeJavaPtr("C:/Program Files/Java/jre7/bin/javaw.exe"));
java_candidates.append(
MakeJavaPtr("C:/Program Files/Java/jre6/bin/javaw.exe"));
java_candidates.append(JDK64s);
java_candidates.append(NEWJDK64s);
java_candidates.append(ADOPTOPENJDK64s);
java_candidates.append(FOUNDATIONJDK64s);
java_candidates.append(ADOPTIUMJDK64s);
java_candidates.append(MICROSOFTJDK64s);
java_candidates.append(ZULU64s);
java_candidates.append(LIBERICA64s);
java_candidates.append(JRE32s);
java_candidates.append(NEWJRE32s);
java_candidates.append(ADOPTOPENJRE32s);
java_candidates.append(ADOPTIUMJRE32s);
java_candidates.append(
MakeJavaPtr("C:/Program Files (x86)/Java/jre8/bin/javaw.exe"));
java_candidates.append(
MakeJavaPtr("C:/Program Files (x86)/Java/jre7/bin/javaw.exe"));
java_candidates.append(
MakeJavaPtr("C:/Program Files (x86)/Java/jre6/bin/javaw.exe"));
java_candidates.append(JDK32s);
java_candidates.append(NEWJDK32s);
java_candidates.append(ADOPTOPENJDK32s);
java_candidates.append(FOUNDATIONJDK32s);
java_candidates.append(ADOPTIUMJDK32s);
java_candidates.append(ZULU32s);
java_candidates.append(LIBERICA32s);
// Scan MeshMC's managed java directory
// ({workdir}/java/{vendor}/{version}/bin/javaw.exe)
QString managedJavaDir = JavaUtils::managedJavaRoot();
QDir managedDir(managedJavaDir);
if (managedDir.exists()) {
QDirIterator vendorIt(managedJavaDir,
QDir::Dirs | QDir::NoDotAndDotDot);
while (vendorIt.hasNext()) {
vendorIt.next();
QDirIterator versionIt(vendorIt.filePath(),
QDir::Dirs | QDir::NoDotAndDotDot);
while (versionIt.hasNext()) {
versionIt.next();
QDirIterator binIt(versionIt.filePath(),
QStringList() << "javaw.exe", QDir::Files,
QDirIterator::Subdirectories);
while (binIt.hasNext()) {
binIt.next();
if (binIt.filePath().contains("/bin/")) {
java_candidates.append(MakeJavaPtr(binIt.filePath()));
}
}
}
}
}
java_candidates.append(MakeJavaPtr(this->GetDefaultJava()->path));
QList candidates;
for (JavaInstallPtr java_candidate : java_candidates) {
if (!candidates.contains(java_candidate->path)) {
candidates.append(java_candidate->path);
}
}
return candidates;
}
#elif defined(Q_OS_MAC)
QList JavaUtils::FindJavaPaths()
{
QList javas;
appendUniquePath(javas, this->GetDefaultJava()->path);
appendJavaHome(javas,
QProcessEnvironment::systemEnvironment().value("JAVA_HOME"));
appendJavaHome(javas,
QProcessEnvironment::systemEnvironment().value("JDK_HOME"));
appendJavaHome(javas,
QProcessEnvironment::systemEnvironment().value("JRE_HOME"));
appendExecutablePath(
javas, "/Applications/Xcode.app/Contents/Applications/Application "
"Loader.app/Contents/MacOS/itms/java/bin/java");
appendJavaHome(
javas,
"/Library/Internet Plug-Ins/JavaAppletPlugin.plugin/Contents/Home");
appendExecutablePath(javas, "/System/Library/Frameworks/JavaVM.framework/"
"Versions/Current/Commands/java");
scanJavaHomes(javas, "/Library/Java/JavaVirtualMachines");
scanJavaHomes(javas, "/System/Library/Java/JavaVirtualMachines");
scanJavaHomes(javas, "/opt/homebrew/opt");
scanJavaHomes(javas, "/usr/local/opt");
scanJavaHomes(javas, "/opt/jdk");
scanJavaHomes(javas, "/opt/jdks");
scanJavaHomes(javas, "/opt/java");
scanJavaHomes(javas, "/usr/local/java");
appendManagedJavaCandidates(javas, JavaUtils::managedJavaRoot());
return javas;
}
#elif defined(Q_OS_LINUX)
QList JavaUtils::FindJavaPaths()
{
qDebug() << "Linux Java detection incomplete - defaulting to \"java\"";
QList javas;
javas.append(this->GetDefaultJava()->path);
auto scanJavaDir = [&](const QString& dirPath) {
QDir dir(dirPath);
if (!dir.exists())
return;
auto entries = dir.entryInfoList(QDir::Dirs | QDir::NoDotAndDotDot |
QDir::NoSymLinks);
for (auto& entry : entries) {
QString prefix;
if (entry.isAbsolute()) {
prefix = entry.absoluteFilePath();
} else {
prefix = entry.filePath();
}
javas.append(FS::PathCombine(prefix, "jre/bin/java"));
javas.append(FS::PathCombine(prefix, "bin/java"));
}
};
// oracle RPMs
scanJavaDir("/usr/java");
// general locations used by distro packaging
scanJavaDir("/usr/lib/jvm");
scanJavaDir("/usr/lib64/jvm");
scanJavaDir("/usr/lib32/jvm");
// javas stored in MeshMC's folder (recursive scan for managed
// java/{vendor}/{name}/... structure)
{
QDir javaBaseDir(JavaUtils::managedJavaRoot());
if (javaBaseDir.exists()) {
QDirIterator it(JavaUtils::managedJavaRoot(),
QStringList() << "java", QDir::Files,
QDirIterator::Subdirectories);
while (it.hasNext()) {
it.next();
if (it.filePath().contains("/bin/")) {
javas.append(it.filePath());
}
}
}
}
// manually installed JDKs in /opt
scanJavaDir("/opt/jdk");
scanJavaDir("/opt/jdks");
return javas;
}
#else
QList JavaUtils::FindJavaPaths()
{
qDebug() << "Unknown operating system build - defaulting to \"java\"";
QList javas;
javas.append(this->GetDefaultJava()->path);
return javas;
}
#endif