Add initial unit test codes using Catch.
Add Kuroga build script.
This commit is contained in:
312
tests/kuroga.py
Executable file
312
tests/kuroga.py
Executable file
@@ -0,0 +1,312 @@
|
||||
#!/usr/bin/env python
|
||||
|
||||
#
|
||||
# Kuroga, single python file meta-build system for ninja
|
||||
# https://github.com/lighttransport/kuroga
|
||||
#
|
||||
# Requirements: python 2.6 or 2.7
|
||||
#
|
||||
# Usage: $ python kuroga.py input.py
|
||||
#
|
||||
|
||||
import imp
|
||||
import re
|
||||
import textwrap
|
||||
import glob
|
||||
import os
|
||||
import sys
|
||||
|
||||
# gcc preset
|
||||
def add_gnu_rule(ninja):
|
||||
ninja.rule('gnucxx', description='CXX $out',
|
||||
command='$gnucxx -MMD -MF $out.d $gnudefines $gnuincludes $gnucxxflags -c $in -o $out',
|
||||
depfile='$out.d', deps='gcc')
|
||||
ninja.rule('gnucc', description='CC $out',
|
||||
command='$gnucc -MMD -MF $out.d $gnudefines $gnuincludes $gnucflags -c $in -o $out',
|
||||
depfile='$out.d', deps='gcc')
|
||||
ninja.rule('gnulink', description='LINK $out', pool='link_pool',
|
||||
command='$gnuld -o $out $in $libs $gnuldflags')
|
||||
ninja.rule('gnuar', description='AR $out', pool='link_pool',
|
||||
command='$gnuar rsc $out $in')
|
||||
ninja.rule('gnustamp', description='STAMP $out', command='touch $out')
|
||||
ninja.newline()
|
||||
|
||||
ninja.variable('gnucxx', 'g++')
|
||||
ninja.variable('gnucc', 'gcc')
|
||||
ninja.variable('gnuld', '$gnucxx')
|
||||
ninja.variable('gnuar', 'ar')
|
||||
ninja.newline()
|
||||
|
||||
# clang preset
|
||||
def add_clang_rule(ninja):
|
||||
ninja.rule('clangcxx', description='CXX $out',
|
||||
command='$clangcxx -MMD -MF $out.d $clangdefines $clangincludes $clangcxxflags -c $in -o $out',
|
||||
depfile='$out.d', deps='gcc')
|
||||
ninja.rule('clangcc', description='CC $out',
|
||||
command='$clangcc -MMD -MF $out.d $clangdefines $clangincludes $clangcflags -c $in -o $out',
|
||||
depfile='$out.d', deps='gcc')
|
||||
ninja.rule('clanglink', description='LINK $out', pool='link_pool',
|
||||
command='$clangld -o $out $in $libs $clangldflags')
|
||||
ninja.rule('clangar', description='AR $out', pool='link_pool',
|
||||
command='$clangar rsc $out $in')
|
||||
ninja.rule('clangstamp', description='STAMP $out', command='touch $out')
|
||||
ninja.newline()
|
||||
|
||||
ninja.variable('clangcxx', 'clang++')
|
||||
ninja.variable('clangcc', 'clang')
|
||||
ninja.variable('clangld', '$clangcxx')
|
||||
ninja.variable('clangar', 'ar')
|
||||
ninja.newline()
|
||||
|
||||
# msvc preset
|
||||
def add_msvc_rule(ninja):
|
||||
ninja.rule('msvccxx', description='CXX $out',
|
||||
command='$msvccxx /TP /showIncludes $msvcdefines $msvcincludes $msvccxxflags -c $in /Fo$out',
|
||||
depfile='$out.d', deps='msvc')
|
||||
ninja.rule('msvccc', description='CC $out',
|
||||
command='$msvccc /TC /showIncludes $msvcdefines $msvcincludes $msvccflags -c $in /Fo$out',
|
||||
depfile='$out.d', deps='msvc')
|
||||
ninja.rule('msvclink', description='LINK $out', pool='link_pool',
|
||||
command='$msvcld $msvcldflags $in $libs /OUT:$out')
|
||||
ninja.rule('msvcar', description='AR $out', pool='link_pool',
|
||||
command='$msvcar $in /OUT:$out')
|
||||
#ninja.rule('msvcstamp', description='STAMP $out', command='touch $out')
|
||||
ninja.newline()
|
||||
|
||||
ninja.variable('msvccxx', 'cl.exe')
|
||||
ninja.variable('msvccc', 'cl.exe')
|
||||
ninja.variable('msvcld', 'link.exe')
|
||||
ninja.variable('msvcar', 'lib.exe')
|
||||
ninja.newline()
|
||||
|
||||
# -- from ninja_syntax.py --
|
||||
def escape_path(word):
|
||||
return word.replace('$ ', '$$ ').replace(' ', '$ ').replace(':', '$:')
|
||||
|
||||
class Writer(object):
|
||||
def __init__(self, output, width=78):
|
||||
self.output = output
|
||||
self.width = width
|
||||
|
||||
def newline(self):
|
||||
self.output.write('\n')
|
||||
|
||||
def comment(self, text, has_path=False):
|
||||
for line in textwrap.wrap(text, self.width - 2, break_long_words=False,
|
||||
break_on_hyphens=False):
|
||||
self.output.write('# ' + line + '\n')
|
||||
|
||||
def variable(self, key, value, indent=0):
|
||||
if value is None:
|
||||
return
|
||||
if isinstance(value, list):
|
||||
value = ' '.join(filter(None, value)) # Filter out empty strings.
|
||||
self._line('%s = %s' % (key, value), indent)
|
||||
|
||||
def pool(self, name, depth):
|
||||
self._line('pool %s' % name)
|
||||
self.variable('depth', depth, indent=1)
|
||||
|
||||
def rule(self, name, command, description=None, depfile=None,
|
||||
generator=False, pool=None, restat=False, rspfile=None,
|
||||
rspfile_content=None, deps=None):
|
||||
self._line('rule %s' % name)
|
||||
self.variable('command', command, indent=1)
|
||||
if description:
|
||||
self.variable('description', description, indent=1)
|
||||
if depfile:
|
||||
self.variable('depfile', depfile, indent=1)
|
||||
if generator:
|
||||
self.variable('generator', '1', indent=1)
|
||||
if pool:
|
||||
self.variable('pool', pool, indent=1)
|
||||
if restat:
|
||||
self.variable('restat', '1', indent=1)
|
||||
if rspfile:
|
||||
self.variable('rspfile', rspfile, indent=1)
|
||||
if rspfile_content:
|
||||
self.variable('rspfile_content', rspfile_content, indent=1)
|
||||
if deps:
|
||||
self.variable('deps', deps, indent=1)
|
||||
|
||||
def build(self, outputs, rule, inputs=None, implicit=None, order_only=None,
|
||||
variables=None):
|
||||
outputs = as_list(outputs)
|
||||
out_outputs = [escape_path(x) for x in outputs]
|
||||
all_inputs = [escape_path(x) for x in as_list(inputs)]
|
||||
|
||||
if implicit:
|
||||
implicit = [escape_path(x) for x in as_list(implicit)]
|
||||
all_inputs.append('|')
|
||||
all_inputs.extend(implicit)
|
||||
if order_only:
|
||||
order_only = [escape_path(x) for x in as_list(order_only)]
|
||||
all_inputs.append('||')
|
||||
all_inputs.extend(order_only)
|
||||
|
||||
self._line('build %s: %s' % (' '.join(out_outputs),
|
||||
' '.join([rule] + all_inputs)))
|
||||
|
||||
if variables:
|
||||
if isinstance(variables, dict):
|
||||
iterator = iter(variables.items())
|
||||
else:
|
||||
iterator = iter(variables)
|
||||
|
||||
for key, val in iterator:
|
||||
self.variable(key, val, indent=1)
|
||||
|
||||
return outputs
|
||||
|
||||
def include(self, path):
|
||||
self._line('include %s' % path)
|
||||
|
||||
def subninja(self, path):
|
||||
self._line('subninja %s' % path)
|
||||
|
||||
def default(self, paths):
|
||||
self._line('default %s' % ' '.join(as_list(paths)))
|
||||
|
||||
def _count_dollars_before_index(self, s, i):
|
||||
"""Returns the number of '$' characters right in front of s[i]."""
|
||||
dollar_count = 0
|
||||
dollar_index = i - 1
|
||||
while dollar_index > 0 and s[dollar_index] == '$':
|
||||
dollar_count += 1
|
||||
dollar_index -= 1
|
||||
return dollar_count
|
||||
|
||||
def _line(self, text, indent=0):
|
||||
"""Write 'text' word-wrapped at self.width characters."""
|
||||
leading_space = ' ' * indent
|
||||
while len(leading_space) + len(text) > self.width:
|
||||
# The text is too wide; wrap if possible.
|
||||
|
||||
# Find the rightmost space that would obey our width constraint and
|
||||
# that's not an escaped space.
|
||||
available_space = self.width - len(leading_space) - len(' $')
|
||||
space = available_space
|
||||
while True:
|
||||
space = text.rfind(' ', 0, space)
|
||||
if (space < 0 or
|
||||
self._count_dollars_before_index(text, space) % 2 == 0):
|
||||
break
|
||||
|
||||
if space < 0:
|
||||
# No such space; just use the first unescaped space we can find.
|
||||
space = available_space - 1
|
||||
while True:
|
||||
space = text.find(' ', space + 1)
|
||||
if (space < 0 or
|
||||
self._count_dollars_before_index(text, space) % 2 == 0):
|
||||
break
|
||||
if space < 0:
|
||||
# Give up on breaking.
|
||||
break
|
||||
|
||||
self.output.write(leading_space + text[0:space] + ' $\n')
|
||||
text = text[space+1:]
|
||||
|
||||
# Subsequent lines are continuations, so indent them.
|
||||
leading_space = ' ' * (indent+2)
|
||||
|
||||
self.output.write(leading_space + text + '\n')
|
||||
|
||||
def close(self):
|
||||
self.output.close()
|
||||
|
||||
|
||||
def as_list(input):
|
||||
if input is None:
|
||||
return []
|
||||
if isinstance(input, list):
|
||||
return input
|
||||
return [input]
|
||||
|
||||
# -- end from ninja_syntax.py --
|
||||
|
||||
def gen(ninja, toolchain, config):
|
||||
|
||||
ninja.variable('ninja_required_version', '1.4')
|
||||
ninja.newline()
|
||||
|
||||
if hasattr(config, "builddir"):
|
||||
builddir = config.builddir[toolchain]
|
||||
ninja.variable(toolchain + 'builddir', builddir)
|
||||
else:
|
||||
builddir = ''
|
||||
|
||||
ninja.variable(toolchain + 'defines', config.defines[toolchain] or [])
|
||||
ninja.variable(toolchain + 'includes', config.includes[toolchain] or [])
|
||||
ninja.variable(toolchain + 'cflags', config.cflags[toolchain] or [])
|
||||
ninja.variable(toolchain + 'cxxflags', config.cxxflags[toolchain] or [])
|
||||
ninja.variable(toolchain + 'ldflags', config.ldflags[toolchain] or [])
|
||||
ninja.newline()
|
||||
|
||||
if hasattr(config, "link_pool_depth"):
|
||||
ninja.pool('link_pool', depth=config.link_pool_depth)
|
||||
else:
|
||||
ninja.pool('link_pool', depth=4)
|
||||
ninja.newline()
|
||||
|
||||
# Add default toolchain(gnu, clang and msvc)
|
||||
add_gnu_rule(ninja)
|
||||
add_clang_rule(ninja)
|
||||
add_msvc_rule(ninja)
|
||||
|
||||
obj_files = []
|
||||
|
||||
cc = toolchain + 'cc'
|
||||
cxx = toolchain + 'cxx'
|
||||
link = toolchain + 'link'
|
||||
ar = toolchain + 'ar'
|
||||
|
||||
if hasattr(config, "cxx_files"):
|
||||
for src in config.cxx_files:
|
||||
srcfile = src
|
||||
obj = os.path.splitext(srcfile)[0] + '.o'
|
||||
obj = os.path.join(builddir, obj);
|
||||
obj_files.append(obj)
|
||||
ninja.build(obj, cxx, srcfile)
|
||||
ninja.newline()
|
||||
|
||||
if hasattr(config, "c_files"):
|
||||
for src in config.c_files:
|
||||
srcfile = src
|
||||
obj = os.path.splitext(srcfile)[0] + '.o'
|
||||
obj = os.path.join(builddir, obj);
|
||||
obj_files.append(obj)
|
||||
ninja.build(obj, cc, srcfile)
|
||||
ninja.newline()
|
||||
|
||||
targetlist = []
|
||||
if hasattr(config, "exe"):
|
||||
ninja.build(config.exe, link, obj_files)
|
||||
targetlist.append(config.exe)
|
||||
|
||||
if hasattr(config, "staticlib"):
|
||||
ninja.build(config.staticlib, ar, obj_files)
|
||||
targetlist.append(config.staticlib)
|
||||
|
||||
ninja.build('all', 'phony', targetlist)
|
||||
ninja.newline()
|
||||
|
||||
ninja.default('all')
|
||||
|
||||
def main():
|
||||
if len(sys.argv) < 2:
|
||||
print("Usage: python kuroga.py config.py")
|
||||
sys.exit(1)
|
||||
|
||||
config = imp.load_source("config", sys.argv[1])
|
||||
|
||||
f = open('build.ninja', 'w')
|
||||
ninja = Writer(f)
|
||||
|
||||
if hasattr(config, "register_toolchain"):
|
||||
config.register_toolchain(ninja)
|
||||
|
||||
gen(ninja, config.toolchain, config)
|
||||
f.close()
|
||||
|
||||
main()
|
||||
Reference in New Issue
Block a user