Skip to content

Commit

Permalink
waf: add support for generating compile_commands.json
Browse files Browse the repository at this point in the history
  • Loading branch information
bugobliterator committed Jun 5, 2024
1 parent 78fcf70 commit 4541d0f
Showing 1 changed file with 84 additions and 1 deletion.
85 changes: 84 additions & 1 deletion Tools/ardupilotwaf/ardupilotwaf.py
Original file line number Diff line number Diff line change
Expand Up @@ -680,7 +680,90 @@ def options(opt):
g.add_option('--ubsan-abort',
action='store_true',
help='''Build using the gcc undefined behaviour sanitizer and abort on error''')



def write_compilation_database(bld):
"""
Write the clang compilation database as JSON
"""
database_file = bld.bldnode.make_node('compile_commands.json')
Logs.info('Build commands will be stored in %s', database_file.path_from(bld.path))
try:
root = database_file.read_json()
except IOError:
root = []
clang_db = dict((x['file'], x) for x in root)
for task in bld.clang_compilation_database_tasks:
try:
cmd = task.last_cmd
except AttributeError:
continue
f_node = task.inputs[0]
filename = f_node.path_from(task.get_cwd())
entry = {
"directory": task.get_cwd().abspath(),
"arguments": cmd,
"file": filename,
}
clang_db[filename] = entry
root = list(clang_db.values())
database_file.write_json(root)

GEN_COMPILE_JSON = False

@feature('cxxstlib', 'cxxprogram')
@before_method('process_rule')
def dry_run_compilation_database(self):
if not hasattr(self, 'bld'):
return
global GEN_COMPILE_JSON
if GEN_COMPILE_JSON:
return
GEN_COMPILE_JSON = True
bld = self.bld
bld.clang_compilation_database_tasks = []
# we need only to generate last_cmd, so override
# exec_command temporarily
def exec_command(bld, *k, **kw):
return 0

for g in bld.groups:
for tg in g:
if bld.targets not in tg.name:
continue
try:
f = tg.post
except AttributeError:
pass
else:
f()

if isinstance(tg, Task.Task):
lst = [tg]
else: lst = tg.tasks
for tsk in lst:
if tsk.__class__.__name__ == "swig":
tsk.runnable_status()
if hasattr(tsk, 'more_tasks'):
lst.extend(tsk.more_tasks)
# Not all dynamic tasks can be processed, in some cases
# one may have to call the method "run()" like this:
#elif tsk.__class__.__name__ == 'src2c':
# tsk.run()
# if hasattr(tsk, 'more_tasks'):
# lst.extend(tsk.more_tasks)

tup = tuple(y for y in [Task.classes.get(x) for x in ('c', 'cxx')] if y)
if isinstance(tsk, tup):
bld.clang_compilation_database_tasks.append(tsk)
tsk.nocache = True
old_exec = tsk.exec_command
tsk.exec_command = exec_command
tsk.run()
tsk.exec_command = old_exec

write_compilation_database(bld)

def build(bld):
bld.add_pre_fun(_process_build_command)
bld.add_pre_fun(_select_programs_from_group)

0 comments on commit 4541d0f

Please sign in to comment.