2
0
mirror of https://gitlab.com/ita1024/waf.git synced 2025-01-09 01:45:08 +01:00

Update playground/dynamic_headers - Issue 1530

This commit is contained in:
Thomas Nagy 2015-01-31 23:07:36 +01:00
parent 3064694440
commit 15d14c7bdf
No known key found for this signature in database
GPG Key ID: 67A565EDFDF90E64
6 changed files with 78 additions and 160 deletions

View File

@ -1,5 +0,0 @@
#include "a.h"
int main() {
return 0;
}

View File

@ -1,77 +1,109 @@
#! /usr/bin/env python
# encoding: utf-8
# Thomas Nagy, 2010 (ita)
VERSION='0.0.1'
APPNAME='dynamic_header_test'
top = '.'
out = 'build'
from waflib import Task, Errors
from waflib.Tools import c
def options(opt):
opt.load('compiler_cxx')
opt.load('compiler_c')
def configure(conf):
conf.load('compiler_cxx')
conf.load('compiler_c')
def build(bld):
bld.program(source='main.cpp', target='app', includes='.')
bld(features = 'c cprogram',
target = 'test',
source = 'src/main.c src/test.c',
includes = 'src')
# --------------------------------------------------------------------
# Dynamic header creation, the file "a.h" from main.cpp is created
# during the build. You may use this code sample to create more complicated
# dynamic header generators
class mock(Task.Task):
run_str = 'cp ${SRC} ${TGT}'
color = 'BLUE'
from waflib import Task
from waflib.Tools.cxx import cxx
class test(Task.Task):
run_str = 'touch ${TGT}'
# it would be better to replace the scan() method, but this is doable too
def runnable_status(self):
ret = super(cxx, self).runnable_status()
try:
shared = self.generator.bld.shared_tasks
except AttributeError:
shared = self.generator.bld.shared_tasks = {}
ret = self.runnable_status_dynamic_headers2()
if ret != Task.ASK_LATER:
if ret != Task.ASK_LATER and not getattr(self, 'all_mock_done', False):
self.all_mock_done = True # run once
bld = self.generator.bld
add = False
# first pass over the unresolved files found by the scanner
for x in self.generator.bld.raw_deps[self.uid()]:
if x == 'a.h':
tgt = self.generator.path.find_or_declare(x)
try:
mock_tasks = bld.mock_tasks
except AttributeError:
mock_tasks = bld.mock_tasks = {}
for x in bld.raw_deps[self.uid()]:
if x.startswith('mock_'):
h_file = x[5:]
for k in [self.inputs[0].parent] + self.generator.includes_nodes:
h_node = k.find_node(h_file)
if h_node:
break
if not h_node:
raise Errors.WafError('no header for %s' % x)
m_node = h_node.parent.find_or_declare(x)
try:
tsk = shared[tgt]
tsk = mock_tasks[m_node]
except KeyError:
tsk = shared[tgt] = self.generator.create_task('test', [], tgt)
self.set_run_after(tsk)
self.generator.bld.producer.outstanding.append(tsk)
add = True
tsk = mock_tasks[m_node] = self.generator.create_task('mock', [h_node], [m_node])
bld.producer.outstanding.insert(0, tsk)
bld.producer.total += 1
# preprocessor cache :-/
try:
for key in list(bld.cache_nd.keys()):
if key[1] == x:
del bld.cache_nd[key]
except (KeyError, AttributeError):
pass
add = True
self.set_run_after(tsk)
if add:
# remove cache entries to force a clean scanner execution
# recompute the task signature
delattr(self, 'cache_sig')
self.generator.bld.cache_nd = {}
del self.generator.bld.task_sigs[(self.uid(), 'imp')] # will have to rescan the C file
del bld.task_sigs[(self.uid(), 'imp')]
return self.runnable_status()
# second pass over the existing nodes - corresponding tasks are still necessary
for x in self.generator.bld.node_deps[self.uid()]:
if x.name == 'a.h':
for x in bld.node_deps[self.uid()]:
if x.name.startswith('mock_'):
h_node = x.parent.get_src().find_node(x.name[5:])
if not h_node:
raise Errors.WafError('no header for %s' % x.name)
try:
tsk = shared[x]
tsk = mock_tasks[x]
except KeyError:
tsk = shared[x] = self.generator.create_task('test', [], x)
self.set_run_after(tsk)
self.generator.bld.producer.outstanding.append(tsk)
add = True
tsk = mock_tasks[x] = self.generator.create_task('mock', [h_node], [x])
bld.producer.outstanding.insert(0, tsk)
bld.producer.total += 1
add = True
self.set_run_after(tsk)
# node get_bld_sig cache :-/
try:
delattr(x, 'cache_sig')
except AttributeError:
pass
if add:
# no need to rescan anything, but recompute the signature after the dependent task is executed
# recompute the task signature
delattr(self, 'cache_sig')
return self.runnable_status()
return ret
cxx.runnable_status = runnable_status
c.c.runnable_status_dynamic_headers2 = c.c.runnable_status
c.c.runnable_status = runnable_status

View File

@ -1,109 +0,0 @@
#! /usr/bin/env python
from waflib import Task, Errors
from waflib.Tools import c
def options(opt):
opt.load('compiler_c')
def configure(conf):
conf.load('compiler_c')
def build(bld):
bld(features = 'c cprogram',
target = 'test',
source = 'src/main.c src/test.c',
includes = 'src')
class mock(Task.Task):
run_str = 'cp ${SRC} ${TGT}'
color = 'BLUE'
# it would be better to replace the scan() method, but this is doable too
def runnable_status(self):
ret = self.runnable_status_dynamic_headers2()
if ret != Task.ASK_LATER and not getattr(self, 'all_mock_done', False):
self.all_mock_done = True # run once
bld = self.generator.bld
add = False
try:
mock_tasks = bld.mock_tasks
except AttributeError:
mock_tasks = bld.mock_tasks = {}
for x in bld.raw_deps[self.uid()]:
if x.startswith('mock_'):
h_file = x[5:]
for k in [self.inputs[0].parent] + self.generator.includes_nodes:
h_node = k.find_node(h_file)
if h_node:
break
if not h_node:
raise Errors.WafError('no header for %s' % x)
m_node = h_node.parent.find_or_declare(x)
try:
tsk = mock_tasks[m_node]
except KeyError:
tsk = mock_tasks[m_node] = self.generator.create_task('mock', [h_node], [m_node])
bld.producer.outstanding.insert(0, tsk)
bld.producer.total += 1
# preprocessor cache :-/
try:
for key in list(bld.cache_nd.keys()):
if key[1] == x:
del bld.cache_nd[key]
except (KeyError, AttributeError):
pass
add = True
self.set_run_after(tsk)
if add:
# recompute the task signature
delattr(self, 'cache_sig')
del bld.task_sigs[(self.uid(), 'imp')]
return self.runnable_status()
for x in bld.node_deps[self.uid()]:
if x.name.startswith('mock_'):
h_node = x.parent.get_src().find_node(x.name[5:])
if not h_node:
raise Errors.WafError('no header for %s' % x.name)
try:
tsk = mock_tasks[x]
except KeyError:
tsk = mock_tasks[x] = self.generator.create_task('mock', [h_node], [x])
bld.producer.outstanding.insert(0, tsk)
bld.producer.total += 1
add = True
self.set_run_after(tsk)
# node get_bld_sig cache :-/
try:
delattr(x, 'cache_sig')
except AttributeError:
pass
if add:
# recompute the task signature
delattr(self, 'cache_sig')
return self.runnable_status()
return ret
c.c.runnable_status_dynamic_headers2 = c.c.runnable_status
c.c.runnable_status = runnable_status