2017-10-02 23:24:00 +02:00
|
|
|
#!/usr/bin/env python
|
|
|
|
|
|
|
|
Import('env')
|
2017-12-12 16:18:55 +01:00
|
|
|
Import('env_modules')
|
|
|
|
|
|
|
|
env_mono = env_modules.Clone()
|
2017-10-02 23:24:00 +02:00
|
|
|
|
2018-06-26 21:03:42 +02:00
|
|
|
# TODO move functions to their own modules
|
2017-10-02 23:24:00 +02:00
|
|
|
|
2018-09-12 02:50:16 +02:00
|
|
|
def make_cs_files_header(src, dst, version_dst):
|
2018-06-26 21:03:42 +02:00
|
|
|
from compat import byte_to_str
|
|
|
|
|
2017-10-07 15:36:20 +02:00
|
|
|
with open(dst, 'w') as header:
|
2018-09-04 22:13:47 +02:00
|
|
|
header.write('/* THIS FILE IS GENERATED DO NOT EDIT */\n')
|
|
|
|
header.write('#ifndef CS_COMPRESSED_H\n')
|
|
|
|
header.write('#define CS_COMPRESSED_H\n\n')
|
|
|
|
header.write('#ifdef TOOLS_ENABLED\n\n')
|
2018-09-11 18:13:45 +02:00
|
|
|
header.write('#include "core/map.h"\n')
|
|
|
|
header.write('#include "core/ustring.h"\n')
|
2017-10-02 23:24:00 +02:00
|
|
|
inserted_files = ''
|
|
|
|
import os
|
2018-08-17 13:49:15 +02:00
|
|
|
latest_mtime = 0
|
2018-09-04 22:13:47 +02:00
|
|
|
cs_file_count = 0
|
2018-08-17 13:49:15 +02:00
|
|
|
for root, _, files in os.walk(src):
|
|
|
|
files = [f for f in files if f.endswith('.cs')]
|
|
|
|
for file in files:
|
2018-09-04 22:13:47 +02:00
|
|
|
cs_file_count += 1
|
2018-08-17 13:49:15 +02:00
|
|
|
filepath = os.path.join(root, file)
|
|
|
|
filepath_src_rel = os.path.relpath(filepath, src)
|
|
|
|
mtime = os.path.getmtime(filepath)
|
|
|
|
latest_mtime = mtime if mtime > latest_mtime else latest_mtime
|
|
|
|
with open(filepath, 'rb') as f:
|
2017-10-02 23:24:00 +02:00
|
|
|
buf = f.read()
|
|
|
|
decomp_size = len(buf)
|
|
|
|
import zlib
|
|
|
|
buf = zlib.compress(buf)
|
2018-09-04 22:13:47 +02:00
|
|
|
name = str(cs_file_count)
|
|
|
|
header.write('\n')
|
|
|
|
header.write('// ' + filepath_src_rel + '\n')
|
|
|
|
header.write('static const int _cs_' + name + '_compressed_size = ' + str(len(buf)) + ';\n')
|
2017-10-02 23:24:00 +02:00
|
|
|
header.write('static const int _cs_' + name + '_uncompressed_size = ' + str(decomp_size) + ';\n')
|
|
|
|
header.write('static const unsigned char _cs_' + name + '_compressed[] = { ')
|
|
|
|
for i, buf_idx in enumerate(range(len(buf))):
|
|
|
|
if i > 0:
|
|
|
|
header.write(', ')
|
2017-10-07 15:36:20 +02:00
|
|
|
header.write(byte_to_str(buf[buf_idx]))
|
2018-10-22 19:27:41 +02:00
|
|
|
inserted_files += '\tr_files.insert("' + filepath_src_rel.replace('\\', '\\\\') + '", ' \
|
2017-10-02 23:24:00 +02:00
|
|
|
'CompressedFile(_cs_' + name + '_compressed_size, ' \
|
|
|
|
'_cs_' + name + '_uncompressed_size, ' \
|
|
|
|
'_cs_' + name + '_compressed));\n'
|
|
|
|
header.write(' };\n')
|
|
|
|
header.write('\nstruct CompressedFile\n' '{\n'
|
|
|
|
'\tint compressed_size;\n' '\tint uncompressed_size;\n' '\tconst unsigned char* data;\n'
|
|
|
|
'\n\tCompressedFile(int p_comp_size, int p_uncomp_size, const unsigned char* p_data)\n'
|
|
|
|
'\t{\n' '\t\tcompressed_size = p_comp_size;\n' '\t\tuncompressed_size = p_uncomp_size;\n'
|
|
|
|
'\t\tdata = p_data;\n' '\t}\n' '\n\tCompressedFile() {}\n' '};\n'
|
|
|
|
'\nvoid get_compressed_files(Map<String, CompressedFile>& r_files)\n' '{\n' + inserted_files + '}\n'
|
|
|
|
)
|
2018-09-04 22:13:47 +02:00
|
|
|
header.write('\n#endif // TOOLS_ENABLED\n')
|
|
|
|
header.write('\n#endif // CS_COMPRESSED_H\n')
|
2017-10-02 23:24:00 +02:00
|
|
|
|
2018-09-12 02:50:16 +02:00
|
|
|
glue_version = int(latest_mtime) # The latest modified time will do for now
|
|
|
|
|
|
|
|
with open(version_dst, 'w') as version_header:
|
|
|
|
version_header.write('/* THIS FILE IS GENERATED DO NOT EDIT */\n')
|
|
|
|
version_header.write('#ifndef CS_GLUE_VERSION_H\n')
|
|
|
|
version_header.write('#define CS_GLUE_VERSION_H\n\n')
|
|
|
|
version_header.write('#define CS_GLUE_VERSION UINT32_C(' + str(glue_version) + ')\n')
|
|
|
|
version_header.write('\n#endif // CS_GLUE_VERSION_H\n')
|
|
|
|
|
2017-10-02 23:24:00 +02:00
|
|
|
|
2017-12-12 16:18:55 +01:00
|
|
|
env_mono.add_source_files(env.modules_sources, '*.cpp')
|
2018-09-04 05:40:41 +02:00
|
|
|
env_mono.add_source_files(env.modules_sources, 'glue/*.cpp')
|
2017-12-12 16:18:55 +01:00
|
|
|
env_mono.add_source_files(env.modules_sources, 'mono_gd/*.cpp')
|
|
|
|
env_mono.add_source_files(env.modules_sources, 'utils/*.cpp')
|
2017-10-02 23:24:00 +02:00
|
|
|
|
|
|
|
if env['tools']:
|
2017-12-12 16:18:55 +01:00
|
|
|
env_mono.add_source_files(env.modules_sources, 'editor/*.cpp')
|
2018-09-04 22:13:47 +02:00
|
|
|
# NOTE: It is safe to generate this file here, since this is still executed serially
|
2018-09-12 21:22:43 +02:00
|
|
|
make_cs_files_header('glue/Managed/Files', 'glue/cs_compressed.gen.h', 'glue/cs_glue_version.gen.h')
|
2017-10-02 23:24:00 +02:00
|
|
|
|
|
|
|
vars = Variables()
|
|
|
|
vars.Add(BoolVariable('mono_glue', 'Build with the mono glue sources', True))
|
2017-10-24 22:47:27 +02:00
|
|
|
vars.Add(BoolVariable('xbuild_fallback', 'If MSBuild is not found, fallback to xbuild', False))
|
2018-02-11 04:48:46 +01:00
|
|
|
vars.Update(env_mono)
|
2017-10-02 23:24:00 +02:00
|
|
|
|
|
|
|
# Glue sources
|
2018-02-11 04:48:46 +01:00
|
|
|
if env_mono['mono_glue']:
|
2018-09-04 05:40:41 +02:00
|
|
|
env_mono.Append(CPPDEFINES=['MONO_GLUE_ENABLED'])
|
2017-10-02 23:24:00 +02:00
|
|
|
|
2018-06-26 21:03:42 +02:00
|
|
|
# Configure TLS checks
|
|
|
|
|
|
|
|
import tls_configure
|
|
|
|
conf = Configure(env_mono)
|
|
|
|
tls_configure.configure(conf)
|
|
|
|
env_mono = conf.Finish()
|
|
|
|
|
2017-10-02 23:24:00 +02:00
|
|
|
|
|
|
|
# Build GodotSharpTools solution
|
|
|
|
|
2017-10-24 22:47:27 +02:00
|
|
|
|
2017-10-02 23:24:00 +02:00
|
|
|
import os
|
2017-10-24 22:47:27 +02:00
|
|
|
|
|
|
|
|
2018-10-22 19:20:29 +02:00
|
|
|
def find_nuget_unix():
|
|
|
|
import os.path
|
|
|
|
import sys
|
|
|
|
|
|
|
|
hint_dirs = ['/opt/novell/mono/bin']
|
|
|
|
if sys.platform == 'darwin':
|
|
|
|
hint_dirs = ['/Library/Frameworks/Mono.framework/Versions/Current/bin', '/usr/local/var/homebrew/linked/mono/bin'] + hint_dirs
|
|
|
|
|
|
|
|
for hint_dir in hint_dirs:
|
|
|
|
hint_path = os.path.join(hint_dir, 'nuget')
|
|
|
|
if os.path.isfile(hint_path):
|
|
|
|
return hint_path
|
|
|
|
elif os.path.isfile(hint_path + '.exe'):
|
|
|
|
return hint_path + '.exe'
|
|
|
|
|
|
|
|
for hint_dir in os.environ['PATH'].split(os.pathsep):
|
|
|
|
hint_dir = hint_dir.strip('"')
|
|
|
|
hint_path = os.path.join(hint_dir, 'nuget')
|
|
|
|
if os.path.isfile(hint_path) and os.access(hint_path, os.X_OK):
|
|
|
|
return hint_path
|
|
|
|
if os.path.isfile(hint_path + '.exe') and os.access(hint_path + '.exe', os.X_OK):
|
|
|
|
return hint_path + '.exe'
|
|
|
|
|
|
|
|
return None
|
|
|
|
|
|
|
|
|
|
|
|
def find_nuget_windows():
|
|
|
|
import mono_reg_utils as monoreg
|
|
|
|
|
|
|
|
mono_root = ''
|
|
|
|
bits = env['bits']
|
|
|
|
|
|
|
|
if bits == '32':
|
|
|
|
if os.getenv('MONO32_PREFIX'):
|
|
|
|
mono_root = os.getenv('MONO32_PREFIX')
|
|
|
|
else:
|
|
|
|
mono_root = monoreg.find_mono_root_dir(bits)
|
|
|
|
else:
|
|
|
|
if os.getenv('MONO64_PREFIX'):
|
|
|
|
mono_root = os.getenv('MONO64_PREFIX')
|
|
|
|
else:
|
|
|
|
mono_root = monoreg.find_mono_root_dir(bits)
|
|
|
|
|
|
|
|
if mono_root:
|
|
|
|
mono_bin_dir = os.path.join(mono_root, 'bin')
|
|
|
|
nuget_mono = os.path.join(mono_bin_dir, 'nuget.bat')
|
|
|
|
|
|
|
|
if os.path.isfile(nuget_mono):
|
|
|
|
return nuget_mono
|
|
|
|
|
|
|
|
# Standalone NuGet
|
|
|
|
|
|
|
|
for hint_dir in os.environ['PATH'].split(os.pathsep):
|
|
|
|
hint_dir = hint_dir.strip('"')
|
|
|
|
hint_path = os.path.join(hint_dir, 'nuget.exe')
|
|
|
|
if os.path.isfile(hint_path) and os.access(hint_path, os.X_OK):
|
|
|
|
return hint_path
|
|
|
|
|
|
|
|
if 'NUGET_PATH' in os.environ:
|
|
|
|
hint_path = os.environ['NUGET_PATH']
|
|
|
|
if os.path.isfile(hint_path) and os.access(hint_path, os.X_OK):
|
|
|
|
return hint_path
|
|
|
|
hint_path = os.path.join(hint_path, 'nuget.exe')
|
|
|
|
if os.path.isfile(hint_path) and os.access(hint_path, os.X_OK):
|
|
|
|
return hint_path
|
|
|
|
|
|
|
|
return None
|
|
|
|
|
|
|
|
|
2017-10-24 22:47:27 +02:00
|
|
|
def find_msbuild_unix(filename):
|
|
|
|
import os.path
|
|
|
|
import sys
|
|
|
|
|
|
|
|
hint_dirs = ['/opt/novell/mono/bin']
|
2018-02-22 13:13:51 +01:00
|
|
|
if sys.platform == 'darwin':
|
2018-07-10 00:25:13 +02:00
|
|
|
hint_dirs = ['/Library/Frameworks/Mono.framework/Versions/Current/bin', '/usr/local/var/homebrew/linked/mono/bin'] + hint_dirs
|
2017-10-24 22:47:27 +02:00
|
|
|
|
|
|
|
for hint_dir in hint_dirs:
|
|
|
|
hint_path = os.path.join(hint_dir, filename)
|
|
|
|
if os.path.isfile(hint_path):
|
|
|
|
return hint_path
|
2018-02-22 13:13:51 +01:00
|
|
|
elif os.path.isfile(hint_path + '.exe'):
|
|
|
|
return hint_path + '.exe'
|
2017-10-24 22:47:27 +02:00
|
|
|
|
2018-02-22 13:13:51 +01:00
|
|
|
for hint_dir in os.environ['PATH'].split(os.pathsep):
|
2017-10-24 22:47:27 +02:00
|
|
|
hint_dir = hint_dir.strip('"')
|
|
|
|
hint_path = os.path.join(hint_dir, filename)
|
|
|
|
if os.path.isfile(hint_path) and os.access(hint_path, os.X_OK):
|
|
|
|
return hint_path
|
2018-02-22 13:13:51 +01:00
|
|
|
if os.path.isfile(hint_path + '.exe') and os.access(hint_path + '.exe', os.X_OK):
|
|
|
|
return hint_path + '.exe'
|
2017-10-24 22:47:27 +02:00
|
|
|
|
|
|
|
return None
|
|
|
|
|
|
|
|
|
|
|
|
def find_msbuild_windows():
|
|
|
|
import mono_reg_utils as monoreg
|
|
|
|
|
2018-09-17 22:54:47 +02:00
|
|
|
mono_root = ''
|
2017-10-29 22:16:02 +01:00
|
|
|
bits = env['bits']
|
2017-10-24 22:47:27 +02:00
|
|
|
|
2017-10-29 22:16:02 +01:00
|
|
|
if bits == '32':
|
|
|
|
if os.getenv('MONO32_PREFIX'):
|
|
|
|
mono_root = os.getenv('MONO32_PREFIX')
|
|
|
|
else:
|
|
|
|
mono_root = monoreg.find_mono_root_dir(bits)
|
2017-10-24 22:47:27 +02:00
|
|
|
else:
|
2017-10-29 22:16:02 +01:00
|
|
|
if os.getenv('MONO64_PREFIX'):
|
|
|
|
mono_root = os.getenv('MONO64_PREFIX')
|
2017-10-24 22:47:27 +02:00
|
|
|
else:
|
2017-10-29 22:16:02 +01:00
|
|
|
mono_root = monoreg.find_mono_root_dir(bits)
|
|
|
|
|
|
|
|
if not mono_root:
|
|
|
|
raise RuntimeError('Cannot find mono root directory')
|
2017-10-24 22:47:27 +02:00
|
|
|
|
2018-05-17 23:40:22 +02:00
|
|
|
framework_path = os.path.join(mono_root, 'lib', 'mono', '4.5')
|
|
|
|
mono_bin_dir = os.path.join(mono_root, 'bin')
|
|
|
|
msbuild_mono = os.path.join(mono_bin_dir, 'msbuild.bat')
|
|
|
|
|
|
|
|
if os.path.isfile(msbuild_mono):
|
2018-06-26 21:03:42 +02:00
|
|
|
# The (Csc/Vbc/Fsc)ToolExe environment variables are required when
|
|
|
|
# building with Mono's MSBuild. They must point to the batch files
|
|
|
|
# in Mono's bin directory to make sure they are executed with Mono.
|
2018-05-17 23:40:22 +02:00
|
|
|
mono_msbuild_env = {
|
|
|
|
'CscToolExe': os.path.join(mono_bin_dir, 'csc.bat'),
|
|
|
|
'VbcToolExe': os.path.join(mono_bin_dir, 'vbc.bat'),
|
|
|
|
'FscToolExe': os.path.join(mono_bin_dir, 'fsharpc.bat')
|
|
|
|
}
|
|
|
|
return (msbuild_mono, framework_path, mono_msbuild_env)
|
|
|
|
|
2017-10-29 22:16:02 +01:00
|
|
|
msbuild_tools_path = monoreg.find_msbuild_tools_path_reg()
|
|
|
|
|
|
|
|
if msbuild_tools_path:
|
2018-05-17 23:40:22 +02:00
|
|
|
return (os.path.join(msbuild_tools_path, 'MSBuild.exe'), framework_path, {})
|
2017-10-24 22:47:27 +02:00
|
|
|
|
|
|
|
return None
|
2017-10-02 23:24:00 +02:00
|
|
|
|
|
|
|
|
|
|
|
def mono_build_solution(source, target, env):
|
2017-10-24 22:47:27 +02:00
|
|
|
import subprocess
|
|
|
|
import mono_reg_utils as monoreg
|
|
|
|
from shutil import copyfile
|
|
|
|
|
2018-10-22 19:20:29 +02:00
|
|
|
sln_path = os.path.abspath(str(source[0]))
|
|
|
|
target_path = os.path.abspath(str(target[0]))
|
2018-05-17 23:40:22 +02:00
|
|
|
|
2018-10-22 19:20:29 +02:00
|
|
|
framework_path = ''
|
2018-05-17 23:40:22 +02:00
|
|
|
msbuild_env = os.environ.copy()
|
|
|
|
|
|
|
|
# Needed when running from Developer Command Prompt for VS
|
|
|
|
if 'PLATFORM' in msbuild_env:
|
|
|
|
del msbuild_env['PLATFORM']
|
2017-10-24 22:47:27 +02:00
|
|
|
|
2018-10-22 19:20:29 +02:00
|
|
|
# Find MSBuild
|
2017-10-02 23:24:00 +02:00
|
|
|
if os.name == 'nt':
|
2017-10-24 22:47:27 +02:00
|
|
|
msbuild_info = find_msbuild_windows()
|
|
|
|
if msbuild_info is None:
|
|
|
|
raise RuntimeError('Cannot find MSBuild executable')
|
2017-10-29 17:28:53 +01:00
|
|
|
msbuild_path = msbuild_info[0]
|
2018-05-17 23:40:22 +02:00
|
|
|
framework_path = msbuild_info[1]
|
|
|
|
msbuild_env.update(msbuild_info[2])
|
2017-10-02 23:24:00 +02:00
|
|
|
else:
|
2017-10-24 22:47:27 +02:00
|
|
|
msbuild_path = find_msbuild_unix('msbuild')
|
|
|
|
if msbuild_path is None:
|
|
|
|
xbuild_fallback = env['xbuild_fallback']
|
|
|
|
|
|
|
|
if xbuild_fallback and os.name == 'nt':
|
2018-02-22 13:13:51 +01:00
|
|
|
print('Option \'xbuild_fallback\' not supported on Windows')
|
2017-10-24 22:47:27 +02:00
|
|
|
xbuild_fallback = False
|
|
|
|
|
|
|
|
if xbuild_fallback:
|
|
|
|
print('Cannot find MSBuild executable, trying with xbuild')
|
|
|
|
print('Warning: xbuild is deprecated')
|
|
|
|
|
|
|
|
msbuild_path = find_msbuild_unix('xbuild')
|
|
|
|
|
|
|
|
if msbuild_path is None:
|
|
|
|
raise RuntimeError('Cannot find xbuild executable')
|
|
|
|
else:
|
|
|
|
raise RuntimeError('Cannot find MSBuild executable')
|
|
|
|
|
|
|
|
print('MSBuild path: ' + msbuild_path)
|
2017-10-02 23:24:00 +02:00
|
|
|
|
2018-10-22 19:20:29 +02:00
|
|
|
# Find NuGet
|
|
|
|
nuget_path = find_nuget_windows() if os.name == 'nt' else find_nuget_unix()
|
|
|
|
if nuget_path is None:
|
|
|
|
raise RuntimeError('Cannot find NuGet executable')
|
|
|
|
|
|
|
|
print('NuGet path: ' + nuget_path)
|
|
|
|
|
|
|
|
# Do NuGet restore
|
|
|
|
|
|
|
|
try:
|
|
|
|
subprocess.check_call([nuget_path, 'restore', sln_path])
|
|
|
|
except subprocess.CalledProcessError:
|
|
|
|
raise RuntimeError('GodotSharpTools: NuGet restore failed')
|
|
|
|
|
|
|
|
# Build solution
|
|
|
|
|
2017-10-24 22:47:27 +02:00
|
|
|
build_config = 'Release'
|
2017-10-02 23:24:00 +02:00
|
|
|
|
|
|
|
msbuild_args = [
|
|
|
|
msbuild_path,
|
2018-10-22 19:20:29 +02:00
|
|
|
sln_path,
|
2017-10-24 22:47:27 +02:00
|
|
|
'/p:Configuration=' + build_config,
|
2017-10-02 23:24:00 +02:00
|
|
|
]
|
|
|
|
|
2018-05-17 23:40:22 +02:00
|
|
|
if framework_path:
|
|
|
|
msbuild_args += ['/p:FrameworkPathOverride=' + framework_path]
|
2017-10-02 23:24:00 +02:00
|
|
|
|
2017-10-24 22:47:27 +02:00
|
|
|
try:
|
|
|
|
subprocess.check_call(msbuild_args, env=msbuild_env)
|
|
|
|
except subprocess.CalledProcessError:
|
2018-10-22 19:20:29 +02:00
|
|
|
raise RuntimeError('GodotSharpTools: Build failed')
|
|
|
|
|
|
|
|
# Copy files
|
2017-10-24 22:47:27 +02:00
|
|
|
|
2018-10-22 19:20:29 +02:00
|
|
|
src_dir = os.path.abspath(os.path.join(sln_path, os.pardir, 'bin', build_config))
|
|
|
|
dst_dir = os.path.abspath(os.path.join(target_path, os.pardir))
|
|
|
|
asm_file = 'GodotSharpTools.dll'
|
2017-10-24 22:47:27 +02:00
|
|
|
|
|
|
|
if not os.path.isdir(dst_dir):
|
|
|
|
if os.path.exists(dst_dir):
|
|
|
|
raise RuntimeError('Target directory is a file')
|
|
|
|
os.makedirs(dst_dir)
|
|
|
|
|
|
|
|
copyfile(os.path.join(src_dir, asm_file), os.path.join(dst_dir, asm_file))
|
2017-10-02 23:24:00 +02:00
|
|
|
|
2018-10-22 19:20:29 +02:00
|
|
|
# Dependencies
|
|
|
|
copyfile(os.path.join(src_dir, "DotNet.Glob.dll"), os.path.join(dst_dir, "DotNet.Glob.dll"))
|
|
|
|
|
2018-10-03 19:01:57 +02:00
|
|
|
if env['tools']:
|
|
|
|
output_dir = Dir('#bin').abspath
|
|
|
|
editor_tools_dir = os.path.join(output_dir, 'GodotSharp', 'Tools')
|
|
|
|
|
|
|
|
mono_sln_builder = Builder(action=mono_build_solution)
|
|
|
|
env_mono.Append(BUILDERS={'MonoBuildSolution': mono_sln_builder})
|
|
|
|
env_mono.MonoBuildSolution(
|
|
|
|
os.path.join(editor_tools_dir, 'GodotSharpTools.dll'),
|
|
|
|
'editor/GodotSharpTools/GodotSharpTools.sln'
|
|
|
|
)
|