Index: test/clang-tidy/check_clang_tidy.cpp =================================================================== --- test/clang-tidy/check_clang_tidy.cpp +++ test/clang-tidy/check_clang_tidy.cpp @@ -1,21 +1,31 @@ // RUN: %check_clang_tidy -check-suffix=USING-A %s misc-unused-using-decls %t -- -- -DUSING_A // RUN: %check_clang_tidy -check-suffix=USING-B %s misc-unused-using-decls %t -- -- -DUSING_B +// RUN: %check_clang_tidy -check-suffix=USING-C,USING-D %s misc-unused-using-decls %t -- -- -DUSING_C_D // RUN: %check_clang_tidy %s misc-unused-using-decls %t -namespace a {class A {}; class B {}; class C {}; } +namespace a {class A {}; class B {}; class C {}; class D {}; class E {};} namespace b { #if defined(USING_A) using a::A; #elif defined(USING_B) using a::B; -#else +#elif defined(USING_C_D) using a::C; +using a::D; +#else +using a::E; #endif } namespace c {} -// CHECK-MESSAGES-USING-A: :[[@LINE-8]]:10: warning: using decl 'A' {{.*}} -// CHECK-MESSAGES-USING-B: :[[@LINE-7]]:10: warning: using decl 'B' {{.*}} -// CHECK-MESSAGES: :[[@LINE-6]]:10: warning: using decl 'C' {{.*}} +// CHECK-MESSAGES-USING-A: warning: using decl 'A' {{.*}} +// CHECK-MESSAGES-USING-B: warning: using decl 'B' {{.*}} +// CHECK-MESSAGES-USING-C: warning: using decl 'C' {{.*}} +// CHECK-MESSAGES-USING-D: warning: using decl 'D' {{.*}} +// CHECK-MESSAGES: warning: using decl 'E' {{.*}} // CHECK-FIXES-USING-A-NOT: using a::A;$ // CHECK-FIXES-USING-B-NOT: using a::B;$ -// CHECK-FIXES-NOT: using a::C;$ +// CHECK-FIXES-USING-C-NOT: using a::C;$ +// CHECK-FIXES-USING-C-NOT: using a::D;$ +// CHECK-FIXES-USING-D-NOT: using a::C;$ +// CHECK-FIXES-USING-D-NOT: using a::D;$ +// CHECK-FIXES-NOT: using a::E;$ Index: test/clang-tidy/check_clang_tidy.py =================================================================== --- test/clang-tidy/check_clang_tidy.py +++ test/clang-tidy/check_clang_tidy.py @@ -18,7 +18,7 @@ Usage: check_clang_tidy.py [-resource-dir=] \ [-assume-filename=] \ - [-check-suffix=] \ + [-check-suffix=] \ \ -- [optional clang-tidy arguments] @@ -38,15 +38,18 @@ f.write(text) f.truncate() +def csv(string): + return string.split(',') + def main(): parser = argparse.ArgumentParser() parser.add_argument('-expect-clang-tidy-error', action='store_true') parser.add_argument('-resource-dir') parser.add_argument('-assume-filename') - parser.add_argument('-check-suffix', default='') parser.add_argument('input_file_name') parser.add_argument('check_name') parser.add_argument('temp_file_name') + parser.add_argument('-check-suffix', default=[], type=csv, help="comma-separated list of FileCheck suffixes") args, extra_args = parser.parse_known_args() @@ -72,14 +75,6 @@ clang_tidy_extra_args.extend( ['-fobjc-abi-version=2', '-fobjc-arc']) - if args.check_suffix and not re.match('^[A-Z0-9\-]+$', args.check_suffix): - sys.exit('Only A..Z, 0..9 and "-" are allowed in check suffix, but "%s" was given' % (args.check_suffix)) - - file_check_suffix = ('-' + args.check_suffix) if args.check_suffix else '' - check_fixes_prefix = 'CHECK-FIXES' + file_check_suffix - check_messages_prefix = 'CHECK-MESSAGES' + file_check_suffix - check_notes_prefix = 'CHECK-NOTES' + file_check_suffix - # Tests should not rely on STL being available, and instead provide mock # implementations of relevant APIs. clang_tidy_extra_args.append('-nostdinc++') @@ -90,15 +85,33 @@ with open(input_file_name, 'r') as input_file: input_text = input_file.read() - has_check_fixes = check_fixes_prefix in input_text - has_check_messages = check_messages_prefix in input_text - has_check_notes = check_notes_prefix in input_text + check_fixes_prefixes = [] + check_messages_prefixes = [] + check_notes_prefixes = [] + + if any(args.check_suffix): + for check in args.check_suffix: + if not re.match('^[A-Z0-9\-]+$', check): + sys.exit('Only A..Z, 0..9 and "-" are ' + + 'allowed in check suffixes list, but "%s" was given' % (check)) + file_check_suffix = '-' + check + check_fixes_prefixes.append('CHECK-FIXES' + file_check_suffix) + check_messages_prefixes.append('CHECK-MESSAGES' + file_check_suffix) + check_notes_prefixes.append('CHECK-NOTES' + file_check_suffix) + else: + check_fixes_prefixes = ['CHECK-FIXES'] + check_messages_prefixes = ['CHECK-MESSAGES'] + check_notes_prefixes = ['CHECK-NOTES'] + + has_check_fixes = any(prefix in input_text for prefix in check_fixes_prefixes) + has_check_messages = any(prefix in input_text for prefix in check_messages_prefixes) + has_check_notes = any(prefix in input_text for prefix in check_notes_prefixes) if not has_check_fixes and not has_check_messages and not has_check_notes: - sys.exit('%s, %s or %s not found in the input' % (check_fixes_prefix, - check_messages_prefix, check_notes_prefix) ) + sys.exit('%s, %s or %s not found in the input' % + (check_fixes_prefixes, check_messages_prefixes, check_notes_prefixes)) - if has_check_notes and has_check_messages: + if any(set(check_notes_prefixes) & set(check_messages_prefixes)): sys.exit('Please use either CHECK-NOTES or CHECK-MESSAGES but not both') # Remove the contents of the CHECK lines to avoid CHECKs matching on @@ -143,7 +156,8 @@ try: subprocess.check_output( ['FileCheck', '-input-file=' + temp_file_name, input_file_name, - '-check-prefix=' + check_fixes_prefix, '-strict-whitespace'], + '-check-prefixes=' + ','.join(check_fixes_prefixes), + '-strict-whitespace'], stderr=subprocess.STDOUT) except subprocess.CalledProcessError as e: print('FileCheck failed:\n' + e.output.decode()) @@ -155,7 +169,7 @@ try: subprocess.check_output( ['FileCheck', '-input-file=' + messages_file, input_file_name, - '-check-prefix=' + check_messages_prefix, + '-check-prefixes=' + ','.join(check_messages_prefixes), '-implicit-check-not={{warning|error}}:'], stderr=subprocess.STDOUT) except subprocess.CalledProcessError as e: @@ -170,7 +184,7 @@ try: subprocess.check_output( ['FileCheck', '-input-file=' + notes_file, input_file_name, - '-check-prefix=' + check_notes_prefix, + '-check-prefixes=' + ','.join(check_notes_prefixes), '-implicit-check-not={{note|warning|error}}:'], stderr=subprocess.STDOUT) except subprocess.CalledProcessError as e: