Home | History | Annotate | Download | only in libcxx
      1 #!/usr/bin/env python
      2 #
      3 # Copyright (C) 2015 The Android Open Source Project
      4 #
      5 # Licensed under the Apache License, Version 2.0 (the "License");
      6 # you may not use this file except in compliance with the License.
      7 # You may obtain a copy of the License at
      8 #
      9 #      http://www.apache.org/licenses/LICENSE-2.0
     10 #
     11 # Unless required by applicable law or agreed to in writing, software
     12 # distributed under the License is distributed on an "AS IS" BASIS,
     13 # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
     14 # See the License for the specific language governing permissions and
     15 # limitations under the License.
     16 #
     17 """Runs the libc++ tests against the platform libc++."""
     18 from __future__ import print_function
     19 
     20 import argparse
     21 import logging
     22 import os
     23 import sys
     24 
     25 
     26 THIS_DIR = os.path.dirname(os.path.realpath(__file__))
     27 ANDROID_DIR = os.path.realpath(os.path.join(THIS_DIR, '../..'))
     28 
     29 
     30 def logger():
     31     """Returns the logger for the module."""
     32     return logging.getLogger(__name__)
     33 
     34 
     35 def call(cmd, *args, **kwargs):
     36     """subprocess.call with logging."""
     37     import subprocess
     38     logger().info('call %s', ' '.join(cmd))
     39     return subprocess.call(cmd, *args, **kwargs)
     40 
     41 
     42 def check_call(cmd, *args, **kwargs):
     43     """subprocess.check_call with logging."""
     44     import subprocess
     45     logger().info('check_call %s', ' '.join(cmd))
     46     return subprocess.check_call(cmd, *args, **kwargs)
     47 
     48 
     49 class ArgParser(argparse.ArgumentParser):
     50     """Parses command line arguments."""
     51     def __init__(self):
     52         super(ArgParser, self).__init__()
     53         self.add_argument(
     54             '--compiler', choices=('clang', 'gcc'), default='clang')
     55         self.add_argument(
     56             '--bitness', choices=(32, 64), type=int, default=32)
     57         self.add_argument('--host', action='store_true')
     58 
     59 
     60 def gen_test_config(bitness, compiler, host):
     61     """Generates the test configuration makefile for buildcmds."""
     62     testconfig_mk_path = os.path.join(THIS_DIR, 'buildcmds/testconfig.mk')
     63     with open(testconfig_mk_path, 'w') as test_config:
     64         if compiler == 'clang':
     65             print('LOCAL_CLANG := true', file=test_config)
     66         elif compiler == 'gcc':
     67             print('LOCAL_CLANG := false', file=test_config)
     68 
     69         if bitness == 32:
     70             print('LOCAL_MULTILIB := 32', file=test_config)
     71         elif bitness == 64:
     72             print('LOCAL_MULTILIB := 64', file=test_config)
     73 
     74         if compiler == 'clang':
     75             print('LOCAL_CXX := $(LOCAL_PATH)/buildcmdscc $(CLANG_CXX)',
     76                   file=test_config)
     77         else:
     78             if host:
     79                 prefix = 'HOST_'
     80             else:
     81                 prefix = 'TARGET_'
     82             print('LOCAL_CXX := $(LOCAL_PATH)/buildcmdscc '
     83                   '$($(LOCAL_2ND_ARCH_VAR_PREFIX){}CXX)'.format(prefix),
     84                   file=test_config)
     85 
     86         if host:
     87             print('include $(BUILD_HOST_EXECUTABLE)', file=test_config)
     88         else:
     89             print('include $(BUILD_EXECUTABLE)', file=test_config)
     90 
     91 
     92 def mmm(path):
     93     """Invokes the Android build command mmm."""
     94     makefile = os.path.join(path, 'Android.mk')
     95     main_mk = 'build/core/main.mk'
     96 
     97     env = dict(os.environ)
     98     env['ONE_SHOT_MAKEFILE'] = makefile
     99     env['LIBCXX_TESTING'] = 'true'
    100     cmd = [
    101         'make', '-j', '-C', ANDROID_DIR, '-f', main_mk,
    102         'MODULES-IN-' + path.replace('/', '-'),
    103     ]
    104     check_call(cmd, env=env)
    105 
    106 
    107 def gen_build_cmds(bitness, compiler, host):
    108     """Generates the build commands file for the test runner."""
    109     gen_test_config(bitness, compiler, host)
    110     mmm('external/libcxx/buildcmds')
    111 
    112 
    113 def main():
    114     """Program entry point."""
    115     logging.basicConfig(level=logging.INFO)
    116 
    117     args, lit_args = ArgParser().parse_known_args()
    118     lit_path = os.path.join(ANDROID_DIR, 'external/llvm/utils/lit/lit.py')
    119     gen_build_cmds(args.bitness, args.compiler, args.host)
    120 
    121     mode_str = 'host' if args.host else 'device'
    122     android_mode_arg = '--param=android_mode=' + mode_str
    123     site_cfg_path = os.path.join(THIS_DIR, 'test/lit.site.cfg')
    124     site_cfg_arg = '--param=libcxx_site_config=' + site_cfg_path
    125     default_test_path = os.path.join(THIS_DIR, 'test')
    126 
    127     have_filter_args = False
    128     for arg in lit_args:
    129         # If the argument is a valid path with default_test_path, it is a test
    130         # filter.
    131         real_path = os.path.realpath(arg)
    132         if not real_path.startswith(default_test_path):
    133             continue
    134         if not os.path.exists(real_path):
    135             continue
    136 
    137         have_filter_args = True
    138         break  # No need to keep scanning.
    139 
    140     lit_args = ['-sv', android_mode_arg, site_cfg_arg] + lit_args
    141     cmd = ['python', lit_path] + lit_args
    142     if not have_filter_args:
    143         cmd.append(default_test_path)
    144     sys.exit(call(cmd))
    145 
    146 
    147 if __name__ == '__main__':
    148     main()
    149