radare2/meson.build
pancake aab694d655 First arch plugin (arch.null), implement basic lib api ##arch
* Still far from usable, no dynamic plugin loading
* No way to retrieve a decoder yet
2022-09-21 08:22:55 +02:00

696 lines
20 KiB
Meson

project('radare2', 'c', license : 'LGPL3', meson_version : '>=0.49', version : '5.7.9')
py3_exe = import('python').find_installation('python3')
git_exe = find_program('git', required: false)
pkgconfig_mod = import('pkgconfig')
# Get r2 version
r2_version = meson.project_version()
vers = r2_version.split('.')
r2_version_major = vers[0].to_int()
r2_version_minor = vers[1].to_int()
r2_version_patch = vers[2].split('-')[0].to_int()
r2_version_number = (r2_version_major * 10000) + (r2_version_minor * 100) + (r2_version_patch)
repo = '.'
if meson.is_subproject()
repo = meson.current_source_dir()
if host_machine.system() == 'windows'
py_cmd = 'print(__import__("os").readlink(r"@0@"))'.format(repo)
py_cmd = run_command(py3_exe, '-c', py_cmd)
if py_cmd.returncode() == 0
repo = py_cmd.stdout().strip()
message('r2 real path: ' + repo)
endif
endif
endif
# by default, not version commit is used
version_commit = '0'
gittap = ''
gittip = 'unknown'
if git_exe.found()
# Get version_commit
git_rev_list = run_command(git_exe, '-C', repo, 'rev-list', '--all', '--count')
if git_rev_list.returncode() == 0
version_commit = git_rev_list.stdout().strip()
endif
if version_commit == ''
version_commit = '0'
endif
# Get gittap
git_describe = run_command(git_exe, '-C', repo, 'describe', '--tags', '--match', '[0-9]*')
if git_describe.returncode() == 0
gittap = git_describe.stdout().strip()
endif
# Get gittip
git_rev_parse = run_command(git_exe, '-C', repo, 'rev-parse', 'HEAD')
if git_rev_parse.returncode() == 0
gittip = git_rev_parse.stdout().strip()
endif
endif
if get_option('r2_version_commit') != ''
version_commit = get_option('r2_version_commit')
endif
if get_option('r2_gittap') != ''
gittap = get_option('r2_gittap')
endif
# gittap is used for the version of each r_ library
# in case it has not been set (e.g. a release tarball) set it
if gittap == ''
gittap = r2_version
endif
if get_option('r2_gittip') != ''
gittip = get_option('r2_gittip')
endif
if gittip == ''
gittip = r2_version
endif
# Get current date
if host_machine.system() == 'windows'
r2birth = run_command('cmd', '/c', 'echo %date%__%time%')
else
r2birth = run_command('sh', '-c', '''
SOURCE_DATE_EPOCH="${SOURCE_DATE_EPOCH:-`date +%s`}";
FORMAT="+%Y-%m-%d__%H:%M:%S";
date -u -d @"$SOURCE_DATE_EPOCH" "$FORMAT" 2>/dev/null ||
date -u -r "$SOURCE_DATE_EPOCH" "$FORMAT" 2>/dev/null ||
date -u "$FORMAT"''')
endif
if r2birth.returncode() != 0
r2birth = ''
else
r2birth = r2birth.stdout().strip()
endif
r2_libversion = host_machine.system() == 'windows' ? '' : r2_version
message('r2 lib version: ' + r2_libversion)
# system dependencies
cc = meson.get_compiler('c')
# required for linux
ldl = cc.find_library('dl', required: false)
pth = dependency('threads', required: false)
utl = cc.find_library('util', required: false)
if host_machine.system() == 'sunos'
# workaround for Solaris until https://github.com/mesonbuild/meson/issues/4328 is fixed
mth = declare_dependency(link_args: '-lm')
else
mth = cc.find_library('m', required: false)
endif
platform_deps = []
platform_inc = ['.', 'libr/include']
if host_machine.system() == 'windows'
platform_deps = [cc.find_library('ws2_32'), cc.find_library('wininet'), cc.find_library('psapi')]
endif
platform_inc = include_directories(platform_inc)
if get_option('static_runtime')
if cc.get_id() == 'msvc'
add_project_arguments('/MT', language: 'c')
endif
endif
if get_option('wasan')
add_project_arguments('/DEBUG', language: 'c')
add_project_arguments('/MD', language: 'c')
add_project_arguments('/fsanitize=address', language: 'c')
# add_project_arguments('/Zi', language: 'c') # generate .pdb iirc
# add_project_arguments('/Z7', language: 'c') # generate .symbols instead of .pdb
# add_project_arguments('/guard:cf', language: 'c')
endif
if cc.get_id() == 'clang-cl'
add_project_arguments('-D__STDC__=1', language: 'c')
add_project_arguments('-D_CRT_DECLARE_NONSTDC_NAMES ', language: 'c')
add_project_arguments('-D_CRT_SECURE_NO_WARNINGS', language: 'c')
add_project_arguments('-D_CRT_NONSTDC_NO_DEPRECATE', language: 'c')
endif
if get_option('default_library') == 'shared'
if host_machine.system() != 'windows' or cc.get_id() != 'msvc' and cc.get_id() != 'clang-cl'
add_project_arguments('-fvisibility=hidden', language: 'c')
endif
endif
library_cflags = ['-DR2_PLUGIN_INCORE=1']
if host_machine.system() == 'windows'
r2_prefix = '.'
r2_libdir = 'lib'
r2_incdir = 'include/libr'
r2_datdir = 'share'
opts1 = [
'r2_libdir',
'r2_incdir',
'r2_datdir'
]
foreach opt : opts1
val = get_option(opt)
if val != ''
set_variable(opt, val)
endif
endforeach
r2_wwwroot = join_paths(r2_datdir, 'www')
r2_sdb = join_paths(r2_datdir)
r2_zigns = join_paths(r2_datdir, 'zigns')
r2_themes = join_paths(r2_datdir, 'cons')
r2_fortunes = join_paths(r2_datdir, 'doc')
r2_flags = join_paths(r2_datdir, 'flag')
r2_hud = join_paths(r2_datdir, 'hud')
opts2 = [
'r2_wwwroot',
'r2_sdb',
'r2_zigns',
'r2_themes',
'r2_fortunes',
'r2_flags',
'r2_hud'
]
foreach opt : opts2
val = get_option(opt)
if val != ''
set_variable(opt, val)
endif
endforeach
opts3 = [
'r2_plugins',
'r2_extras',
'r2_bindings'
]
r2_plugins = join_paths(r2_libdir, 'plugins')
r2_extras = join_paths(r2_libdir, 'extras')
r2_bindings = join_paths(r2_libdir, 'bindings')
foreach opt : opts1 + opts2 + opts3
val = get_variable(opt)
val = '\\\\'.join(val.split('/'))
set_variable(opt, val)
endforeach
else
r2_prefix = get_option('prefix')
r2_libdir = get_option('libdir')
r2_incdir = join_paths(get_option('includedir'), 'libr')
r2_datdir = get_option('datadir')
r2_datdir_r2 = join_paths(r2_datdir, 'radare2')
r2_wwwroot = join_paths(r2_datdir_r2, r2_version, 'www')
r2_sdb = join_paths(r2_datdir_r2, r2_version)
r2_zigns = join_paths(r2_datdir_r2, r2_version, 'zigns')
r2_themes = join_paths(r2_datdir_r2, r2_version, 'cons')
r2_fortunes = join_paths(r2_datdir, 'doc', 'radare2')
r2_flags = join_paths(r2_datdir_r2, r2_version, 'flag')
r2_hud = join_paths(r2_datdir_r2, r2_version, 'hud')
r2_plugins = join_paths(r2_libdir, 'radare2', r2_version)
r2_extras = join_paths(r2_libdir, 'radare2-extras', r2_version)
r2_bindings = join_paths(r2_libdir, 'radare2-bindings', r2_version)
endif
r2_zsh_compdir = join_paths(r2_datdir, 'zsh', 'site-functions')
# load plugin configuration
subdir('libr')
conf_data = configuration_data()
conf_data.set('plugins_core', '&r_core_plugin_' + ', &r_core_plugin_'.join(core_plugins) + ', 0')
conf_data.set('plugins_esil', '&r_esil_plugin_' + ', &r_esil_plugin_'.join(esil_plugins) + ', 0')
conf_data.set('plugins_anal', '&r_anal_plugin_' + ', &r_anal_plugin_'.join(anal_plugins) + ', 0')
conf_data.set('plugins_arch', '&r_arch_plugin_' + ', &r_arch_plugin_'.join(arch_plugins) + ', 0')
conf_data.set('plugins_asm', '&r_asm_plugin_' + ', &r_asm_plugin_'.join(asm_plugins) + ', 0')
conf_data.set('plugins_bp', '&r_bp_plugin_' + ', &r_bp_plugin_'.join(bp_plugins) + ', 0')
conf_data.set('plugins_bin', '&r_bin_plugin_' + ', &r_bin_plugin_'.join(bin_plugins) + ', 0')
conf_data.set('plugins_bin_ldr', '&r_bin_ldr_plugin_' + ', &r_bin_ldr_plugin_'.join(bin_ldr_plugins) + ', 0')
conf_data.set('plugins_bin_xtr', '&r_bin_xtr_plugin_' + ', &r_bin_xtr_plugin_'.join(bin_xtr_plugins) + ', 0')
conf_data.set('plugins_crypto', '&r_crypto_plugin_' + ', &r_crypto_plugin_'.join(crypto_plugins) + ', 0')
conf_data.set('plugins_io', '&r_io_plugin_' + ', &r_io_plugin_'.join(io_plugins) + ', 0')
conf_data.set('plugins_fs', '&r_fs_plugin_' + ', &r_fs_plugin_'.join(fs_plugins) + ', 0')
conf_data.set('plugins_debug', '&r_debug_plugin_' + ', &r_debug_plugin_'.join(debug_plugins) + ', 0')
conf_data.set('plugins_egg', '&r_egg_plugin_' + ', &r_egg_plugin_'.join(egg_plugins) + ', 0')
conf_data.set('plugins_lang', '&r_lang_plugin_' + ', &r_lang_plugin_'.join(lang_plugins) + ', 0')
conf_data.set('plugins_parse', '&r_parse_plugin_' + ', &r_parse_plugin_'.join(parse_plugins) + ', 0')
# userconf_h = configure_file(
# input: 'libr/include/r_userconf.h.acr',
# output: 'r_userconf.h',
# configuration: conf_data
#)
config_h = configure_file(
input: 'libr/config.h.in',
output: 'config.h',
configuration: conf_data
)
# handle magic library
sys_magic = cc.find_library('magic', required: false)
use_syslib_magic = false
if sys_magic.found() and get_option('use_sys_magic')
use_syslib_magic = true
endif
# handle xxhash library
sys_xxhash = dependency('libxxhash', required: false)
use_sys_xxhash = false
if not sys_xxhash.found()
sys_xxhash = dependency('xxhash', required: false)
endif
if not sys_xxhash.found()
sys_xxhash = cc.find_library('xxhash', required: false)
endif
if sys_xxhash.found() and get_option('use_sys_xxhash')
message('Using system xxhash library')
use_sys_xxhash = true
else
message('Using bundled xxhash library')
endif
# handle openssl library
if get_option('use_ssl')
static_sys_openssl = get_option('static_sys_openssl')
sys_openssl = dependency('openssl', required: false, static: static_sys_openssl)
use_sys_openssl = false
if sys_openssl.found() and get_option('use_sys_openssl')
message('Using system openssl library')
use_sys_openssl = true
else
message('Using bundled openssl code')
endif
else
sys_openssl = false
use_sys_openssl = false
endif
use_fork = get_option('use_fork')
use_dylink = get_option('use_dylink')
# handle libuv library
if get_option('use_libuv')
libuv_dep = dependency('libuv', version: '>=1.0.0', required: false)
use_libuv = libuv_dep.found()
if not libuv_dep.found()
warning('use_libuv option was set to true, but libuv was not found.')
endif
else
use_libuv = false
endif
if use_libuv
message('Using libuv')
else
message('Not using libuv, thus using fallback server implementations')
endif
has_debugger = get_option('debugger')
have_ptrace = not ['windows', 'cygwin', 'sunos'].contains(host_machine.system())
can_ptrace_wrap = ['linux'].contains(host_machine.system())
have_ptrace = have_ptrace and has_debugger
use_ptrace_wrap = can_ptrace_wrap and has_debugger
message('HAVE_PTRACE: @0@'.format(have_ptrace))
message('USE_PTRACE_WRAP: @0@'.format(use_ptrace_wrap))
checks_level = get_option('checks_level')
if checks_level == 9999
if get_option('buildtype') == 'release'
checks_level = 1
else
checks_level = 2
endif
endif
message('R2_CHECKS_LEVEL: @0@'.format(checks_level))
userconf = configuration_data()
userconf.set('R_CHECKS_LEVEL', checks_level)
userconf.set10('HAVE_LIB_MAGIC', sys_magic.found())
userconf.set10('USE_LIB_MAGIC', use_syslib_magic)
userconf.set10('HAVE_LIB_XXHASH', sys_xxhash.found())
userconf.set10('USE_LIB_XXHASH', use_sys_xxhash)
userconf.set10('DEBUGGER', has_debugger)
userconf.set('ETCDIR', join_paths(r2_prefix, 'etc'))
userconf.set('PREFIX', r2_prefix)
if host_machine.system() == 'windows'
userconf.set('LIBDIR', r2_libdir)
userconf.set('INCLUDEDIR', r2_incdir)
userconf.set('DATADIR_R2', r2_datdir)
userconf.set10('HAVE_JEMALLOC', false)
else
userconf.set('LIBDIR', join_paths(r2_prefix, r2_libdir))
userconf.set('INCLUDEDIR', join_paths(r2_prefix, r2_incdir))
userconf.set('DATADIR_R2', r2_datdir_r2)
userconf.set10('HAVE_JEMALLOC', true)
endif
userconf.set('DATADIR', join_paths(r2_prefix, r2_datdir))
userconf.set('WWWROOT', join_paths(r2_prefix, r2_wwwroot))
userconf.set('SDB', r2_sdb)
userconf.set('ZIGNS', r2_zigns)
userconf.set('THEMES', r2_themes)
userconf.set('FORTUNES', r2_fortunes)
userconf.set('FLAGS', r2_flags)
userconf.set('HUD', r2_hud)
userconf.set('PLUGINS', r2_plugins)
userconf.set('EXTRAS', r2_extras)
userconf.set('BINDINGS', r2_bindings)
userconf.set10('HAVE_OPENSSL', use_sys_openssl)
userconf.set10('HAVE_LIBUV', use_libuv)
userconf.set10('HAVE_FORK', use_fork)
userconf.set10('HAVE_GPERF', get_option('sdb_cgen'))
userconf.set10('WANT_DYLINK', use_dylink)
userconf.set10('WANT_THREADS', get_option('want_threads'))
userconf.set10('WANT_CAPSTONE', get_option('want_capstone'))
userconf.set10('HAVE_PTRACE', have_ptrace)
userconf.set10('USE_PTRACE_WRAP', use_ptrace_wrap)
userconf.set10('WITH_GPL', not get_option('nogpl'))
ok = cc.has_header_symbol('sys/personality.h', 'ADDR_NO_RANDOMIZE')
userconf.set10('HAVE_DECL_ADDR_NO_RANDOMIZE', ok)
lrt = []
if not cc.has_function('clock_gettime', prefix: '#include <time.h>') and cc.has_header_symbol('features.h', '__GLIBC__')
lrt = cc.find_library('rt', required: true)
endif
foreach item : [
['arc4random_uniform', '#include <stdlib.h>', []],
['explicit_bzero', '#include <string.h>', []],
['explicit_memset', '#include <string.h>', []],
['clock_nanosleep', '#include <time.h>', []],
['clock_gettime', '#include <time.h>', [lrt]],
['sigaction', '#include <signal.h>', []]
]
func = item[0]
ok = cc.has_function(func, prefix: item[1], dependencies: item[2])
userconf.set10('HAVE_@0@'.format(func.to_upper()), ok)
endforeach
r_userconf_h = configure_file(
input: 'libr/include/r_userconf.h.acr',
output: 'r_userconf.h',
configuration: userconf,
install_dir: r2_incdir
)
versionconf = configuration_data()
versionconf.set('MESON_VERSION', meson.version())
versionconf.set('VERSIONCOMMIT', version_commit)
versionconf.set('R2_VERSION_MAJOR', r2_version_major)
versionconf.set('R2_VERSION_MINOR', r2_version_minor)
versionconf.set('R2_VERSION_PATCH', r2_version_patch)
versionconf.set('R2_VERSION_NUMBER', r2_version_number)
versionconf.set('R2_VERSION', r2_version)
versionconf.set('R2_GITTAP', gittap)
versionconf.set('R2_GITTIP', gittip)
versionconf.set('R2_BIRTH', r2birth)
r_version_h = configure_file(
input: 'libr/include/r_version.h.in',
output: 'r_version.h',
configuration: versionconf,
install_dir: r2_incdir
)
# Copy missing header
run_command(py3_exe, '-c', '__import__("shutil").copyfile("shlr/spp/config.def.h", "shlr/spp/config.h")')
pcconf = configuration_data()
pcconf.set('PREFIX', get_option('prefix'))
pcconf.set('LIBDIR', join_paths(get_option('prefix'), get_option('libdir')))
pcconf.set('VERSION', r2_version)
libr_pc = configure_file(
input: 'libr/libr.pc.acr',
output: 'libr.pc',
configuration: pcconf,
install_dir: join_paths(get_option('libdir'), 'pkgconfig')
)
subdir('shlr/zip')
# handle sdb dependency
# NOTE: copying most of the stuff from sdb to here for the moment, since we
# should use subpackages to handle this well
sdb_files = [
'shlr/sdb/src/array.c',
'shlr/sdb/src/base64.c',
'shlr/sdb/src/buffer.c',
'shlr/sdb/src/set.c',
'shlr/sdb/src/cdb.c',
'shlr/sdb/src/cdb_make.c',
'shlr/sdb/src/dict.c',
'shlr/sdb/src/diff.c',
'shlr/sdb/src/disk.c',
'shlr/sdb/src/fmt.c',
'shlr/sdb/src/ht_uu.c',
'shlr/sdb/src/ht_up.c',
'shlr/sdb/src/ht_pp.c',
'shlr/sdb/src/ht_pu.c',
'shlr/sdb/src/journal.c',
'shlr/sdb/src/json.c',
'shlr/sdb/src/lock.c',
'shlr/sdb/src/ls.c',
'shlr/sdb/src/match.c',
'shlr/sdb/src/ns.c',
'shlr/sdb/src/num.c',
'shlr/sdb/src/query.c',
'shlr/sdb/src/sdb.c',
'shlr/sdb/src/sdbht.c',
'shlr/sdb/src/util.c',
'shlr/sdb/src/text.c'
]
sdb_inc = [platform_inc, include_directories(join_paths('shlr','sdb','src'))]
# Create sdb_version.h
r = run_command(py3_exe, join_paths('sys','sdb_version.py'), join_paths('shlr','sdb','config.mk'))
if r.returncode() == 0
sdb_version = r.stdout().strip().split('\n')[0]
else
sdb_version = 'unknown'
endif
run_command(py3_exe, '-c', 'with open("shlr/sdb/src/sdb_version.h", "w") as f: f.write("#define SDB_VERSION \"' + sdb_version + '\"")')
sdb_inc_files = [
'shlr/sdb/src/buffer.h',
'shlr/sdb/src/cdb.h',
'shlr/sdb/src/set.h',
'shlr/sdb/src/cdb_make.h',
'shlr/sdb/src/config.h',
'shlr/sdb/src/dict.h',
'shlr/sdb/src/ht_inc.h',
'shlr/sdb/src/ht_pp.h',
'shlr/sdb/src/ht_up.h',
'shlr/sdb/src/ht_uu.h',
'shlr/sdb/src/ht_pu.h',
'shlr/sdb/src/ls.h',
'shlr/sdb/src/sdb.h',
'shlr/sdb/src/sdbht.h',
'shlr/sdb/src/sdb_version.h',
'shlr/sdb/src/types.h'
]
install_headers(sdb_inc_files, install_dir: join_paths(r2_incdir, 'sdb'))
libr2sdb = static_library('r2sdb', sdb_files,
include_directories: sdb_inc,
implicit_include_directories: false,
c_args: host_machine.system() == 'windows' ? '-DSDB_IPI= -DSDB_API=__declspec(dllexport)' : [],
)
sdb_dep = declare_dependency(
link_whole: libr2sdb,
include_directories: sdb_inc
)
sdb_exe = executable('sdb', ['shlr/sdb/src/main.c'] + sdb_files,
include_directories: [
include_directories('shlr/sdb/src')
],
implicit_include_directories: false,
native: true,
)
sdb_gen_cmd_cgen = [
sdb_exe,
'-t',
'-C',
'-o',
'@OUTPUT@',
'@INPUT@'
]
sdb_gen_cmd = [
sdb_exe,
'@OUTPUT@',
'==',
'@INPUT@'
]
# handle spp dependency
spp_files = [
'shlr/spp/spp.c'
]
spp_inc = [platform_inc, include_directories(join_paths('shlr','spp'))]
libr2spp = static_library('r2spp', spp_files,
dependencies: sdb_dep,
include_directories: spp_inc,
c_args: ['-DHAVE_R_UTIL', '-DUSE_R2=1'],
implicit_include_directories: false
)
spp_dep = declare_dependency(
link_with: libr2spp,
include_directories: spp_inc
)
pkgcfg_sanitize_libs = ''
if get_option('b_sanitize').contains('address')
pkgcfg_sanitize_libs += ' -lasan'
endif
if get_option('b_sanitize').contains('undefined')
pkgcfg_sanitize_libs += ' -lubsan'
endif
if get_option('b_sanitize').contains('thread')
pkgcfg_sanitize_libs += ' -fsanitize=thread'
endif
rpath_exe = ''
rpath_lib = ''
if get_option('local') and get_option('default_library') == 'shared'
rpath_exe = '$ORIGIN/../' + get_option('libdir')
rpath_lib = '$ORIGIN'
endif
subdir('shlr/lz4')
subdir('libr/util')
subdir('libr/socket')
subdir('libr/crypto')
subdir('shlr')
subdir('libr/cons')
subdir('shlr/gdb')
subdir('libr/io')
subdir('libr/bp')
subdir('libr/syscall')
subdir('libr/search')
subdir('libr/magic')
subdir('libr/arch')
subdir('libr/flag')
subdir('libr/reg')
subdir('libr/fs')
subdir('libr/bin')
subdir('libr/config')
subdir('libr/parse')
subdir('libr/lang')
subdir('libr/asm')
subdir('libr/anal')
subdir('libr/egg')
subdir('libr/debug')
subdir('libr/core')
subdir('libr/cons/d')
subdir('libr/magic/d')
subdir('libr/flag/d')
subdir('libr/main')
subdir('binr')
if meson.is_subproject()
libr2_dep = declare_dependency(
dependencies: [
r_anal_dep,
r_asm_dep,
r_bp_dep,
r_config_dep,
r_cons_dep,
r_core_dep,
r_main_dep,
r_crypto_dep,
r_debug_dep,
r_egg_dep,
r_flag_dep,
r_fs_dep,
r_bin_dep,
r_io_dep,
r_lang_dep,
r_magic_dep,
r_parse_dep,
r_reg_dep,
r_search_dep,
r_socket_dep,
r_syscall_dep,
r_util_dep
],
include_directories: include_directories('.', 'libr/include'),
version: r2_version
)
endif
if get_option('use_webui')
install_subdir('shlr/www',
install_dir: r2_wwwroot,
strip_directory: true
)
endif
subdir('test/unit')
subdir('test/fuzz')
install_data(
'doc/fortunes.fun',
'doc/fortunes.tips',
install_dir: r2_fortunes
)
if cli_enabled
install_man(
'man/r2agent.1',
'man/r2-docker.1',
'man/r2pm.1',
'man/rabin2.1',
'man/radare2.1',
'man/radiff2.1',
'man/rafind2.1',
'man/ragg2.1',
'man/rahash2.1',
'man/rarun2.1',
'man/rasm2.1',
'man/rax2.1',
'man/ravc2.1',
'man/esil.7'
)
install_data('doc/hud',
install_dir: r2_hud,
rename: 'main'
)
install_data(
'doc/zsh/_r2',
'doc/zsh/_rabin2',
'doc/zsh/_radiff2',
'doc/zsh/_rafind2',
'doc/zsh/_ragg2',
'doc/zsh/_rahash2',
'doc/zsh/_rasm2',
'doc/zsh/_rax2',
install_dir: r2_zsh_compdir
)
if host_machine.system() == 'windows'
meson.add_install_script('sys/create_r2.bat')
else
meson.add_install_script('sys/create_r2.sh')
endif
endif