Book a Demo!
CoCalc Logo Icon
StoreFeaturesDocsShareSupportNewsAboutPoliciesSign UpSign In
Ardupilot
GitHub Repository: Ardupilot/ardupilot
Path: blob/master/Tools/ardupilotwaf/dronecangen.py
9843 views
1
# encoding: utf-8
2
3
# flake8: noqa
4
5
"""
6
generate DSDLC headers for uavcan
7
"""
8
9
from waflib import Logs, Task, Utils, Node
10
from waflib.TaskGen import feature, before_method, extension
11
import os
12
import os.path
13
from xml.etree import ElementTree as et
14
import subprocess
15
16
class dronecangen(Task.Task):
17
"""generate uavcan header files"""
18
color = 'BLUE'
19
before = 'cxx c'
20
21
def run(self):
22
python = self.env.get_flat('PYTHON')
23
out = self.env.get_flat('OUTPUT_DIR')
24
src = self.env.get_flat('SRC')
25
dsdlc = self.env.get_flat("DC_DSDL_COMPILER_DIR")
26
27
cmd = ['{}'.format(python),
28
'{}/dronecan_dsdlc.py'.format(dsdlc),
29
'-O{}'.format(out)] + [x.abspath() for x in self.inputs]
30
ret = self.exec_command(cmd)
31
if ret != 0:
32
# ignore if there was a signal to the interpreter rather
33
# than a real error in the script. Some environments use a
34
# signed and some an unsigned return for this
35
if ret > 128 or ret < 0:
36
Logs.warn('dronecangen crashed with code: {}'.format(ret))
37
ret = 0
38
else:
39
Logs.warn('dronecangen: cmd=%s ' % str(cmd))
40
# re-run command with stdout visible to see errors
41
subprocess.call(cmd)
42
Logs.error('dronecangen returned {} error code'.format(ret))
43
return ret
44
45
def post_run(self):
46
super(dronecangen, self).post_run()
47
for header in self.generator.output_dir.ant_glob("*.h **/*.h", remove=False):
48
header.sig = header.cache_sig = self.cache_sig
49
50
def options(opt):
51
opt.load('python')
52
53
@feature('dronecangen')
54
@before_method('process_rule')
55
def process_dronecangen(self):
56
if not hasattr(self, 'output_dir'):
57
self.bld.fatal('dronecangen: missing option output_dir')
58
59
inputs = self.to_nodes(self.source)
60
# depend on each message file in the source so rebuilds will occur properly
61
deps = []
62
for inp in inputs:
63
deps.extend(inp.ant_glob("**/*.uavcan"))
64
# also depend on the generator source itself
65
dsdlc_dir = self.env.get_flat("DC_DSDL_COMPILER_DIR")
66
dsdlc = self.bld.root.find_node(dsdlc_dir) # expected to be absolute
67
if dsdlc is None:
68
self.bld.fatal("dronecangen: waf couldn't find dsdlc at abspath {}".format(dsdlc_dir))
69
deps.extend(dsdlc.ant_glob("**/*.py **/*.em"))
70
outputs = []
71
72
self.source = []
73
74
if not isinstance(self.output_dir, Node.Node):
75
self.output_dir = self.bld.bldnode.find_or_declare(self.output_dir)
76
77
task = self.create_task('dronecangen', inputs, outputs)
78
task.dep_nodes = deps
79
task.env['OUTPUT_DIR'] = self.output_dir.abspath()
80
81
task.env.env = dict(os.environ)
82
83
def configure(cfg):
84
"""
85
setup environment for uavcan header generator
86
"""
87
env = cfg.env
88
env.DC_DSDL_COMPILER_DIR = cfg.srcnode.make_node('modules/DroneCAN/dronecan_dsdlc/').abspath()
89
cfg.msg('DC_DSDL compiler in', env.DC_DSDL_COMPILER_DIR)
90
91