123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734 |
- #!/usr/bin/python3
- # Check header contents against the given standard.
- # Copyright (C) 2018-2019 Free Software Foundation, Inc.
- # This file is part of the GNU C Library.
- #
- # The GNU C Library is free software; you can redistribute it and/or
- # modify it under the terms of the GNU Lesser General Public
- # License as published by the Free Software Foundation; either
- # version 2.1 of the License, or (at your option) any later version.
- #
- # The GNU C Library is distributed in the hope that it will be useful,
- # but WITHOUT ANY WARRANTY; without even the implied warranty of
- # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
- # Lesser General Public License for more details.
- #
- # You should have received a copy of the GNU Lesser General Public
- # License along with the GNU C Library; if not, see
- # <http://www.gnu.org/licenses/>.
- import argparse
- import fnmatch
- import os.path
- import re
- import subprocess
- import sys
- import tempfile
- import glibcconform
- class CompileSubTest(object):
- """A compilation subtest."""
- def __init__(self, name, text):
- """Initialize a CompileSubTest object."""
- self.run_early = False
- self.name = name
- self.text = text
- def run(self, header_tests):
- """Run a compilation subtest."""
- header_tests.compile_test(self.name, self.text)
- class ExecuteSubTest(object):
- """An execution subtest."""
- def __init__(self, name, text):
- """Initialize an ExecuteSubTest object."""
- self.run_early = False
- self.name = name
- self.text = text
- def run(self, header_tests):
- """Run an execution subtest."""
- header_tests.execute_test(self.name, self.text)
- class ElementTest(object):
- """Test for an element of a structure or union type."""
- def __init__(self, dummy, type_name, member_type, member_name, *rest):
- """Initialize an ElementTest object."""
- self.type_name = type_name
- self.member_type = member_type
- self.member_name = member_name
- self.rest = ' '.join(rest)
- self.allow_name = self.member_name
- def gen_subtests(self):
- """Generate subtests for an ElementTest."""
- text = ('%(type_name)s a_%(num)d;\n'
- '%(type_name)s b_%(num)d;\n'
- 'extern void xyzzy_%(num)d '
- '(__typeof__ (&b_%(num)d.%(member_name)s), '
- '__typeof__ (&a_%(num)d.%(member_name)s), unsigned);\n'
- 'void foobarbaz_%(num)d (void) {\n'
- 'xyzzy_%(num)d (&a_%(num)d.%(member_name)s, '
- '&b_%(num)d.%(member_name)s, '
- 'sizeof (a_%(num)d.%(member_name)s));\n'
- '}\n'
- % vars(self))
- self.subtests.append(CompileSubTest(
- 'Availability of member %s' % self.member_name,
- text))
- text = ('%(type_name)s a2_%(num)d;\n'
- 'extern %(member_type)s b2_%(num)d%(rest)s;\n'
- 'extern __typeof__ (a2_%(num)d.%(member_name)s) b2_%(num)d;\n'
- % vars(self))
- self.subtests.append(CompileSubTest(
- 'Type of member %s' % self.member_name,
- text))
- class ConstantTest(object):
- """Test for a macro or constant."""
- def __init__(self, symbol_type, symbol, extra1=None, extra2=None,
- extra3=None):
- """Initialize a ConstantTest object."""
- self.symbol_type = symbol_type
- self.symbol = symbol
- # A comparison operation may be specified without a type.
- if extra2 is not None and extra3 is None:
- self.c_type = None
- self.op = extra1
- self.value = extra2
- else:
- self.c_type = extra1
- self.op = extra2
- self.value = extra3
- self.allow_name = self.symbol
- def gen_subtests(self):
- """Generate subtests for a ConstantTest."""
- if 'macro' in self.symbol_type:
- text = ('#ifndef %(symbol)s\n'
- '# error "Macro %(symbol)s not defined"\n'
- '#endif\n'
- % vars(self))
- self.subtests.append(CompileSubTest(
- 'Availability of macro %s' % self.symbol,
- text))
- if 'constant' in self.symbol_type:
- text = ('__typeof__ (%(symbol)s) a_%(num)d = %(symbol)s;\n'
- % vars(self))
- self.subtests.append(CompileSubTest(
- 'Availability of constant %s' % self.symbol,
- text))
- if self.symbol_type == 'macro-int-constant':
- sym_bits_def_neg = ''.join(
- '# if %s & (1LL << %d)\n'
- '# define conformtest_%d_bit_%d 0LL\n'
- '# else\n'
- '# define conformtest_%d_bit_%d (1LL << %d)\n'
- '# endif\n'
- % (self.symbol, i, self.num, i, self.num, i, i)
- for i in range(63))
- sym_bits_or_neg = '|'.join('conformtest_%d_bit_%d' % (self.num, i)
- for i in range(63))
- sym_bits_def_pos = ''.join(
- '# if %s & (1ULL << %d)\n'
- '# define conformtest_%d_bit_%d (1ULL << %d)\n'
- '# else\n'
- '# define conformtest_%d_bit_%d 0ULL\n'
- '# endif\n'
- % (self.symbol, i, self.num, i, i, self.num, i)
- for i in range(64))
- sym_bits_or_pos = '|'.join('conformtest_%d_bit_%d' % (self.num, i)
- for i in range(64))
- text = ('#if %s < 0\n'
- '# define conformtest_%d_negative 1\n'
- '%s'
- '# define conformtest_%d_value ~(%s)\n'
- '#else\n'
- '# define conformtest_%d_negative 0\n'
- '%s'
- '# define conformtest_%d_value (%s)\n'
- '#endif\n'
- '_Static_assert (((%s < 0) == conformtest_%d_negative) '
- '&& (%s == conformtest_%d_value), '
- '"value match inside and outside #if");\n'
- % (self.symbol, self.num, sym_bits_def_neg, self.num,
- sym_bits_or_neg, self.num, sym_bits_def_pos, self.num,
- sym_bits_or_pos, self.symbol, self.num, self.symbol,
- self.num))
- self.subtests.append(CompileSubTest(
- '#if usability of symbol %s'% self.symbol,
- text))
- if self.c_type is not None:
- if self.c_type.startswith('promoted:'):
- c_type = self.c_type[len('promoted:'):]
- text = ('__typeof__ ((%s) 0 + (%s) 0) a2_%d;\n'
- % (c_type, c_type, self.num))
- else:
- text = '__typeof__ ((%s) 0) a2_%d;\n' % (self.c_type, self.num)
- text += 'extern __typeof__ (%s) a2_%d;\n' % (self.symbol, self.num)
- self.subtests.append(CompileSubTest(
- 'Type of symbol %s' % self.symbol,
- text))
- if self.op is not None:
- text = ('_Static_assert (%(symbol)s %(op)s %(value)s, '
- '"value constraint");\n'
- % vars(self))
- self.subtests.append(CompileSubTest(
- 'Value of symbol %s' % self.symbol,
- text))
- class SymbolTest(object):
- """Test for a symbol (not a compile-time constant)."""
- def __init__(self, dummy, symbol, value=None):
- """Initialize a SymbolTest object."""
- self.symbol = symbol
- self.value = value
- self.allow_name = self.symbol
- def gen_subtests(self):
- """Generate subtests for a SymbolTest."""
- text = ('void foobarbaz_%(num)d (void) {\n'
- '__typeof__ (%(symbol)s) a_%(num)d = %(symbol)s;\n'
- '}\n'
- % vars(self))
- self.subtests.append(CompileSubTest(
- 'Availability of symbol %s' % self.symbol,
- text))
- if self.value is not None:
- text = ('int main (void) { return %(symbol)s != %(symbol)s; }\n'
- % vars(self))
- self.subtests.append(ExecuteSubTest(
- 'Value of symbol %s' % self.symbol,
- text))
- class TypeTest(object):
- """Test for a type name."""
- def __init__(self, dummy, type_name):
- """Initialize a TypeTest object."""
- self.type_name = type_name
- if type_name.startswith('struct '):
- self.allow_name = type_name[len('struct '):]
- self.maybe_opaque = False
- elif type_name.startswith('union '):
- self.allow_name = type_name[len('union '):]
- self.maybe_opaque = False
- else:
- self.allow_name = type_name
- self.maybe_opaque = True
- def gen_subtests(self):
- """Generate subtests for a TypeTest."""
- text = ('%s %sa_%d;\n'
- % (self.type_name, '*' if self.maybe_opaque else '', self.num))
- self.subtests.append(CompileSubTest(
- 'Availability of type %s' % self.type_name,
- text))
- class TagTest(object):
- """Test for a tag name."""
- def __init__(self, dummy, type_name):
- """Initialize a TagTest object."""
- self.type_name = type_name
- if type_name.startswith('struct '):
- self.allow_name = type_name[len('struct '):]
- elif type_name.startswith('union '):
- self.allow_name = type_name[len('union '):]
- else:
- raise ValueError('unexpected kind of tag: %s' % type_name)
- def gen_subtests(self):
- """Generate subtests for a TagTest."""
- # If the tag is not declared, these function prototypes have
- # incompatible types.
- text = ('void foo_%(num)d (%(type_name)s *);\n'
- 'void foo_%(num)d (%(type_name)s *);\n'
- % vars(self))
- self.subtests.append(CompileSubTest(
- 'Availability of tag %s' % self.type_name,
- text))
- class FunctionTest(object):
- """Test for a function."""
- def __init__(self, dummy, return_type, function_name, *args):
- """Initialize a FunctionTest object."""
- self.function_name_full = function_name
- self.args = ' '.join(args)
- if function_name.startswith('(*'):
- # Function returning a pointer to function.
- self.return_type = '%s (*' % return_type
- self.function_name = function_name[len('(*'):]
- else:
- self.return_type = return_type
- self.function_name = function_name
- self.allow_name = self.function_name
- def gen_subtests(self):
- """Generate subtests for a FunctionTest."""
- text = ('%(return_type)s (*foobarbaz_%(num)d) %(args)s '
- '= %(function_name)s;\n'
- % vars(self))
- self.subtests.append(CompileSubTest(
- 'Availability of function %s' % self.function_name,
- text))
- text = ('extern %(return_type)s (*foobarbaz2_%(num)d) %(args)s;\n'
- 'extern __typeof__ (&%(function_name)s) foobarbaz2_%(num)d;\n'
- % vars(self))
- self.subtests.append(CompileSubTest(
- 'Type of function %s' % self.function_name,
- text))
- class VariableTest(object):
- """Test for a variable."""
- def __init__(self, dummy, var_type, var_name, *rest):
- """Initialize a VariableTest object."""
- self.var_type = var_type
- self.var_name = var_name
- self.rest = ' '.join(rest)
- self.allow_name = var_name
- def gen_subtests(self):
- """Generate subtests for a VariableTest."""
- text = ('typedef %(var_type)s xyzzy_%(num)d%(rest)s;\n'
- 'xyzzy_%(num)d *foobarbaz_%(num)d = &%(var_name)s;\n'
- % vars(self))
- self.subtests.append(CompileSubTest(
- 'Availability of variable %s' % self.var_name,
- text))
- text = ('extern %(var_type)s %(var_name)s%(rest)s;\n'
- % vars(self))
- self.subtests.append(CompileSubTest(
- 'Type of variable %s' % self.var_name,
- text))
- class MacroFunctionTest(object):
- """Test for a possibly macro-only function."""
- def __init__(self, dummy, return_type, function_name, *args):
- """Initialize a MacroFunctionTest object."""
- self.return_type = return_type
- self.function_name = function_name
- self.args = ' '.join(args)
- self.allow_name = function_name
- def gen_subtests(self):
- """Generate subtests for a MacroFunctionTest."""
- text = ('#ifndef %(function_name)s\n'
- '%(return_type)s (*foobarbaz_%(num)d) %(args)s '
- '= %(function_name)s;\n'
- '#endif\n'
- % vars(self))
- self.subtests.append(CompileSubTest(
- 'Availability of macro %s' % self.function_name,
- text))
- text = ('#ifndef %(function_name)s\n'
- 'extern %(return_type)s (*foobarbaz2_%(num)d) %(args)s;\n'
- 'extern __typeof__ (&%(function_name)s) foobarbaz2_%(num)d;\n'
- '#endif\n'
- % vars(self))
- self.subtests.append(CompileSubTest(
- 'Type of macro %s' % self.function_name,
- text))
- class MacroStrTest(object):
- """Test for a string-valued macro."""
- def __init__(self, dummy, macro_name, value):
- """Initialize a MacroStrTest object."""
- self.macro_name = macro_name
- self.value = value
- self.allow_name = macro_name
- def gen_subtests(self):
- """Generate subtests for a MacroStrTest."""
- text = ('#ifndef %(macro_name)s\n'
- '# error "Macro %(macro_name)s not defined"\n'
- '#endif\n'
- % vars(self))
- self.subtests.append(CompileSubTest(
- 'Availability of macro %s' % self.macro_name,
- text))
- # We can't include <string.h> here.
- text = ('extern int (strcmp)(const char *, const char *);\n'
- 'int main (void) { return (strcmp) (%(macro_name)s, '
- '%(value)s) != 0; }\n'
- % vars(self))
- self.subtests.append(ExecuteSubTest(
- 'Value of macro %s' % self.macro_name,
- text))
- class HeaderTests(object):
- """The set of tests run for a header."""
- def __init__(self, header, standard, cc, flags, cross, xfail):
- """Initialize a HeaderTests object."""
- self.header = header
- self.standard = standard
- self.cc = cc
- self.flags = flags
- self.cross = cross
- self.xfail_str = xfail
- self.cflags_namespace = ('%s -fno-builtin %s -D_ISOMAC'
- % (flags, glibcconform.CFLAGS[standard]))
- # When compiling the conformance test programs, use of
- # __attribute__ in headers is disabled because of attributes
- # that affect the types of functions as seen by typeof.
- self.cflags = "%s '-D__attribute__(x)='" % self.cflags_namespace
- self.tests = []
- self.allow = set()
- self.allow_fnmatch = set()
- self.headers_handled = set()
- self.num_tests = 0
- self.total = 0
- self.skipped = 0
- self.errors = 0
- self.xerrors = 0
- def add_allow(self, name, pattern_ok):
- """Add an identifier as an allowed token for this header.
- If pattern_ok, fnmatch patterns are OK as well as
- identifiers.
- """
- if re.fullmatch(r'[A-Za-z_][A-Za-z0-9_]*', name):
- self.allow.add(name)
- elif pattern_ok:
- self.allow_fnmatch.add(name)
- else:
- raise ValueError('bad identifier: %s' % name)
- def check_token(self, bad_tokens, token):
- """Check whether an identifier token is allowed, and record it in
- bad_tokens if not.
- """
- if token.startswith('_'):
- return
- if token in glibcconform.KEYWORDS[self.standard]:
- return
- if token in self.allow:
- return
- for pattern in self.allow_fnmatch:
- if fnmatch.fnmatch(token, pattern):
- return
- bad_tokens.add(token)
- def handle_test_line(self, line, allow):
- """Handle a single line in the test data.
- If allow is true, the header is one specified in allow-header
- and so tests are marked as allowed for namespace purposes but
- otherwise ignored.
- """
- orig_line = line
- xfail = False
- if line.startswith('xfail-'):
- xfail = True
- line = line[len('xfail-'):]
- else:
- match = re.match(r'xfail\[(.*?)\]-(.*)', line)
- if match:
- xfail_cond = match.group(1)
- line = match.group(2)
- # "xfail[cond]-" or "xfail[cond1|cond2|...]-" means a
- # failure of the test is allowed if any of the listed
- # conditions are in the --xfail command-line option
- # argument.
- if self.xfail_str and re.search(r'\b(%s)\b' % xfail_cond,
- self.xfail_str):
- xfail = True
- optional = False
- if line.startswith('optional-'):
- optional = True
- line = line[len('optional-'):]
- # Tokens in test data are space-separated, except for {...}
- # tokens that may contain spaces.
- tokens = []
- while line:
- match = re.match(r'\{(.*?)\}(.*)', line)
- if match:
- tokens.append(match.group(1))
- line = match.group(2)
- else:
- match = re.match(r'([^ ]*)(.*)', line)
- tokens.append(match.group(1))
- line = match.group(2)
- line = line.strip()
- if tokens[0] == 'allow-header':
- if len(tokens) != 2 or xfail or optional:
- raise ValueError('bad allow-header line: %s' % orig_line)
- if tokens[1] not in self.headers_handled:
- self.load_tests(tokens[1], True)
- return
- if tokens[0] == 'allow':
- if len(tokens) != 2 or xfail or optional:
- raise ValueError('bad allow line: %s' % orig_line)
- self.add_allow(tokens[1], True)
- return
- test_classes = {'element': ElementTest,
- 'macro': ConstantTest,
- 'constant': ConstantTest,
- 'macro-constant': ConstantTest,
- 'macro-int-constant': ConstantTest,
- 'symbol': SymbolTest,
- 'type': TypeTest,
- 'tag': TagTest,
- 'function': FunctionTest,
- 'variable': VariableTest,
- 'macro-function': MacroFunctionTest,
- 'macro-str': MacroStrTest}
- test = test_classes[tokens[0]](*tokens)
- test.xfail = xfail
- test.optional = optional
- test.num = self.num_tests
- test.subtests = []
- self.num_tests += 1
- self.add_allow(test.allow_name, False)
- if not allow:
- test.gen_subtests()
- self.tests.append(test)
- def load_tests(self, header, allow):
- """Load tests of a header.
- If allow is true, the header is one specified in allow-header
- and so tests are marked as allowed for namespace purposes but
- otherwise ignored.
- """
- self.headers_handled.add(header)
- header_s = header.replace('/', '_')
- temp_file = os.path.join(self.temp_dir, 'header-data-%s' % header_s)
- cmd = ('%s -E -D%s -std=c99 -x c data/%s-data > %s'
- % (self.cc, self.standard, header, temp_file))
- subprocess.check_call(cmd, shell=True)
- with open(temp_file, 'r') as tests:
- for line in tests:
- line = line.strip()
- if line == '' or line.startswith('#'):
- continue
- self.handle_test_line(line, allow)
- def note_error(self, name, xfail):
- """Note a failing test."""
- if xfail:
- print('XFAIL: %s' % name)
- self.xerrors += 1
- else:
- print('FAIL: %s' % name)
- self.errors += 1
- sys.stdout.flush()
- def note_skip(self, name):
- """Note a skipped test."""
- print('SKIP: %s' % name)
- self.skipped += 1
- sys.stdout.flush()
- def compile_test(self, name, text):
- """Run a compilation test; return True if it passes."""
- self.total += 1
- if self.group_ignore:
- return False
- optional = self.group_optional
- self.group_optional = False
- if self.group_skip:
- self.note_skip(name)
- return False
- c_file = os.path.join(self.temp_dir, 'test.c')
- o_file = os.path.join(self.temp_dir, 'test.o')
- with open(c_file, 'w') as c_file_out:
- c_file_out.write('#include <%s>\n%s' % (self.header, text))
- cmd = ('%s %s -c %s -o %s' % (self.cc, self.cflags, c_file, o_file))
- try:
- subprocess.check_call(cmd, shell=True)
- except subprocess.CalledProcessError:
- if optional:
- print('MISSING: %s' % name)
- sys.stdout.flush()
- self.group_ignore = True
- else:
- self.note_error(name, self.group_xfail)
- self.group_skip = True
- return False
- print('PASS: %s' % name)
- sys.stdout.flush()
- return True
- def execute_test(self, name, text):
- """Run an execution test."""
- self.total += 1
- if self.group_ignore:
- return False
- if self.group_skip:
- self.note_skip(name)
- return
- c_file = os.path.join(self.temp_dir, 'test.c')
- exe_file = os.path.join(self.temp_dir, 'test')
- with open(c_file, 'w') as c_file_out:
- c_file_out.write('#include <%s>\n%s' % (self.header, text))
- cmd = ('%s %s %s -o %s' % (self.cc, self.cflags, c_file, exe_file))
- try:
- subprocess.check_call(cmd, shell=True)
- except subprocess.CalledProcessError:
- self.note_error(name, self.group_xfail)
- return
- if self.cross:
- self.note_skip(name)
- return
- try:
- subprocess.check_call(exe_file, shell=True)
- except subprocess.CalledProcessError:
- self.note_error(name, self.group_xfail)
- return
- print('PASS: %s' % name)
- sys.stdout.flush()
- def check_namespace(self, name):
- """Check the namespace of a header."""
- c_file = os.path.join(self.temp_dir, 'namespace.c')
- out_file = os.path.join(self.temp_dir, 'namespace-out')
- with open(c_file, 'w') as c_file_out:
- c_file_out.write('#include <%s>\n' % self.header)
- cmd = ('%s %s -E %s -P -Wp,-dN > %s'
- % (self.cc, self.cflags_namespace, c_file, out_file))
- subprocess.check_call(cmd, shell=True)
- bad_tokens = set()
- with open(out_file, 'r') as content:
- for line in content:
- line = line.strip()
- if not line:
- continue
- if re.match(r'# [1-9]', line):
- continue
- match = re.match(r'#define (.*)', line)
- if match:
- self.check_token(bad_tokens, match.group(1))
- continue
- match = re.match(r'#undef (.*)', line)
- if match:
- bad_tokens.discard(match.group(1))
- continue
- # Tokenize the line and check identifiers found. The
- # handling of strings does not allow for escaped
- # quotes, no allowance is made for character
- # constants, and hex floats may be wrongly split into
- # tokens including identifiers, but this is sufficient
- # in practice and matches the old perl script.
- line = re.sub(r'"[^"]*"', '', line)
- line = line.strip()
- for token in re.split(r'[^A-Za-z0-9_]+', line):
- if re.match(r'[A-Za-z_]', token):
- self.check_token(bad_tokens, token)
- if bad_tokens:
- for token in sorted(bad_tokens):
- print(' Namespace violation: "%s"' % token)
- self.note_error(name, False)
- else:
- print('PASS: %s' % name)
- sys.stdout.flush()
- def run(self):
- """Load and run tests of a header."""
- with tempfile.TemporaryDirectory() as self.temp_dir:
- self.load_tests(self.header, False)
- self.group_optional = False
- self.group_xfail = False
- self.group_ignore = False
- self.group_skip = False
- available = self.compile_test('Availability of <%s>' % self.header,
- '')
- if available:
- # As an optimization, try running all non-optional,
- # non-XFAILed compilation tests in a single execution
- # of the compiler.
- combined_list = []
- for test in self.tests:
- if not test.optional and not test.xfail:
- for subtest in test.subtests:
- if isinstance(subtest, CompileSubTest):
- combined_list.append(subtest.text)
- subtest.run_early = True
- combined_ok = self.compile_test('Combined <%s> test'
- % self.header,
- '\n'.join(combined_list))
- # Now run the other tests, or all tests if the
- # combined test failed.
- for test in self.tests:
- # A test may run more than one subtest. If the
- # initial subtest for an optional symbol fails,
- # others are not run at all; if the initial
- # subtest for an optional symbol succeeds, others
- # are run and are not considered optional; if the
- # initial subtest for a required symbol fails,
- # others are skipped.
- self.group_optional = test.optional
- self.group_xfail = test.xfail
- self.group_ignore = False
- self.group_skip = False
- for subtest in test.subtests:
- if combined_ok and subtest.run_early:
- self.total += 1
- print('PASSCOMBINED: %s' % subtest.name)
- sys.stdout.flush()
- else:
- subtest.run(self)
- namespace_name = 'Namespace of <%s>' % self.header
- if available:
- self.check_namespace(namespace_name)
- else:
- self.note_skip(namespace_name)
- print('-' * 76)
- print(' Total number of tests : %4d' % self.total)
- print(' Number of failed tests : %4d' % self.errors)
- print(' Number of xfailed tests : %4d' % self.xerrors)
- print(' Number of skipped tests : %4d' % self.skipped)
- sys.exit(1 if self.errors else 0)
- def main():
- """The main entry point."""
- parser = argparse.ArgumentParser(description='Check header contents.')
- parser.add_argument('--header', metavar='HEADER',
- help='name of header')
- parser.add_argument('--standard', metavar='STD',
- help='standard to use when processing header')
- parser.add_argument('--cc', metavar='CC',
- help='C compiler to use')
- parser.add_argument('--flags', metavar='CFLAGS',
- help='Compiler flags to use with CC')
- parser.add_argument('--cross', action='store_true',
- help='Do not run compiled test programs')
- parser.add_argument('--xfail', metavar='COND',
- help='Name of condition for XFAILs')
- args = parser.parse_args()
- tests = HeaderTests(args.header, args.standard, args.cc, args.flags,
- args.cross, args.xfail)
- tests.run()
- if __name__ == '__main__':
- main()
|