1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
|
#! /usr/bin/env python
# Thomas Nagy, 2011 (ita)
"""
Create _moc.cpp files
The builds are 30-40% faster when .moc files are included,
you should NOT use this tool. If you really
really want it:
def configure(conf):
conf.load('compiler_cxx qt4')
conf.load('slow_qt4')
See playground/slow_qt/wscript for a complete example.
"""
from waflib.TaskGen import extension
from waflib import Task
import waflib.Tools.qt4
import waflib.Tools.cxx
@extension(*waflib.Tools.qt4.EXT_QT4)
def cxx_hook(self, node):
return self.create_compiled_task('cxx_qt', node)
class cxx_qt(Task.classes['cxx']):
def runnable_status(self):
ret = Task.classes['cxx'].runnable_status(self)
if ret != Task.ASK_LATER and not getattr(self, 'moc_done', None):
try:
cache = self.generator.moc_cache
except AttributeError:
cache = self.generator.moc_cache = {}
deps = self.generator.bld.node_deps[self.uid()]
for x in [self.inputs[0]] + deps:
if x.read().find('Q_OBJECT') > 0:
# process "foo.h -> foo.moc" only if "foo.cpp" is in the sources for the current task generator
# this code will work because it is in the main thread (runnable_status)
if x.name.rfind('.') > -1: # a .h file...
name = x.name[:x.name.rfind('.')]
for tsk in self.generator.compiled_tasks:
if tsk.inputs and tsk.inputs[0].name.startswith(name):
break
else:
# no corresponding file, continue
continue
# the file foo.cpp could be compiled for a static and a shared library - hence the %number in the name
cxx_node = x.parent.get_bld().make_node(x.name.replace('.', '_') + '_%d_moc.cpp' % self.generator.idx)
if cxx_node in cache:
continue
cache[cxx_node] = self
tsk = Task.classes['moc'](env=self.env, generator=self.generator)
tsk.set_inputs(x)
tsk.set_outputs(cxx_node)
if x.name.endswith('.cpp'):
# moc is trying to be too smart but it is too dumb:
# why forcing the #include when Q_OBJECT is in the cpp file?
gen = self.generator.bld.producer
gen.outstanding.append(tsk)
gen.total += 1
self.set_run_after(tsk)
else:
cxxtsk = Task.classes['cxx'](env=self.env, generator=self.generator)
cxxtsk.set_inputs(tsk.outputs)
cxxtsk.set_outputs(cxx_node.change_ext('.o'))
cxxtsk.set_run_after(tsk)
try:
self.more_tasks.extend([tsk, cxxtsk])
except AttributeError:
self.more_tasks = [tsk, cxxtsk]
try:
link = self.generator.link_task
except AttributeError:
pass
else:
link.set_run_after(cxxtsk)
link.inputs.extend(cxxtsk.outputs)
link.inputs.sort(key=lambda x: x.abspath())
self.moc_done = True
for t in self.run_after:
if not t.hasrun:
return Task.ASK_LATER
return ret
|