axmol/tools/tolua/genbindings.py

251 lines
9.3 KiB
Python

#!/usr/bin/python
# This script is used to generate luabinding glue codes.
# Android ndk version must be ndk-r9b.
import sys
import os, os.path
import shutil
import subprocess
import re
from contextlib import contextmanager
def _check_ndk_root_env():
''' Checking the environment ANDROID_NDK, which will be used for building
'''
try:
ANDROID_NDK = os.environ['ANDROID_NDK']
except Exception:
print("ANDROID_NDK not defined. Please define ANDROID_NDK in your environment.")
sys.exit(1)
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():
'''
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(_check_ndk_root_env(), "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"
'''
versionFile = _find_first_file_in_dir(_check_ndk_root_env(), "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():
cur_platform= '??'
llvm_path = '??'
ndk_root = _check_ndk_root_env()
# del the " in the path
ndk_root = re.sub(r"\"", "", 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(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(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(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(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(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(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__), '..', '..'))
cocos_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()
# 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', ndk_root)
config.set('DEFAULT', 'clangllvmdir', llvm_path)
config.set('DEFAULT', 'gcc_toolchain_dir', gcc_toolchain_path)
config.set('DEFAULT', 'axisdir', cocos_root)
config.set('DEFAULT', 'cxxgeneratordir', cxx_generator_root)
config.set('DEFAULT', 'extra_flags', extraFlags)
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') 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 = {
'cocos2dx.ini' : ('cocos2d-x', 'lua_cocos2dx_auto'), \
'cocos2dx_backend.ini' : ('cocos2dx_backend', 'lua_cocos2dx_backend_auto'), \
'cocos2dx_extension.ini' : ('cocos2dx_extension', 'lua_cocos2dx_extension_auto'), \
'cocos2dx_ui.ini' : ('cocos2dx_ui', 'lua_cocos2dx_ui_auto'), \
'cocos2dx_studio.ini' : ('cocos2dx_studio', 'lua_cocos2dx_studio_auto'), \
'cocos2dx_spine.ini' : ('cocos2dx_spine', 'lua_cocos2dx_spine_auto'), \
'cocos2dx_physics.ini' : ('cocos2dx_physics', 'lua_cocos2dx_physics_auto'), \
'cocos2dx_video.ini' : ('cocos2dx_video', 'lua_cocos2dx_video_auto'), \
'cocos2dx_controller.ini' : ('cocos2dx_controller', 'lua_cocos2dx_controller_auto'), \
'cocos2dx_3d.ini': ('cocos2dx_3d', 'lua_cocos2dx_3d_auto'), \
'cocos2dx_audioengine.ini': ('cocos2dx_audioengine', 'lua_cocos2dx_audioengine_auto'), \
'cocos2dx_csloader.ini' : ('cocos2dx_csloader', 'lua_cocos2dx_csloader_auto'), \
'cocos2dx_webview.ini' : ('cocos2dx_webview', 'lua_cocos2dx_webview_auto'), \
'cocos2dx_physics3d.ini' : ('cocos2dx_physics3d', 'lua_cocos2dx_physics3d_auto'), \
'cocos2dx_navmesh.ini' : ('cocos2dx_navmesh', 'lua_cocos2dx_navmesh_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__':
main()