| 1 | import os
 | 
|---|
| 2 | import subprocess
 | 
|---|
| 3 | import sys
 | 
|---|
| 4 | from . import tools
 | 
|---|
| 5 | 
 | 
|---|
| 6 | global original_path
 | 
|---|
| 7 | 
 | 
|---|
| 8 | try :
 | 
|---|
| 9 |         original_path = os.getcwd()
 | 
|---|
| 10 |         testpath = os.path.dirname(os.path.abspath(os.path.join(os.getcwd(), sys.argv[0])))
 | 
|---|
| 11 |         sys.path.append(testpath)
 | 
|---|
| 12 |         import config
 | 
|---|
| 13 | 
 | 
|---|
| 14 |         SRCDIR = os.path.abspath(config.SRCDIR)
 | 
|---|
| 15 |         BUILDDIR = os.path.abspath(config.BUILDDIR)
 | 
|---|
| 16 |         distribute = config.DISTRIBUTE
 | 
|---|
| 17 |         os.chdir(testpath)
 | 
|---|
| 18 | 
 | 
|---|
| 19 | except:
 | 
|---|
| 20 |         print('ERROR: missing config.py, re-run configure script.', file=sys.stderr)
 | 
|---|
| 21 |         sys.exit(1)
 | 
|---|
| 22 | 
 | 
|---|
| 23 | class Architecture:
 | 
|---|
| 24 |         KnownArchitectures = {
 | 
|---|
| 25 |                 'x64'         : 'x64',
 | 
|---|
| 26 |                 'x86-64'      : 'x64',
 | 
|---|
| 27 |                 'x86_64'      : 'x64',
 | 
|---|
| 28 |                 'x86'         : 'x86',
 | 
|---|
| 29 |                 'aarch64'     : 'arm64',
 | 
|---|
| 30 |                 'arm64'       : 'arm64',
 | 
|---|
| 31 |                 'ARM64'       : 'arm64',
 | 
|---|
| 32 |                 'i386'        : 'x86',
 | 
|---|
| 33 |                 'i486'        : 'x86',
 | 
|---|
| 34 |                 'i686'        : 'x86',
 | 
|---|
| 35 |                 'Intel 80386' : 'x86',
 | 
|---|
| 36 |                 'arm'         : 'arm32',
 | 
|---|
| 37 |                 'ARM'         : 'arm32',
 | 
|---|
| 38 |                 'arm32'       : 'arm32',
 | 
|---|
| 39 |                 'ARM32'       : 'arm32',
 | 
|---|
| 40 |         }
 | 
|---|
| 41 | 
 | 
|---|
| 42 |         CrossCompileFlags = {
 | 
|---|
| 43 |                 'x64'  : 'ARCH_FLAGS=-m64',
 | 
|---|
| 44 |                 'x86'  : 'ARCH_FLAGS=-m32',
 | 
|---|
| 45 |                 'arm64': 'ARCH_FLAGS=',
 | 
|---|
| 46 |                 'arm32': 'ARCH_FLAGS=',
 | 
|---|
| 47 |         }
 | 
|---|
| 48 | 
 | 
|---|
| 49 |         def __init__(self, arch):
 | 
|---|
| 50 |                 try:
 | 
|---|
| 51 |                         canonical_host = Architecture.make_canonical( config.HOSTARCH )
 | 
|---|
| 52 |                 except KeyError:
 | 
|---|
| 53 |                         print("Unknown host architecture %s" % config.HOSTARCH, file=sys.stderr)
 | 
|---|
| 54 |                         sys.exit(1)
 | 
|---|
| 55 | 
 | 
|---|
| 56 |                 if arch:
 | 
|---|
| 57 |                         try:
 | 
|---|
| 58 |                                 arch = Architecture.make_canonical( arch )
 | 
|---|
| 59 |                         except KeyError:
 | 
|---|
| 60 |                                 print("Unknown architecture %s" % arch, file=sys.stderr)
 | 
|---|
| 61 |                                 sys.exit(1)
 | 
|---|
| 62 | 
 | 
|---|
| 63 |                 if arch and arch != canonical_host:
 | 
|---|
| 64 |                         self.target = arch
 | 
|---|
| 65 |                         self.cross_compile = True
 | 
|---|
| 66 |                 else:
 | 
|---|
| 67 |                         self.target = canonical_host
 | 
|---|
| 68 |                         self.cross_compile = False
 | 
|---|
| 69 | 
 | 
|---|
| 70 | 
 | 
|---|
| 71 |                 try :
 | 
|---|
| 72 |                         self.flags = Architecture.CrossCompileFlags[self.target]
 | 
|---|
| 73 |                 except KeyError:
 | 
|---|
| 74 |                         print("Cross compilation not available for architecture %s" % self.target, file=sys.stderr)
 | 
|---|
| 75 |                         sys.exit(1)
 | 
|---|
| 76 | 
 | 
|---|
| 77 |                 self.string = self.target
 | 
|---|
| 78 | 
 | 
|---|
| 79 |         def update(self):
 | 
|---|
| 80 |                 if not self.cross_compile:
 | 
|---|
| 81 |                         self.target = machine_default()
 | 
|---|
| 82 |                         self.string = self.target
 | 
|---|
| 83 |                         print("updated to %s" % self.target)
 | 
|---|
| 84 | 
 | 
|---|
| 85 |         def filter(self, tests):
 | 
|---|
| 86 |                 return [test for test in tests if not test.arch or self.target == test.arch]
 | 
|---|
| 87 | 
 | 
|---|
| 88 |         @staticmethod
 | 
|---|
| 89 |         def make_canonical(arch):
 | 
|---|
| 90 |                 return Architecture.KnownArchitectures[arch]
 | 
|---|
| 91 | 
 | 
|---|
| 92 | 
 | 
|---|
| 93 | class Debug:
 | 
|---|
| 94 |         def __init__(self, value):
 | 
|---|
| 95 |                 self.string = "debug" if value else "no debug"
 | 
|---|
| 96 |                 self.flags  = """DEBUG_FLAGS=%s""" % ("-debug -O0" if value else "-nodebug -O2")
 | 
|---|
| 97 |                 self.path   = "debug" if value else "nodebug"
 | 
|---|
| 98 | 
 | 
|---|
| 99 | class Install:
 | 
|---|
| 100 |         def __init__(self, value):
 | 
|---|
| 101 |                 if value:
 | 
|---|
| 102 |                         distribute = False
 | 
|---|
| 103 | 
 | 
|---|
| 104 |                 self.string = "installed" if value else "in tree"
 | 
|---|
| 105 |                 self.flags  = """installed=%s""" % ("yes" if value else "no")
 | 
|---|
| 106 | 
 | 
|---|
| 107 | class Timeouts:
 | 
|---|
| 108 |         def __init__(self, ts, tg):
 | 
|---|
| 109 |                 self.single = Timeouts.check(ts)
 | 
|---|
| 110 |                 self.total  = Timeouts.check(tg)
 | 
|---|
| 111 | 
 | 
|---|
| 112 |         @staticmethod
 | 
|---|
| 113 |         def check(value):
 | 
|---|
| 114 |                 if value < 1:
 | 
|---|
| 115 |                         print("Timeouts must be at least 1 second", file=sys.stderr)
 | 
|---|
| 116 |                         sys.exit(1)
 | 
|---|
| 117 | 
 | 
|---|
| 118 |                 return value
 | 
|---|
| 119 | 
 | 
|---|
| 120 | def init( options ):
 | 
|---|
| 121 |         global all_arch
 | 
|---|
| 122 |         global all_debug
 | 
|---|
| 123 |         global all_install
 | 
|---|
| 124 |         global arch
 | 
|---|
| 125 |         global debug
 | 
|---|
| 126 |         global archive
 | 
|---|
| 127 |         global install
 | 
|---|
| 128 |         global invariant
 | 
|---|
| 129 | 
 | 
|---|
| 130 |         global continue_
 | 
|---|
| 131 |         global dry_run
 | 
|---|
| 132 |         global generating
 | 
|---|
| 133 |         global make
 | 
|---|
| 134 |         global make_jobfds
 | 
|---|
| 135 |         global output_width
 | 
|---|
| 136 |         global timeout
 | 
|---|
| 137 |         global timeout2gdb
 | 
|---|
| 138 | 
 | 
|---|
| 139 |         all_arch     = [Architecture(o) for o in list(dict.fromkeys(options.arch   ))] if options.arch else [Architecture(None)]
 | 
|---|
| 140 |         all_debug    = [Debug(o)        for o in list(dict.fromkeys(options.debug  ))]
 | 
|---|
| 141 |         all_install  = [Install(o)      for o in list(dict.fromkeys(options.install))]
 | 
|---|
| 142 |         archive      = os.path.abspath(os.path.join(original_path, options.archive_errors)) if options.archive_errors else None
 | 
|---|
| 143 |         invariant    = options.invariant
 | 
|---|
| 144 |         continue_    = options.continue_
 | 
|---|
| 145 |         dry_run      = options.dry_run # must be called before tools.config_hash()
 | 
|---|
| 146 |         generating   = options.regenerate_expected
 | 
|---|
| 147 |         make         = ['make']
 | 
|---|
| 148 |         make_jobfds  = []
 | 
|---|
| 149 |         output_width = 24
 | 
|---|
| 150 |         timeout      = Timeouts(options.timeout, options.global_timeout)
 | 
|---|
| 151 |         timeout2gdb  = options.timeout_with_gdb
 | 
|---|
| 152 | 
 | 
|---|
| 153 |         # if we distribute, distcc errors will fail tests, use log file for distcc
 | 
|---|
| 154 |         # don't use "'DISTCC_LOG' not in os.environ" because it can be set to ''
 | 
|---|
| 155 |         if distribute and not os.environ.get('DISTCC_LOG'):
 | 
|---|
| 156 |                 os.putenv('DISTCC_LOG', os.path.join(BUILDDIR, 'distcc_error.log'))
 | 
|---|
| 157 | 
 | 
|---|
| 158 | def update_make_cmd(flags):
 | 
|---|
| 159 |         global make
 | 
|---|
| 160 |         make = ['make', *flags]
 | 
|---|
| 161 | 
 | 
|---|
| 162 | def update_make_fds(r, w):
 | 
|---|
| 163 |         global make_jobfds
 | 
|---|
| 164 |         make_jobfds = (r, w)
 | 
|---|
| 165 | 
 | 
|---|
| 166 | def validate():
 | 
|---|
| 167 |         """Validate the current configuration and update globals"""
 | 
|---|
| 168 | 
 | 
|---|
| 169 |         global distcc
 | 
|---|
| 170 |         distcc       = "DISTCC_CFA_PATH=~/.cfadistcc/%s/cfa" % tools.config_hash()
 | 
|---|
| 171 |         make_ret, out, err = tools.make( ".validate", output_file=subprocess.PIPE, error=subprocess.PIPE )
 | 
|---|
| 172 |         if make_ret != 0:
 | 
|---|
| 173 |                 print("ERROR: Invalid configuration %s:%s" % (arch.string, debug.string), file=sys.stderr)
 | 
|---|
| 174 |                 print("       verify returned : \n%s" % err, file=sys.stderr)
 | 
|---|
| 175 |                 sys.exit(1)
 | 
|---|
| 176 | 
 | 
|---|
| 177 | def prep_output(tests):
 | 
|---|
| 178 |         global output_width
 | 
|---|
| 179 |         output_width = max(map(lambda t: len(t.target()), tests))
 | 
|---|
| 180 |         # 35 is the maximum width of the name field before we get line wrapping.
 | 
|---|
| 181 |         output_width = min(output_width, 35)
 | 
|---|