mirror of
https://gitlab.com/ita1024/waf.git
synced 2025-01-11 10:55:08 +01:00
124 lines
2.7 KiB
Python
124 lines
2.7 KiB
Python
#!/usr/bin/env python
|
|
# encoding: utf-8
|
|
# Thomas Nagy, 2008-2010 (ita)
|
|
|
|
"""
|
|
Execute the tasks with gcc -MD, read the dependencies from the .d file
|
|
and prepare the dependency calculation for the next run
|
|
"""
|
|
|
|
import os, re, threading
|
|
from waflib import Task, Logs, Utils, Errors
|
|
from waflib.Tools import c_preproc
|
|
from waflib.TaskGen import before_method, after_method, feature
|
|
|
|
lock = threading.Lock()
|
|
|
|
preprocessor_flag = '-MD'
|
|
|
|
@feature('cc')
|
|
@before_method('apply_core')
|
|
def add_mmd_cc(self):
|
|
if self.env.get_flat('CFLAGS').find(preprocessor_flag) < 0:
|
|
self.env.append_value('CFLAGS', [preprocessor_flag])
|
|
|
|
@feature('cxx')
|
|
@before_method('apply_core')
|
|
def add_mmd_cxx(self):
|
|
if self.env.get_flat('CXXFLAGS').find(preprocessor_flag) < 0:
|
|
self.env.append_value('CXXFLAGS', [preprocessor_flag])
|
|
|
|
def scan(self):
|
|
"the scanner does not do anything initially"
|
|
nodes = self.generator.bld.node_deps.get(self.uid(), [])
|
|
names = []
|
|
return (nodes, names)
|
|
|
|
re_o = re.compile("\.o$")
|
|
def post_run(self):
|
|
# The following code is executed by threads, it is not safe, so a lock is needed...
|
|
|
|
if getattr(self, 'cached', None):
|
|
return Task.Task.post_run(self)
|
|
|
|
name = self.outputs[0].abspath()
|
|
name = re_o.sub('.d', name)
|
|
txt = Utils.readf(name)
|
|
#os.unlink(name)
|
|
|
|
txt = txt.replace('\\\n', '')
|
|
|
|
lst = txt.strip().split(':')
|
|
val = ":".join(lst[1:])
|
|
val = val.split()
|
|
|
|
nodes = []
|
|
bld = self.generator.bld
|
|
|
|
f = re.compile("^(\.\.)[\\/](.*)$")
|
|
for x in val:
|
|
|
|
node = None
|
|
if os.path.isabs(x):
|
|
|
|
if not c_preproc.go_absolute:
|
|
continue
|
|
|
|
lock.acquire()
|
|
try:
|
|
node = bld.root.find_resource(x)
|
|
finally:
|
|
lock.release()
|
|
else:
|
|
path = bld.bldnode
|
|
x = [k for k in Utils.split_path(x) if k and k != '.']
|
|
while lst and x[0] == '..':
|
|
x = x[1:]
|
|
path = path.parent
|
|
|
|
# when calling find_resource, make sure the path does not begin by '..'
|
|
try:
|
|
lock.acquire()
|
|
node = path.find_resource(x)
|
|
finally:
|
|
lock.release()
|
|
|
|
if not node:
|
|
raise ValueError('could not find %r for %r' % (x, self))
|
|
else:
|
|
if id(node) == id(self.inputs[0]):
|
|
# ignore the source file, it is already in the dependencies
|
|
# this way, successful config tests may be retrieved from the cache
|
|
continue
|
|
|
|
nodes.append(node)
|
|
|
|
Logs.debug('deps: real scanner for %s returned %s' % (str(self), str(nodes)))
|
|
|
|
bld.node_deps[self.uid()] = nodes
|
|
bld.raw_deps[self.uid()] = []
|
|
|
|
try:
|
|
del self.cache_sig
|
|
except:
|
|
pass
|
|
|
|
Task.Task.post_run(self)
|
|
|
|
def sig_implicit_deps(self):
|
|
try:
|
|
return Task.Task.sig_implicit_deps(self)
|
|
except Errors.WafError:
|
|
return Utils.SIG_NIL
|
|
|
|
for name in 'cc cxx'.split():
|
|
try:
|
|
cls = Task.classes[name]
|
|
except KeyError:
|
|
pass
|
|
else:
|
|
cls.post_run = post_run
|
|
cls.scan = scan
|
|
cls.sig_implicit_deps = sig_implicit_deps
|
|
|