axmol/tools/tolua/genbindings.py

281 lines
10 KiB
Python

#!/usr/bin/python
# This script is used to generate luabinding glue codes.
# Android ndk version must be ndk-r9b.
import argparse
import sys
import os, os.path
import shutil
import subprocess
import re
from contextlib import contextmanager
g_ndk_root = None
def _check_ndk_root_env():
''' Checking the environment ANDROID_NDK, which will be used for building
'''
try:
ANDROID_NDK = None
sdkRoot = os.environ.get('ANDROID_HOME', None)
for _, ndkVers, _ in os.walk("{0}{1}ndk".format(sdkRoot, os.path.sep)):
for ndkVer in ndkVers:
if (ndkVer == '23.2.8568313'):
ANDROID_NDK = "{0}{1}ndk{1}{2}".format(sdkRoot, os.path.sep, ndkVer)
break
break
if ANDROID_NDK == None:
ANDROID_NDK = os.environ.get('ANDROID_NDK', None)
except:
print('Exception occurred when check_ndk_root_env!')
return ANDROID_NDK
def _check_python_bin_env():
''' Checking the environment PYTHON_BIN, which will be used for building
'''
try:
PYTHON_BIN = os.environ['PYTHON_BIN']
except Exception:
print("PYTHON_BIN not defined, use current python.")
PYTHON_BIN = sys.executable
return PYTHON_BIN
def _find_first_file_in_dir(dir, fn):
if os.path.isfile(dir):
if os.path.basename(dir) == fn:
return os.path.join(os.path.dirname(dir), fn)
else :
return None
elif os.path.isdir(dir):
for subdir in os.listdir(dir):
searchPath = _find_first_file_in_dir(os.path.join(dir, subdir), fn)
if searchPath is not None:
return searchPath
else:
return None
def _find_all_files_match(dir, cond, all):
if cond(dir):
all.append(dir)
elif os.path.isdir(dir):
for subdir in os.listdir(dir):
_find_all_files_match(os.path.join(dir, subdir), cond, all)
def _find_toolchain_include_path():
global g_ndk_root
'''
Search gcc prebuilt include path
for instance: "$ANDROID_NDK/toolchains/arm-linux-androideabi-4.9/prebuilt/windows-x86_64/lib/gcc/arm-linux-androideabi/4.9.x/include"
'''
foundFiles = []
_find_all_files_match(os.path.join(g_ndk_root, "toolchains"), lambda x : os.path.basename(x) == "stdarg.h" and "arm-linux-androideabi" in x , foundFiles)
if len(foundFiles) == 0:
return ""
else:
return "-I" + os.path.dirname(foundFiles[0])
def _find_llvm_include_path():
'''
Search llvm prebuilt include path.
for instance: "$ANDROID_NDK/toolchains/llvm/prebuilt/windows-x86_64/lib64/clang/6.0.2/include"
'''
global g_ndk_root
versionFile = _find_first_file_in_dir(g_ndk_root, "AndroidVersion.txt")
if versionFile is None:
return ""
versionDir = os.path.dirname(versionFile)
includeDir = _find_first_file_in_dir(versionDir, "stdarg.h")
llvmIncludePath = os.path.dirname(includeDir)
return "-I"+llvmIncludePath
def _defaultIncludePath():
'''default include path for libclang, llvm & gcc include path
'''
llvmInclude = _find_llvm_include_path()
toolchainInclude = _find_toolchain_include_path()
exactIncludes = llvmInclude + " " + toolchainInclude
return exactIncludes
class CmdError(Exception):
pass
@contextmanager
def _pushd(newDir):
previousDir = os.getcwd()
os.chdir(newDir)
yield
os.chdir(previousDir)
def _run_cmd(command):
ret = subprocess.call(command, shell=True)
if ret != 0:
message = "Error running command"
raise CmdError(message)
def main():
global g_ndk_root
cur_platform= '??'
llvm_path = '??'
if (g_ndk_root == None or not os.path.isdir(g_ndk_root)):
g_ndk_root = _check_ndk_root_env()
if not os.path.isdir(g_ndk_root):
print("The ndk-r23c root not specified, please specifiy via --ndk_root '/path/to/ndk'")
sys.exit(1)
# del the " in the path
g_ndk_root = re.sub(r"\"", "", g_ndk_root)
python_bin = _check_python_bin_env()
platform = sys.platform
if platform == 'win32':
cur_platform = 'windows'
elif platform == 'darwin':
cur_platform = platform
elif 'linux' in platform:
cur_platform = 'linux'
else:
print('Your platform is not supported!')
sys.exit(1)
x86_llvm_path = ""
x64_llvm_path = os.path.abspath(os.path.join(g_ndk_root, 'toolchains/llvm/prebuilt', '%s-%s' % (cur_platform, 'x86_64')))
if not os.path.exists(x64_llvm_path):
x86_llvm_path = os.path.abspath(os.path.join(g_ndk_root, 'toolchains/llvm/prebuilt', '%s' % (cur_platform)))
if not os.path.exists(x86_llvm_path):
x86_llvm_path = os.path.abspath(os.path.join(g_ndk_root, 'toolchains/llvm/prebuilt', '%s-%s' % (cur_platform, 'x86')))
if os.path.isdir(x64_llvm_path):
llvm_path = x64_llvm_path
elif os.path.isdir(x86_llvm_path):
llvm_path = x86_llvm_path
else:
print('llvm toolchain not found!')
print('path: %s or path: %s are not valid! ' % (x86_llvm_path, x64_llvm_path))
sys.exit(1)
# x86_gcc_toolchain_path = ""
# x64_gcc_toolchain_path = os.path.abspath(os.path.join(g_ndk_root, 'toolchains/arm-linux-androideabi-4.9/prebuilt', '%s-%s' % (cur_platform, 'x86_64')))
# if not os.path.exists(x64_gcc_toolchain_path):
# x86_gcc_toolchain_path = os.path.abspath(os.path.join(g_ndk_root, 'toolchains/arm-linux-androideabi-4.9/prebuilt', '%s' % (cur_platform)))
# if not os.path.exists(x86_gcc_toolchain_path):
# x86_gcc_toolchain_path = os.path.abspath(os.path.join(g_ndk_root, 'toolchains/arm-linux-androideabi-4.9/prebuilt', '%s-%s' % (cur_platform, 'x86')))
# if os.path.isdir(x64_gcc_toolchain_path):
# gcc_toolchain_path = x64_gcc_toolchain_path
# elif os.path.isdir(x86_gcc_toolchain_path):
# gcc_toolchain_path = x86_gcc_toolchain_path
# else:
# print('gcc toolchain not found!')
# print('path: %s or path: %s are not valid! ' % (x64_gcc_toolchain_path, x86_gcc_toolchain_path))
# sys.exit(1)
project_root = os.path.abspath(os.path.join(os.path.dirname(__file__), '..', '..'))
ax_root = os.path.abspath(os.path.join(project_root, ''))
cxx_generator_root = os.path.abspath(os.path.join(project_root, 'tools/bindings-generator'))
extraFlags = _defaultIncludePath()
extraFlags += " -DFMT_USE_CONCEPTS=0"
# save config to file
if(sys.version_info.major >= 3):
import configparser # import ConfigParser
config = configparser.ConfigParser()
else:
import ConfigParser
config = ConfigParser.ConfigParser()
config.set('DEFAULT', 'androidndkdir', g_ndk_root)
config.set('DEFAULT', 'clangllvmdir', llvm_path)
# config.set('DEFAULT', 'gcc_toolchain_dir', gcc_toolchain_path)
config.set('DEFAULT', 'axdir', ax_root)
config.set('DEFAULT', 'cxxgeneratordir', cxx_generator_root)
config.set('DEFAULT', 'extra_flags', extraFlags)
config.set('DEFAULT', 'cxx_std', 'c++20')
conf_ini_file = os.path.abspath(os.path.join(os.path.dirname(__file__), 'userconf.ini'))
print('generating userconf.ini...')
with open(conf_ini_file, 'w', newline='\n') as configfile:
config.write(configfile)
# set proper environment variables
if 'linux' in platform or platform == 'darwin':
os.putenv('LD_LIBRARY_PATH', '%s/libclang' % cxx_generator_root)
if platform == 'win32':
path_env = os.environ['PATH']
os.putenv('PATH', r'%s;%s\libclang;%s\tools\win32;' % (path_env, cxx_generator_root, cxx_generator_root))
try:
tolua_root = '%s/tools/tolua' % project_root
output_dir = '%s/extensions/scripting/lua-bindings/auto' % project_root
cmd_args = {
'ax_base.ini' : ('ax_base', 'axlua_base_auto'), \
'ax_backend.ini' : ('ax_backend', 'axlua_backend_auto'), \
'ax_extension.ini' : ('ax_extension', 'axlua_extension_auto'), \
'ax_ui.ini' : ('ax_ui', 'axlua_ui_auto'), \
'ax_studio.ini' : ('ax_studio', 'axlua_studio_auto'), \
'ax_spine.ini' : ('ax_spine', 'axlua_spine_auto'), \
'ax_physics.ini' : ('ax_physics', 'axlua_physics_auto'), \
'ax_video.ini' : ('ax_video', 'axlua_video_auto'), \
'ax_controller.ini' : ('ax_controller', 'axlua_controller_auto'), \
'ax_3d.ini': ('ax_3d', 'axlua_3d_auto'), \
'ax_audioengine.ini': ('ax_audioengine', 'axlua_audioengine_auto'), \
'ax_csloader.ini' : ('ax_csloader', 'axlua_csloader_auto'), \
'ax_webview.ini' : ('ax_webview', 'axlua_webview_auto'), \
'ax_physics3d.ini' : ('ax_physics3d', 'axlua_physics3d_auto'), \
'ax_navmesh.ini' : ('ax_navmesh', 'axlua_navmesh_auto'), \
'ax_fairygui.ini' : ('ax_fairygui', 'axlua_fairygui_auto'), \
}
target = 'lua'
generator_py = '%s/generator.py' % cxx_generator_root
for key in cmd_args.keys():
args = cmd_args[key]
cfg = '%s/%s' % (tolua_root, key)
print('Generating bindings for %s...' % (key[:-4]))
command = '"%s" %s %s -s %s -t %s -o %s -n %s' % (python_bin, generator_py, cfg, args[0], target, output_dir, args[1])
print(command)
_run_cmd(command)
print('---------------------------------')
print('Generating lua bindings succeeds.')
print('---------------------------------')
except Exception as e:
if e.__class__.__name__ == 'CmdError':
print('---------------------------------')
print('Generating lua bindings fails.')
print('---------------------------------')
sys.exit(1)
else:
raise
# -------------- main --------------
if __name__ == '__main__':
parser = argparse.ArgumentParser(description='Install android sdk/ndk')
parser.add_argument("--ndk_root", help="Specificy ndk root")
args = parser.parse_args()
g_ndk_root = args.ndk_root
main()