2014-02-09 22:10:30 -03:00
|
|
|
import os
|
|
|
|
import sys
|
|
|
|
import platform
|
2022-06-01 08:54:08 +01:00
|
|
|
import subprocess
|
2014-02-09 22:10:30 -03:00
|
|
|
|
2022-08-23 22:21:46 +09:00
|
|
|
from typing import TYPE_CHECKING
|
|
|
|
|
|
|
|
if TYPE_CHECKING:
|
|
|
|
from SCons import Environment
|
|
|
|
|
2016-10-30 19:05:14 +01:00
|
|
|
|
2014-02-09 22:10:30 -03:00
|
|
|
def get_name():
|
2016-10-30 18:44:57 +01:00
|
|
|
return "Android"
|
2014-02-09 22:10:30 -03:00
|
|
|
|
2016-10-30 19:05:14 +01:00
|
|
|
|
2014-02-09 22:10:30 -03:00
|
|
|
def can_build():
|
2022-06-01 08:54:08 +01:00
|
|
|
return os.path.exists(get_env_android_sdk_root())
|
2017-12-18 15:39:09 +00:00
|
|
|
|
|
|
|
|
2014-02-09 22:10:30 -03:00
|
|
|
def get_opts():
|
2023-03-01 14:09:30 -08:00
|
|
|
from SCons.Variables import BoolVariable
|
|
|
|
|
2016-10-30 18:44:57 +01:00
|
|
|
return [
|
2022-06-01 08:54:08 +01:00
|
|
|
("ANDROID_SDK_ROOT", "Path to the Android SDK", get_env_android_sdk_root()),
|
2022-10-18 21:39:43 -07:00
|
|
|
(
|
|
|
|
"ndk_platform",
|
|
|
|
'Target platform (android-<api>, e.g. "android-' + str(get_min_target_api()) + '")',
|
|
|
|
"android-" + str(get_min_target_api()),
|
|
|
|
),
|
2023-03-01 14:09:30 -08:00
|
|
|
BoolVariable("store_release", "Editor build for Google Play Store (for official builds only)", False),
|
2016-10-30 18:44:57 +01:00
|
|
|
]
|
2014-02-09 22:10:30 -03:00
|
|
|
|
2016-03-20 17:22:48 +08:00
|
|
|
|
2023-04-19 12:23:22 +03:00
|
|
|
def get_doc_classes():
|
|
|
|
return [
|
|
|
|
"EditorExportPlatformAndroid",
|
|
|
|
]
|
|
|
|
|
|
|
|
|
|
|
|
def get_doc_path():
|
|
|
|
return "doc_classes"
|
|
|
|
|
|
|
|
|
2021-01-05 13:40:42 -08:00
|
|
|
# Return the ANDROID_SDK_ROOT environment variable.
|
2022-06-01 08:54:08 +01:00
|
|
|
def get_env_android_sdk_root():
|
|
|
|
return os.environ.get("ANDROID_SDK_ROOT", -1)
|
2021-01-05 13:40:42 -08:00
|
|
|
|
|
|
|
|
2022-06-01 08:54:08 +01:00
|
|
|
def get_min_sdk_version(platform):
|
|
|
|
return int(platform.split("-")[1])
|
|
|
|
|
|
|
|
|
|
|
|
def get_android_ndk_root(env):
|
|
|
|
return env["ANDROID_SDK_ROOT"] + "/ndk/" + get_ndk_version()
|
|
|
|
|
|
|
|
|
|
|
|
# This is kept in sync with the value in 'platform/android/java/app/config.gradle'.
|
|
|
|
def get_ndk_version():
|
|
|
|
return "23.2.8568313"
|
2021-01-05 13:40:42 -08:00
|
|
|
|
|
|
|
|
2022-10-18 21:39:43 -07:00
|
|
|
# This is kept in sync with the value in 'platform/android/java/app/config.gradle'.
|
|
|
|
def get_min_target_api():
|
|
|
|
return 21
|
|
|
|
|
|
|
|
|
2014-02-09 22:10:30 -03:00
|
|
|
def get_flags():
|
2016-10-30 18:44:57 +01:00
|
|
|
return [
|
2021-12-15 17:38:10 -08:00
|
|
|
("arch", "arm64"), # Default for convenience.
|
SCons: Unify tools/target build type configuration
Implements https://github.com/godotengine/godot-proposals/issues/3371.
New `target` presets
====================
The `tools` option is removed and `target` changes to use three new presets,
which match the builds users are familiar with. These targets control the
default optimization level and enable editor-specific and debugging code:
- `editor`: Replaces `tools=yes target=release_debug`.
* Defines: `TOOLS_ENABLED`, `DEBUG_ENABLED`, `-O2`/`/O2`
- `template_debug`: Replaces `tools=no target=release_debug`.
* Defines: `DEBUG_ENABLED`, `-O2`/`/O2`
- `template_release`: Replaces `tools=no target=release`.
* Defines: `-O3`/`/O2`
New `dev_build` option
======================
The previous `target=debug` is now replaced by a separate `dev_build=yes`
option, which can be used in combination with either of the three targets,
and changes the following:
- `dev_build`: Defines `DEV_ENABLED`, disables optimization (`-O0`/`/0d`),
enables generating debug symbols, does not define `NDEBUG` so `assert()`
works in thirdparty libraries, adds a `.dev` suffix to the binary name.
Note: Unlike previously, `dev_build` defaults to off so that users who
compile Godot from source get an optimized and small build by default.
Engine contributors should now set `dev_build=yes` in their build scripts or
IDE configuration manually.
Changed binary names
====================
The name of generated binaries and object files are changed too, to follow
this format:
`godot.<platform>.<target>[.dev][.double].<arch>[.<extra_suffix>][.<ext>]`
For example:
- `godot.linuxbsd.editor.dev.arm64`
- `godot.windows.template_release.double.x86_64.mono.exe`
Be sure to update your links/scripts/IDE config accordingly.
More flexible `optimize` and `debug_symbols` options
====================================================
The optimization level and whether to generate debug symbols can be further
specified with the `optimize` and `debug_symbols` options. So the default
values listed above for the various `target` and `dev_build` combinations
are indicative and can be replaced when compiling, e.g.:
`scons p=linuxbsd target=template_debug dev_build=yes optimize=debug`
will make a "debug" export template with dev-only code enabled, `-Og`
optimization level for GCC/Clang, and debug symbols. Perfect for debugging
complex crashes at runtime in an exported project.
2022-09-22 08:28:55 +02:00
|
|
|
("target", "template_debug"),
|
SCons: Re-disable exceptions for Android, iOS, and Web
We made a mistake when cherry-picking #80612 with 269b115d9cff1ad1bf1cd8675cd26c8a24065c89,
where the global flag was defaulted to false to preserve the 4.1-stable behavior for desktop
platforms, but we forgot that the refactoring removed the force disabling of exceptions for
Android, iOS, and Web.
This reintroduces this behavior so it should be back to the same as in 4.1/4.1.1, and the
export templates should get back to their original size.
Only difference, the old code used to keep exceptions for the Web editor, but I see no reason
for it, so I disable them like with the templates.
2023-11-01 22:14:49 +01:00
|
|
|
("disable_exceptions", True), # Reduces size.
|
2016-10-30 18:44:57 +01:00
|
|
|
]
|
2014-02-09 22:10:30 -03:00
|
|
|
|
|
|
|
|
2022-06-01 08:54:08 +01:00
|
|
|
# Check if Android NDK version is installed
|
|
|
|
# If not, install it.
|
2021-01-05 13:40:42 -08:00
|
|
|
def install_ndk_if_needed(env):
|
|
|
|
print("Checking for Android NDK...")
|
2022-06-01 08:54:08 +01:00
|
|
|
sdk_root = env["ANDROID_SDK_ROOT"]
|
|
|
|
if not os.path.exists(get_android_ndk_root(env)):
|
2021-01-05 13:40:42 -08:00
|
|
|
extension = ".bat" if os.name == "nt" else ""
|
2022-06-01 08:54:08 +01:00
|
|
|
sdkmanager = sdk_root + "/cmdline-tools/latest/bin/sdkmanager" + extension
|
|
|
|
if os.path.exists(sdkmanager):
|
|
|
|
# Install the Android NDK
|
|
|
|
print("Installing Android NDK...")
|
|
|
|
ndk_download_args = "ndk;" + get_ndk_version()
|
|
|
|
subprocess.check_call([sdkmanager, ndk_download_args])
|
|
|
|
else:
|
|
|
|
print("Cannot find " + sdkmanager)
|
|
|
|
print(
|
|
|
|
"Please ensure ANDROID_SDK_ROOT is correct and cmdline-tools are installed, or install NDK version "
|
|
|
|
+ get_ndk_version()
|
|
|
|
+ " manually."
|
|
|
|
)
|
|
|
|
sys.exit()
|
|
|
|
env["ANDROID_NDK_ROOT"] = get_android_ndk_root(env)
|
2021-01-05 13:40:42 -08:00
|
|
|
|
|
|
|
|
2022-08-23 22:21:46 +09:00
|
|
|
def configure(env: "Environment"):
|
2021-12-15 17:38:10 -08:00
|
|
|
# Validate arch.
|
|
|
|
supported_arches = ["x86_32", "x86_64", "arm32", "arm64"]
|
|
|
|
if env["arch"] not in supported_arches:
|
|
|
|
print(
|
|
|
|
'Unsupported CPU architecture "%s" for Android. Supported architectures are: %s.'
|
|
|
|
% (env["arch"], ", ".join(supported_arches))
|
|
|
|
)
|
|
|
|
sys.exit()
|
|
|
|
|
2022-10-18 21:39:43 -07:00
|
|
|
if get_min_sdk_version(env["ndk_platform"]) < get_min_target_api():
|
|
|
|
print(
|
|
|
|
"WARNING: minimum supported Android target api is %d. Forcing target api %d."
|
|
|
|
% (get_min_target_api(), get_min_target_api())
|
|
|
|
)
|
|
|
|
env["ndk_platform"] = "android-" + str(get_min_target_api())
|
|
|
|
|
2021-01-05 13:40:42 -08:00
|
|
|
install_ndk_if_needed(env)
|
2022-06-01 08:54:08 +01:00
|
|
|
ndk_root = env["ANDROID_NDK_ROOT"]
|
2016-10-30 18:44:57 +01:00
|
|
|
|
2019-08-27 11:16:33 +02:00
|
|
|
# Architecture
|
2016-10-30 18:44:57 +01:00
|
|
|
|
2021-12-15 17:38:10 -08:00
|
|
|
if env["arch"] == "arm32":
|
2022-06-01 08:54:08 +01:00
|
|
|
target_triple = "armv7a-linux-androideabi"
|
2021-12-15 17:38:10 -08:00
|
|
|
elif env["arch"] == "arm64":
|
2022-06-01 08:54:08 +01:00
|
|
|
target_triple = "aarch64-linux-android"
|
2021-12-15 17:38:10 -08:00
|
|
|
elif env["arch"] == "x86_32":
|
2022-06-01 08:54:08 +01:00
|
|
|
target_triple = "i686-linux-android"
|
2021-12-15 17:38:10 -08:00
|
|
|
elif env["arch"] == "x86_64":
|
2022-06-01 08:54:08 +01:00
|
|
|
target_triple = "x86_64-linux-android"
|
|
|
|
|
|
|
|
target_option = ["-target", target_triple + str(get_min_sdk_version(env["ndk_platform"]))]
|
2022-07-07 13:26:31 +03:00
|
|
|
env.Append(ASFLAGS=[target_option, "-c"])
|
2022-06-01 08:54:08 +01:00
|
|
|
env.Append(CCFLAGS=target_option)
|
|
|
|
env.Append(LINKFLAGS=target_option)
|
2016-10-30 18:44:57 +01:00
|
|
|
|
2022-07-21 15:15:54 +02:00
|
|
|
# LTO
|
2022-09-13 17:01:47 +02:00
|
|
|
|
|
|
|
if env["lto"] == "auto": # LTO benefits for Android (size, performance) haven't been clearly established yet.
|
|
|
|
env["lto"] = "none"
|
|
|
|
|
2022-07-21 15:15:54 +02:00
|
|
|
if env["lto"] != "none":
|
|
|
|
if env["lto"] == "thin":
|
|
|
|
env.Append(CCFLAGS=["-flto=thin"])
|
|
|
|
env.Append(LINKFLAGS=["-flto=thin"])
|
|
|
|
else:
|
|
|
|
env.Append(CCFLAGS=["-flto"])
|
|
|
|
env.Append(LINKFLAGS=["-flto"])
|
|
|
|
|
2019-08-27 11:16:33 +02:00
|
|
|
# Compiler configuration
|
2017-06-30 19:21:38 +02:00
|
|
|
|
2020-03-30 08:28:32 +02:00
|
|
|
env["SHLIBSUFFIX"] = ".so"
|
2017-06-30 19:21:38 +02:00
|
|
|
|
2020-03-30 08:28:32 +02:00
|
|
|
if env["PLATFORM"] == "win32":
|
2017-06-30 19:21:38 +02:00
|
|
|
env.use_windows_spawn_fix()
|
|
|
|
|
2020-03-30 08:28:32 +02:00
|
|
|
if sys.platform.startswith("linux"):
|
2016-11-13 23:54:06 +01:00
|
|
|
host_subpath = "linux-x86_64"
|
2020-03-30 08:28:32 +02:00
|
|
|
elif sys.platform.startswith("darwin"):
|
2016-11-02 10:54:51 +01:00
|
|
|
host_subpath = "darwin-x86_64"
|
2020-03-30 08:28:32 +02:00
|
|
|
elif sys.platform.startswith("win"):
|
|
|
|
if platform.machine().endswith("64"):
|
2016-11-02 10:54:51 +01:00
|
|
|
host_subpath = "windows-x86_64"
|
2016-10-30 18:44:57 +01:00
|
|
|
else:
|
2016-11-13 23:54:06 +01:00
|
|
|
host_subpath = "windows"
|
2016-10-30 18:57:40 +01:00
|
|
|
|
2022-06-01 08:54:08 +01:00
|
|
|
toolchain_path = ndk_root + "/toolchains/llvm/prebuilt/" + host_subpath
|
|
|
|
compiler_path = toolchain_path + "/bin"
|
2016-10-30 18:44:57 +01:00
|
|
|
|
2022-06-01 08:54:08 +01:00
|
|
|
env["CC"] = compiler_path + "/clang"
|
|
|
|
env["CXX"] = compiler_path + "/clang++"
|
|
|
|
env["AR"] = compiler_path + "/llvm-ar"
|
|
|
|
env["RANLIB"] = compiler_path + "/llvm-ranlib"
|
2022-07-07 13:26:31 +03:00
|
|
|
env["AS"] = compiler_path + "/clang"
|
2019-07-30 15:33:24 +02:00
|
|
|
|
2020-03-30 08:28:32 +02:00
|
|
|
env.Append(
|
2020-07-26 19:38:10 +02:00
|
|
|
CCFLAGS=(
|
2022-06-01 08:54:08 +01:00
|
|
|
"-fpic -ffunction-sections -funwind-tables -fstack-protector-strong -fvisibility=hidden -fno-strict-aliasing".split()
|
2020-07-26 19:38:10 +02:00
|
|
|
)
|
2020-03-30 08:28:32 +02:00
|
|
|
)
|
2014-10-07 01:31:49 -03:00
|
|
|
|
2022-06-01 08:54:08 +01:00
|
|
|
if get_min_sdk_version(env["ndk_platform"]) >= 24:
|
2019-03-26 18:51:13 +01:00
|
|
|
env.Append(CPPDEFINES=[("_FILE_OFFSET_BITS", 64)])
|
|
|
|
|
2021-12-15 17:38:10 -08:00
|
|
|
if env["arch"] == "x86_32":
|
2022-06-01 08:54:08 +01:00
|
|
|
# The NDK adds this if targeting API < 24, so we can drop it when Godot targets it at least
|
2020-03-30 08:28:32 +02:00
|
|
|
env.Append(CCFLAGS=["-mstackrealign"])
|
2021-12-15 17:38:10 -08:00
|
|
|
elif env["arch"] == "arm32":
|
2020-03-30 08:28:32 +02:00
|
|
|
env.Append(CCFLAGS="-march=armv7-a -mfloat-abi=softfp".split())
|
|
|
|
env.Append(CPPDEFINES=["__ARM_ARCH_7__", "__ARM_ARCH_7A__"])
|
2021-10-12 15:27:30 +02:00
|
|
|
env.Append(CPPDEFINES=["__ARM_NEON__"])
|
2021-12-15 17:38:10 -08:00
|
|
|
elif env["arch"] == "arm64":
|
2020-03-30 08:28:32 +02:00
|
|
|
env.Append(CCFLAGS=["-mfix-cortex-a53-835769"])
|
|
|
|
env.Append(CPPDEFINES=["__ARM_ARCH_8A__"])
|
2017-07-25 12:28:31 +02:00
|
|
|
|
2019-08-27 11:16:33 +02:00
|
|
|
# Link flags
|
2019-07-30 15:49:31 +02:00
|
|
|
|
2022-06-01 08:54:08 +01:00
|
|
|
env.Append(LINKFLAGS="-Wl,--gc-sections -Wl,--no-undefined -Wl,-z,now".split())
|
|
|
|
env.Append(LINKFLAGS="-Wl,-soname,libgodot_android.so")
|
2020-03-30 08:28:32 +02:00
|
|
|
|
|
|
|
env.Prepend(CPPPATH=["#platform/android"])
|
2022-10-03 11:43:20 +02:00
|
|
|
env.Append(CPPDEFINES=["ANDROID_ENABLED", "UNIX_ENABLED"])
|
2022-11-11 15:30:06 -08:00
|
|
|
env.Append(LIBS=["OpenSLES", "EGL", "android", "log", "z", "dl"])
|
2021-08-12 14:24:54 +03:00
|
|
|
|
|
|
|
if env["vulkan"]:
|
|
|
|
env.Append(CPPDEFINES=["VULKAN_ENABLED"])
|
|
|
|
if not env["use_volk"]:
|
|
|
|
env.Append(LIBS=["vulkan"])
|
2022-11-11 15:30:06 -08:00
|
|
|
|
|
|
|
if env["opengl3"]:
|
|
|
|
env.Append(CPPDEFINES=["GLES3_ENABLED"])
|
|
|
|
env.Append(LIBS=["GLESv3"])
|