Home | History | Annotate | Download | only in build
      1 # Copyright (C) 2009 Kevin Ollivier  All rights reserved.
      2 #
      3 # Redistribution and use in source and binary forms, with or without
      4 # modification, are permitted provided that the following conditions
      5 # are met:
      6 # 1. Redistributions of source code must retain the above copyright
      7 #    notice, this list of conditions and the following disclaimer.
      8 # 2. Redistributions in binary form must reproduce the above copyright
      9 #    notice, this list of conditions and the following disclaimer in the
     10 #    documentation and/or other materials provided with the distribution.
     11 #
     12 # THIS SOFTWARE IS PROVIDED BY APPLE COMPUTER, INC. ``AS IS'' AND ANY
     13 # EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
     14 # IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
     15 # PURPOSE ARE DISCLAIMED.  IN NO EVENT SHALL APPLE COMPUTER, INC. OR
     16 # CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
     17 # EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
     18 # PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR
     19 # PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY
     20 # OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
     21 # (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
     22 # OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. 
     23 #
     24 # Helper functions for the WebKit build.
     25 
     26 import commands
     27 import glob
     28 import os
     29 import platform
     30 import re
     31 import shutil
     32 import sys
     33 import urllib
     34 import urlparse
     35 
     36 def get_output(command):
     37     """
     38     Windows-compatible function for getting output from a command.
     39     """
     40     if sys.platform.startswith('win'):
     41         f = os.popen(command)
     42         return f.read().strip()
     43     else:
     44         return commands.getoutput(command)
     45     
     46 def get_excludes(root, patterns):
     47     """
     48     Get a list of exclude patterns going down several dirs. 
     49     TODO: Make this fully recursive.
     50     """
     51     excludes = []
     52     
     53     for pattern in patterns:
     54         subdir_pattern = os.sep + '*'
     55         for subdir in [subdir_pattern, subdir_pattern*2, subdir_pattern*3]:
     56             adir = root + subdir + os.sep + pattern
     57             files = glob.glob(adir)
     58             for afile in files:
     59                 excludes.append(os.path.basename(afile))
     60 
     61     return excludes
     62     
     63 def get_dirs_for_features(root, features, dirs):
     64     """
     65     Find which directories to include in the list of build dirs based upon the 
     66     enabled port(s) and features.
     67     """
     68     outdirs = dirs
     69     for adir in dirs:
     70         for feature in features:
     71             relpath = os.path.join(adir, feature)
     72             featuredir = os.path.join(root, relpath)
     73             if os.path.exists(featuredir) and not relpath in outdirs:
     74                 outdirs.append(relpath)
     75             
     76     return outdirs
     77 
     78 def download_if_newer(url, destdir):
     79     """
     80     Checks if the file on the server is newer than the one in the user's tree,
     81     and if so, downloads it. 
     82     
     83     Returns the filename of the downloaded file if downloaded, or None if 
     84     the existing file matches the one on the server.
     85     """
     86     obj = urlparse.urlparse(url)
     87     filename = os.path.basename(obj.path)
     88     destfile = os.path.join(destdir, filename)
     89     
     90     urlobj = urllib.urlopen(url)
     91     size = long(urlobj.info().getheader('Content-Length'))
     92 
     93     def download_callback(downloaded, block_size, total_size): 
     94         downloaded = block_size * downloaded
     95         if downloaded > total_size:
     96             downloaded = total_size
     97         sys.stdout.write('%s %d of %d bytes downloaded\r' % (filename, downloaded, total_size))
     98     
     99     # NB: We don't check modified time as Python doesn't yet handle timezone conversion
    100     # properly when converting strings to time objects.
    101     if not os.path.exists(destfile) or os.path.getsize(destfile) != size:
    102         urllib.urlretrieve(url, destfile, download_callback)
    103         print ''
    104         return destfile
    105     
    106     return None
    107     
    108 def update_wx_deps(conf, wk_root, msvc_version='msvc2008'):
    109     """
    110     Download and update tools needed to build the wx port.
    111     """
    112     import Logs
    113     Logs.info('Ensuring wxWebKit dependencies are up-to-date.')
    114     
    115     wklibs_dir = os.path.join(wk_root, 'WebKitLibraries')
    116     waf = download_if_newer('http://wxwebkit.wxcommunity.com/downloads/deps/waf', os.path.join(wk_root, 'Tools', 'wx'))
    117     if waf:
    118         # TODO: Make the build restart itself after an update.
    119         Logs.warn('Build system updated, please restart build.')
    120         sys.exit(1)
    121 
    122     # since this module is still experimental
    123     wxpy_dir = os.path.join(wk_root, 'Source', 'WebKit', 'wx', 'bindings', 'python')
    124     swig_module = download_if_newer('http://wxwebkit.wxcommunity.com/downloads/deps/swig.py.txt', wxpy_dir)
    125     if swig_module:
    126         shutil.copy(os.path.join(wxpy_dir, 'swig.py.txt'), os.path.join(wxpy_dir, 'swig.py'))
    127 
    128     if sys.platform.startswith('win'):
    129         Logs.info('downloading deps package')
    130         archive = download_if_newer('http://wxwebkit.wxcommunity.com/downloads/deps/wxWebKitDeps-%s.zip' % msvc_version, wklibs_dir)
    131         if archive and os.path.exists(archive):
    132             os.system('unzip -o %s -d %s' % (archive, os.path.join(wklibs_dir, msvc_version)))
    133     
    134     elif sys.platform.startswith('darwin'):
    135         # export the right compiler for building the dependencies
    136         if platform.release().startswith('10'): # Snow Leopard
    137             os.environ['CC'] = conf.env['CC'][0]
    138             os.environ['CXX'] = conf.env['CXX'][0]
    139         os.system('%s/Tools/wx/install-unix-extras' % wk_root)
    140         
    141 def includeDirsForSources(sources):
    142     include_dirs = []
    143     for group in sources:
    144         for source in group:
    145             dirname = os.path.dirname(source)
    146             if not dirname in include_dirs:
    147                 include_dirs.append(dirname)
    148             
    149     return include_dirs
    150 
    151 def flattenSources(sources):
    152     flat_sources = []
    153     for group in sources:
    154         flat_sources.extend(group)
    155         
    156     return flat_sources
    157 
    158 def git_branch_name():
    159     try:
    160         branches = commands.getoutput("git branch --no-color")
    161         match = re.search('^\* (.*)', branches, re.MULTILINE)
    162         if match:
    163             return ".%s" % match.group(1)
    164     except:
    165         pass
    166 
    167     return ""
    168 
    169 def get_config(wk_root):
    170     config_file = os.path.join(wk_root, 'WebKitBuild', 'Configuration')
    171     config = 'Debug'
    172 
    173     if os.path.exists(config_file):
    174         config = open(config_file).read()
    175 
    176     return config
    177 
    178 def svn_revision():
    179     if os.system("git-svn info") == 0:
    180         info = commands.getoutput("git-svn info ../..")
    181     else:
    182         info = commands.getoutput("svn info")
    183     
    184     for line in info.split("\n"):
    185         if line.startswith("Revision: "):
    186             return line.replace("Revision: ", "").strip()
    187     
    188     return ""
    189