os/mm/mmtestenv/mmtesttools/Build/buildutils/mmbuild2.py
author sl@SLION-WIN7.fritz.box
Fri, 15 Jun 2012 03:10:57 +0200
changeset 0 bde4ae8d615e
permissions -rw-r--r--
First public contribution.
sl@0
     1
sl@0
     2
# Copyright (c) 2008-2009 Nokia Corporation and/or its subsidiary(-ies).
sl@0
     3
# All rights reserved.
sl@0
     4
# This component and the accompanying materials are made available
sl@0
     5
# under the terms of "Eclipse Public License v1.0"
sl@0
     6
# which accompanies this distribution, and is available
sl@0
     7
# at the URL "http://www.eclipse.org/legal/epl-v10.html".
sl@0
     8
#
sl@0
     9
# Initial Contributors:
sl@0
    10
# Nokia Corporation - initial contribution.
sl@0
    11
#
sl@0
    12
# Contributors:
sl@0
    13
#
sl@0
    14
# Description:
sl@0
    15
# mmbuild2 - python script intended to replace mmbuild.pl
sl@0
    16
#
sl@0
    17
sl@0
    18
from optparse import (make_option,OptionParser)
sl@0
    19
import sys
sl@0
    20
import os
sl@0
    21
import subprocess
sl@0
    22
import MbcUtils
sl@0
    23
sl@0
    24
# some literals. TODO - use explicit type?
sl@0
    25
_KAll = "all"
sl@0
    26
_KMmf = "mmf"
sl@0
    27
_KMisc = "misc"
sl@0
    28
_KIcl = "icl"
sl@0
    29
sl@0
    30
_KMainTempConfig = ".config.xml"
sl@0
    31
_KTestTempConfig = ".testconfig.xml"
sl@0
    32
_KTestBuildTempConfig = ".testbuildconfig.xml"
sl@0
    33
sl@0
    34
class Unimplemented(Exception):
sl@0
    35
    """Raised for invalid commands"""
sl@0
    36
sl@0
    37
    def __init__(self, msg):
sl@0
    38
        self.__msg = msg
sl@0
    39
sl@0
    40
    def __str__(self):
sl@0
    41
        return self.__msg
sl@0
    42
sl@0
    43
class Script(object):
sl@0
    44
sl@0
    45
sl@0
    46
    # TODO this option_list should be replaced by add_option calls to parameter
sl@0
    47
    # option_list parameter to OptionParser is down as deprecated
sl@0
    48
    option_table = [
sl@0
    49
        make_option ("-t", "--test",
sl@0
    50
                     action="store_true", dest="buildTest", default=False,
sl@0
    51
                     help="""build test code rather than main code (without -b, that built by "abld build" or equiv)"""),
sl@0
    52
        make_option ("-b", "--testBuild",
sl@0
    53
                     action="store_true", dest="buildTestBuild", default=False,
sl@0
    54
                     help="""when given with -t, build that code that requires "abld test build" """),
sl@0
    55
         make_option ("-x", "--additional", "--opt", "--extra",
sl@0
    56
                     action="store_true", dest="extra", default=False,
sl@0
    57
                     help="build additional modules (from Xopt.mbc files)"),
sl@0
    58
        make_option ("-i", "--icl",
sl@0
    59
                     action="store_const", const=_KIcl, dest="toBuild", default=_KAll,
sl@0
    60
                     help="build icl modules only"),
sl@0
    61
        make_option ("-m", "--mmf",
sl@0
    62
                     action="store_const", const=_KMmf, dest="toBuild", default=_KAll,
sl@0
    63
                     help="build mmf modules only"),
sl@0
    64
        make_option ("-c", "--misc",
sl@0
    65
                     action="store_const", const=_KMisc, dest="toBuild", default=_KAll,
sl@0
    66
                     help="build misc modules only"),
sl@0
    67
        make_option ("-f", "--full", "--all",
sl@0
    68
                     action="store_const", const=_KAll, dest="toBuild", default=_KAll,
sl@0
    69
                     help="build all modules (which depends on if -x is given too)"),
sl@0
    70
        make_option ("-k", "--keepgoing",
sl@0
    71
                     action="store_true", dest="keepgoing", default=False,
sl@0
    72
                     help="keep going if errors are found"),
sl@0
    73
        make_option ("-g", "--gccxml",
sl@0
    74
                     action="store_true", dest="gccxml", default=False,
sl@0
    75
                     help="build for gccxml"),
sl@0
    76
        make_option ("-s", "--skipmake",
sl@0
    77
                     action="store_true", dest="skipmake", default=False,
sl@0
    78
                     help="skip building makefile (ignored)"),
sl@0
    79
        make_option ("--iclTestdata",
sl@0
    80
                     action="store_true", dest="icl_tests", default=False,
sl@0
    81
                     help="build ICL Tests"),
sl@0
    82
        ]
sl@0
    83
sl@0
    84
    description = \
sl@0
    85
"""Script for selection appropriate set of multimedia
sl@0
    86
components and build them together, so build order does not matter.
sl@0
    87
One, and only one of "setup", "build" or "clean" must be given.
sl@0
    88
"""
sl@0
    89
sl@0
    90
    def __init__(self):
sl@0
    91
        # need to implement commands as property and not class constant or
sl@0
    92
        # we get a forward declaration problem
sl@0
    93
        self.iCommands = {"setup": self.doSetup, "build": self.doBuild, "clean": self.doClean}
sl@0
    94
        self.iOptions = None
sl@0
    95
        self.iProdCodeList = []
sl@0
    96
        self.iTestCodeList = []
sl@0
    97
        self.iTestBuildCodeList = []
sl@0
    98
        self.iOptionParser = None
sl@0
    99
        self.iDebug = False
sl@0
   100
sl@0
   101
    def main(self):
sl@0
   102
        "main function of script"
sl@0
   103
        self.iOptionParser = OptionParser(
sl@0
   104
            option_list = Script.option_table,
sl@0
   105
            usage="%prog [-t [-b]] [-i|-m|-c|-f] (setup|build [winscw]|clean)",
sl@0
   106
            description = Script.description
sl@0
   107
            )
sl@0
   108
        (self.iOptions, args) = self.iOptionParser.parse_args()
sl@0
   109
sl@0
   110
        self.DebugPrint (str(self.iOptions))
sl@0
   111
        self.DebugPrint (str(args))
sl@0
   112
sl@0
   113
        if (len(args)==0 or not args[0] in self.iCommands or
sl@0
   114
                not self.CheckOptions()):
sl@0
   115
            self.iOptionParser.print_usage()
sl@0
   116
            return -1
sl@0
   117
sl@0
   118
        remainingArgs = args[1:len(args)]
sl@0
   119
sl@0
   120
        return self.iCommands[args[0]](remainingArgs) # effective switch to doBuild, doSetup or doClean
sl@0
   121
sl@0
   122
    def CheckOptions(self):
sl@0
   123
        "Check for any invalid option combinations. Warn about ignored ones etc"
sl@0
   124
        if self.iOptions.skipmake:
sl@0
   125
            print ("Warning ignoring -s - option is no longer valid in raptor version")
sl@0
   126
        if not self.iOptions.buildTest and self.iOptions.buildTestBuild:
sl@0
   127
            return False # can't do -b without -t
sl@0
   128
        return True # all other combinations OK
sl@0
   129
sl@0
   130
    def doBuild(self, remainingArgs):
sl@0
   131
        # if we have a remainingArg, only "winscw" is allowed
sl@0
   132
        if not (len(remainingArgs)==0 or len(remainingArgs)==1 and remainingArgs[0]=="winscw"):
sl@0
   133
            self.iOptionParser.print_usage()
sl@0
   134
            return -1
sl@0
   135
sl@0
   136
        # for normal build need to run "sbs -s .config.xml build"
sl@0
   137
        # for test build need to run "sbs -s .testconfig.xml build"
sl@0
   138
        # use --logfile=- to send output to stdout instead of log file
sl@0
   139
        configFile = _KMainTempConfig
sl@0
   140
        testBuild = False
sl@0
   141
        if self.iOptions.buildTest:
sl@0
   142
            if not self.iOptions.buildTestBuild:
sl@0
   143
                # build test config instead when -t etc given
sl@0
   144
                configFile = _KTestTempConfig
sl@0
   145
            else:
sl@0
   146
                # build test config instead when -t -b etc given
sl@0
   147
                configFile = _KTestBuildTempConfig
sl@0
   148
                testBuild = True
sl@0
   149
        sbs_command = self.sbsCommand()
sl@0
   150
        commands = [sbs_command, "-s", configFile]
sl@0
   151
        commands += ["--logfile=-"] # send output to stdout
sl@0
   152
        if self.iOptions.keepgoing:
sl@0
   153
            commands += ["--keepgoing"]
sl@0
   154
        if testBuild:
sl@0
   155
            commands += ["--config=winscw.test"]
sl@0
   156
        else:
sl@0
   157
            commands += ["--config=winscw"]
sl@0
   158
        if not(len(remainingArgs)>0 and remainingArgs[0]=="winscw"):
sl@0
   159
            # not the winscw scenario - we want both winscw and armv5
sl@0
   160
            if testBuild:
sl@0
   161
                commands += ["--config=armv5.test"]
sl@0
   162
            else:
sl@0
   163
                commands += ["--config=armv5"]
sl@0
   164
        commands += ["--filters=FilterSquashLog"] # reduce log size
sl@0
   165
        commands += ["--tries=2"] # retry on failure - e.g. for license fails
sl@0
   166
        commands += ["build"]
sl@0
   167
        self.DebugPrint("""command="%s" """ % str(commands))
sl@0
   168
        print "------------------ sbs start : %s" % str(commands)
sl@0
   169
        sys.stdout.flush() # flush any output here so appears correct in log
sl@0
   170
        subprocess.check_call(commands) # raises exception on error
sl@0
   171
        print "------------------ sbs end"
sl@0
   172
        return 0
sl@0
   173
sl@0
   174
    def sbsCommand(self):
sl@0
   175
        "sbs command - that can be used by subprocess"
sl@0
   176
        # For some reason, have to work out batch file to run by longhand
sl@0
   177
        # rather than just saying "sbs" and letting command work it out.
sl@0
   178
        # Could use sys.command() instead, but that is deprecated
sl@0
   179
        sbs_home = os.getenv("SBS_HOME")
sl@0
   180
        assert sbs_home, "SBS_HOME must be defined to use this script"
sl@0
   181
        sbs_command = os.path.join(sbs_home, "bin", "sbs.bat")
sl@0
   182
        return sbs_command
sl@0
   183
sl@0
   184
    def doSetup(self, remainingArgs):
sl@0
   185
        if len(remainingArgs)!=0:
sl@0
   186
            self.iOptionParser.print_usage()
sl@0
   187
            return -1
sl@0
   188
sl@0
   189
        self.buildMbcLists()
sl@0
   190
sl@0
   191
        self.DebugPrint ("prodCodeList=%s" % str(self.iProdCodeList))
sl@0
   192
        self.DebugPrint ("testCodeList=%s" % str(self.iTestCodeList))
sl@0
   193
        self.DebugPrint ("testBuildCodeList=%s" % str(self.iTestBuildCodeList))
sl@0
   194
sl@0
   195
        mbcParser = MbcUtils.MbcParser(self.iProdCodeList)
sl@0
   196
        folders = mbcParser()
sl@0
   197
        self.DebugPrint ("folders=%s" % str (folders))
sl@0
   198
        getFolderList = MbcUtils.GetFolderList(folders)
sl@0
   199
        groupFolders = getFolderList()
sl@0
   200
        self.DebugPrint ("prodCodeFolderList=%s" % str(groupFolders))
sl@0
   201
sl@0
   202
        generator = MbcUtils.ConfigFileGenerator(groupFolders, _KMainTempConfig)
sl@0
   203
        generator.write()
sl@0
   204
sl@0
   205
        mbcParser = MbcUtils.MbcParser(self.iTestCodeList)
sl@0
   206
        folders = mbcParser()
sl@0
   207
        self.DebugPrint ("testfolders=%s" % str (folders))
sl@0
   208
        getFolderList = MbcUtils.GetFolderList(folders)
sl@0
   209
        groupFolders = getFolderList()
sl@0
   210
        self.DebugPrint ("testCodeFolderList=%s" % str(groupFolders))
sl@0
   211
sl@0
   212
        generator = MbcUtils.ConfigFileGenerator(groupFolders, _KTestTempConfig)
sl@0
   213
        generator.write()
sl@0
   214
sl@0
   215
        mbcParser = MbcUtils.MbcParser(self.iTestBuildCodeList)
sl@0
   216
        folders = mbcParser()
sl@0
   217
        self.DebugPrint ("testBuildfolders=%s" % str (folders))
sl@0
   218
        getFolderList = MbcUtils.GetFolderList(folders)
sl@0
   219
        groupFolders = getFolderList()
sl@0
   220
        self.DebugPrint ("testBuildCodeFolderList=%s" % str(groupFolders))
sl@0
   221
sl@0
   222
        generator = MbcUtils.ConfigFileGenerator(groupFolders, _KTestBuildTempConfig)
sl@0
   223
        generator.write()
sl@0
   224
        return 0
sl@0
   225
sl@0
   226
    def buildMbcLists(self):
sl@0
   227
        # some boolean values
sl@0
   228
        want_mmf = self.iOptions.toBuild in [_KAll, _KMmf];
sl@0
   229
        want_icl = self.iOptions.toBuild in [_KAll, _KIcl];
sl@0
   230
        want_misc = self.iOptions.toBuild in [_KAll, _KMisc];
sl@0
   231
        want_extra = self.iOptions.extra
sl@0
   232
        want_icl_tests = self.iOptions.icl_tests
sl@0
   233
sl@0
   234
        # now build up the lists depending on which "component" we want
sl@0
   235
        # perhaps this should be reworked as a table, but not obvious as to how
sl@0
   236
        self.iTestCodeList += ["AllTests.mbc", "TestFrameworkTest.mbc",
sl@0
   237
                               "targettools.mbc"]
sl@0
   238
        if want_mmf:
sl@0
   239
            self.iProdCodeList += ["mmf.mbc"]
sl@0
   240
            self.iTestCodeList += ["mmfTest.mbc"]
sl@0
   241
            self.iTestBuildCodeList += ["mmfTestBuild.mbc"]
sl@0
   242
            # assume mmfPhys and mmfOptPhys can currently be ignored
sl@0
   243
            # they should generate another list and be built as special cases
sl@0
   244
            # self.iProdCodeList += ["mmfPhys.mbc"]
sl@0
   245
            if want_extra:
sl@0
   246
                self.iProdCodeList += ["mmfOpt.mbc"]
sl@0
   247
                self.iTestCodeList += ["mmfOptTest.mbc"]
sl@0
   248
                self.iTestBuildCodeList += ["mmfOptTestBuild.mbc"]
sl@0
   249
                # self.iProdCodeList += ["mmfOptPhys.mbc"]
sl@0
   250
            else:
sl@0
   251
                self.iTestCodeList += ["mmfNotOptTest.mbc"]
sl@0
   252
        if want_icl:
sl@0
   253
            self.iProdCodeList += ["icl.mbc"]
sl@0
   254
            if want_icl_tests:
sl@0
   255
            	self.iTestCodeList += ["iclTest.mbc"]
sl@0
   256
            	self.iTestBuildCodeList += ["iclTestBuild.mbc"]
sl@0
   257
            	if want_extra:
sl@0
   258
            		self.iTestCodeList += ["iclOptTest.mbc"]
sl@0
   259
            		self.iTestBuildCodeList += ["iclOptTestBuild.mbc"]
sl@0
   260
            if want_extra:
sl@0
   261
                self.iProdCodeList += ["iclOpt.mbc"]   
sl@0
   262
        if want_misc:
sl@0
   263
            self.iProdCodeList += ["misc.mbc"]
sl@0
   264
            self.iTestCodeList += ["miscTest.mbc"]
sl@0
   265
            self.iTestBuildCodeList += ["miscTestBuild.mbc"]
sl@0
   266
            if want_extra:
sl@0
   267
                self.iProdCodeList += ["miscOpt.mbc"]
sl@0
   268
                self.iTestCodeList += ["miscOptTest.mbc"]
sl@0
   269
                self.iTestBuildCodeList += ["miscOptTestBuild.mbc"]
sl@0
   270
sl@0
   271
    def doClean(self, remainingArgs):
sl@0
   272
        "clean is called. Note -t clean means clean test code too, not instead"
sl@0
   273
        # for normal clean need to run "sbs -s .config.xml clean"
sl@0
   274
        # for test clean need to run "sbs -s .testconfig.xml clean"
sl@0
   275
        # use --logfile=- to send output to stdout instead of log file
sl@0
   276
        whatToClean = [_KMainTempConfig]
sl@0
   277
        if self.iOptions.buildTest:
sl@0
   278
            whatToClean += [_KTestTempConfig]
sl@0
   279
        for configFile in whatToClean:
sl@0
   280
            sbs_command = self.sbsCommand()
sl@0
   281
            commands = [sbs_command, "-s", configFile]
sl@0
   282
            commands += ["--logfile=-"] # send output to stdout
sl@0
   283
            commands += ["reallyclean"]
sl@0
   284
            self.DebugPrint ("""command="%s" """ % str(commands))
sl@0
   285
            subprocess.check_call(commands) # raises exception on error
sl@0
   286
        return 0
sl@0
   287
sl@0
   288
    def DebugPrint(self, str):
sl@0
   289
        "print a string if self.iDebug is set - would be turned on manually"
sl@0
   290
        if self.iDebug:
sl@0
   291
            print(str)
sl@0
   292
sl@0
   293
if __name__ == "__main__":
sl@0
   294
    script = Script()
sl@0
   295
    sys.exit(script.main())