xref: /petsc/config/install.py (revision c3a89c15c66be42c1ea3c2eab6faebba95f76ffa)
10ee81e68SLisandro Dalcin#!/usr/bin/env python
2e551db17SScott Krugerimport os, re, shutil, sys
30ee81e68SLisandro Dalcin
40ee81e68SLisandro Dalcinif os.environ.has_key('PETSC_DIR'):
50ee81e68SLisandro Dalcin  PETSC_DIR = os.environ['PETSC_DIR']
60ee81e68SLisandro Dalcinelse:
7af0996ceSBarry Smith  fd = file(os.path.join('lib','petsc','conf','petscvariables'))
80ee81e68SLisandro Dalcin  a = fd.readline()
90ee81e68SLisandro Dalcin  a = fd.readline()
100ee81e68SLisandro Dalcin  PETSC_DIR = a.split('=')[1][0:-1]
110ee81e68SLisandro Dalcin  fd.close()
120ee81e68SLisandro Dalcin
130ee81e68SLisandro Dalcinif os.environ.has_key('PETSC_ARCH'):
140ee81e68SLisandro Dalcin  PETSC_ARCH = os.environ['PETSC_ARCH']
150ee81e68SLisandro Dalcinelse:
16af0996ceSBarry Smith  fd = file(os.path.join('lib','petsc','conf','petscvariables'))
170ee81e68SLisandro Dalcin  a = fd.readline()
180ee81e68SLisandro Dalcin  PETSC_ARCH = a.split('=')[1][0:-1]
190ee81e68SLisandro Dalcin  fd.close()
200ee81e68SLisandro Dalcin
210ee81e68SLisandro Dalcinprint '*** Using PETSC_DIR='+PETSC_DIR+' PETSC_ARCH='+PETSC_ARCH+' ***'
220ee81e68SLisandro Dalcinsys.path.insert(0, os.path.join(PETSC_DIR, 'config'))
230ee81e68SLisandro Dalcinsys.path.insert(0, os.path.join(PETSC_DIR, 'config', 'BuildSystem'))
240ee81e68SLisandro Dalcin
250ee81e68SLisandro Dalcinimport script
260ee81e68SLisandro Dalcin
270ee81e68SLisandro Dalcintry:
280ee81e68SLisandro Dalcin  WindowsError
290ee81e68SLisandro Dalcinexcept NameError:
300ee81e68SLisandro Dalcin  WindowsError = None
310ee81e68SLisandro Dalcin
320ee81e68SLisandro Dalcinclass Installer(script.Script):
330ee81e68SLisandro Dalcin  def __init__(self, clArgs = None):
340ee81e68SLisandro Dalcin    import RDict
350ee81e68SLisandro Dalcin    argDB = RDict.RDict(None, None, 0, 0, readonly = True)
36af0996ceSBarry Smith    argDB.saveFilename = os.path.join(PETSC_DIR, PETSC_ARCH, 'lib','petsc','conf', 'RDict.db')
370ee81e68SLisandro Dalcin    argDB.load()
380ee81e68SLisandro Dalcin    script.Script.__init__(self, argDB = argDB)
390ee81e68SLisandro Dalcin    if not clArgs is None: self.clArgs = clArgs
400ee81e68SLisandro Dalcin    self.copies = []
410ee81e68SLisandro Dalcin    return
420ee81e68SLisandro Dalcin
430ee81e68SLisandro Dalcin  def setupHelp(self, help):
440ee81e68SLisandro Dalcin    import nargs
450ee81e68SLisandro Dalcin    script.Script.setupHelp(self, help)
460ee81e68SLisandro Dalcin    help.addArgument('Installer', '-destDir=<path>', nargs.Arg(None, None, 'Destination Directory for install'))
470ee81e68SLisandro Dalcin    return
480ee81e68SLisandro Dalcin
490ee81e68SLisandro Dalcin
500ee81e68SLisandro Dalcin  def setupModules(self):
510ee81e68SLisandro Dalcin    self.setCompilers  = self.framework.require('config.setCompilers',         None)
520ee81e68SLisandro Dalcin    self.arch          = self.framework.require('PETSc.options.arch',        None)
530ee81e68SLisandro Dalcin    self.petscdir      = self.framework.require('PETSc.options.petscdir',    None)
540ee81e68SLisandro Dalcin    self.compilers     = self.framework.require('config.compilers',            None)
550ee81e68SLisandro Dalcin    return
560ee81e68SLisandro Dalcin
570ee81e68SLisandro Dalcin  def setup(self):
580ee81e68SLisandro Dalcin    script.Script.setup(self)
590ee81e68SLisandro Dalcin    self.framework = self.loadConfigure()
600ee81e68SLisandro Dalcin    self.setupModules()
610ee81e68SLisandro Dalcin    return
620ee81e68SLisandro Dalcin
630ee81e68SLisandro Dalcin  def setupDirectories(self):
640ee81e68SLisandro Dalcin    self.rootDir    = self.petscdir.dir
650ee81e68SLisandro Dalcin    self.destDir    = os.path.abspath(self.argDB['destDir'])
664a08bad0SBarry Smith    self.installDir = os.path.abspath(os.path.expanduser(self.framework.argDB['prefix']))
670ee81e68SLisandro Dalcin    self.arch       = self.arch.arch
6802e047dfSSatish Balay    self.archDir           = os.path.join(self.rootDir, self.arch)
690ee81e68SLisandro Dalcin    self.rootIncludeDir    = os.path.join(self.rootDir, 'include')
700ee81e68SLisandro Dalcin    self.archIncludeDir    = os.path.join(self.rootDir, self.arch, 'include')
71af0996ceSBarry Smith    self.rootConfDir       = os.path.join(self.rootDir, 'lib','petsc','conf')
72af0996ceSBarry Smith    self.archConfDir       = os.path.join(self.rootDir, self.arch, 'lib','petsc','conf')
73*c3a89c15SBarry Smith    self.rootBinDir        = os.path.join(self.rootDir, 'lib','petsc','bin')
740ee81e68SLisandro Dalcin    self.archBinDir        = os.path.join(self.rootDir, self.arch, 'bin')
750ee81e68SLisandro Dalcin    self.archLibDir        = os.path.join(self.rootDir, self.arch, 'lib')
760ee81e68SLisandro Dalcin    self.destIncludeDir    = os.path.join(self.destDir, 'include')
77af0996ceSBarry Smith    self.destConfDir       = os.path.join(self.destDir, 'lib','petsc','conf')
780ee81e68SLisandro Dalcin    self.destLibDir        = os.path.join(self.destDir, 'lib')
79*c3a89c15SBarry Smith    self.destBinDir        = os.path.join(self.destDir, 'lib','petsc','bin')
800ee81e68SLisandro Dalcin    self.installIncludeDir = os.path.join(self.installDir, 'include')
81*c3a89c15SBarry Smith    self.installBinDir     = os.path.join(self.installDir, 'lib','petsc','bin')
820ee81e68SLisandro Dalcin    self.rootShareDir      = os.path.join(self.rootDir, 'share')
830ee81e68SLisandro Dalcin    self.destShareDir      = os.path.join(self.destDir, 'share')
8479eaf171SScott Kruger    self.rootSrcDir        = os.path.join(self.rootDir, 'src')
850ee81e68SLisandro Dalcin
860ee81e68SLisandro Dalcin    self.ranlib      = self.compilers.RANLIB
870ee81e68SLisandro Dalcin    self.arLibSuffix = self.compilers.AR_LIB_SUFFIX
880ee81e68SLisandro Dalcin    return
890ee81e68SLisandro Dalcin
900ee81e68SLisandro Dalcin  def checkPrefix(self):
910ee81e68SLisandro Dalcin    if not self.installDir:
920ee81e68SLisandro Dalcin      print '********************************************************************'
930ee81e68SLisandro Dalcin      print 'PETSc is built without prefix option. So "make install" is not appropriate.'
940ee81e68SLisandro Dalcin      print 'If you need a prefix install of PETSc - rerun configure with --prefix option.'
950ee81e68SLisandro Dalcin      print '********************************************************************'
960ee81e68SLisandro Dalcin      sys.exit(1)
970ee81e68SLisandro Dalcin    return
980ee81e68SLisandro Dalcin
990ee81e68SLisandro Dalcin  def checkDestdir(self):
1000ee81e68SLisandro Dalcin    if os.path.exists(self.destDir):
1010ee81e68SLisandro Dalcin      if os.path.samefile(self.destDir, self.rootDir):
1020ee81e68SLisandro Dalcin        print '********************************************************************'
1030ee81e68SLisandro Dalcin        print 'Incorrect prefix usage. Specified destDir same as current PETSC_DIR'
1040ee81e68SLisandro Dalcin        print '********************************************************************'
1050ee81e68SLisandro Dalcin        sys.exit(1)
1060ee81e68SLisandro Dalcin      if os.path.samefile(self.destDir, os.path.join(self.rootDir,self.arch)):
1070ee81e68SLisandro Dalcin        print '********************************************************************'
1080ee81e68SLisandro Dalcin        print 'Incorrect prefix usage. Specified destDir same as current PETSC_DIR/PETSC_ARCH'
1090ee81e68SLisandro Dalcin        print '********************************************************************'
1100ee81e68SLisandro Dalcin        sys.exit(1)
1110ee81e68SLisandro Dalcin      if not os.path.isdir(os.path.realpath(self.destDir)):
1120ee81e68SLisandro Dalcin        print '********************************************************************'
1130ee81e68SLisandro Dalcin        print 'Specified destDir', self.destDir, 'is not a directory. Cannot proceed!'
1140ee81e68SLisandro Dalcin        print '********************************************************************'
1150ee81e68SLisandro Dalcin        sys.exit(1)
1160ee81e68SLisandro Dalcin      if not os.access(self.destDir, os.W_OK):
1170ee81e68SLisandro Dalcin        print '********************************************************************'
1180ee81e68SLisandro Dalcin        print 'Unable to write to ', self.destDir, 'Perhaps you need to do "sudo make install"'
1190ee81e68SLisandro Dalcin        print '********************************************************************'
1200ee81e68SLisandro Dalcin        sys.exit(1)
1210ee81e68SLisandro Dalcin    return
1220ee81e68SLisandro Dalcin
1230ee81e68SLisandro Dalcin  def copyfile(self, src, dst, symlinks = False, copyFunc = shutil.copy2):
1240ee81e68SLisandro Dalcin    """Copies a single file    """
1250ee81e68SLisandro Dalcin    copies = []
1260ee81e68SLisandro Dalcin    errors = []
1270ee81e68SLisandro Dalcin    if not os.path.exists(dst):
1280ee81e68SLisandro Dalcin      os.makedirs(dst)
1290ee81e68SLisandro Dalcin    elif not os.path.isdir(dst):
1300ee81e68SLisandro Dalcin      raise shutil.Error, 'Destination is not a directory'
1310ee81e68SLisandro Dalcin    srcname = src
1320ee81e68SLisandro Dalcin    dstname = os.path.join(dst, os.path.basename(src))
1330ee81e68SLisandro Dalcin    try:
1340ee81e68SLisandro Dalcin      if symlinks and os.path.islink(srcname):
1350ee81e68SLisandro Dalcin        linkto = os.readlink(srcname)
1360ee81e68SLisandro Dalcin        os.symlink(linkto, dstname)
1370ee81e68SLisandro Dalcin      else:
1380ee81e68SLisandro Dalcin        copyFunc(srcname, dstname)
1390ee81e68SLisandro Dalcin        copies.append((srcname, dstname))
1400ee81e68SLisandro Dalcin    except (IOError, os.error), why:
1410ee81e68SLisandro Dalcin      errors.append((srcname, dstname, str(why)))
1420ee81e68SLisandro Dalcin    except shutil.Error, err:
1430ee81e68SLisandro Dalcin      errors.extend((srcname,dstname,str(err.args[0])))
1440ee81e68SLisandro Dalcin    if errors:
1450ee81e68SLisandro Dalcin      raise shutil.Error, errors
1460ee81e68SLisandro Dalcin    return copies
1470ee81e68SLisandro Dalcin
14826e8aaceSBarry Smith  def copyexamplefiles(self, src, dst, copyFunc = shutil.copy2):
14926e8aaceSBarry Smith    """Copies all files, but not directories in a single file    """
15026e8aaceSBarry Smith    names  = os.listdir(src)
15126e8aaceSBarry Smith    for name in names:
15226e8aaceSBarry Smith      if not name.endswith('.html'):
15326e8aaceSBarry Smith        srcname = os.path.join(src, name)
15426e8aaceSBarry Smith        if os.path.isfile(srcname):
15526e8aaceSBarry Smith           self.copyfile(srcname,dst)
15626e8aaceSBarry Smith
15726e8aaceSBarry Smith  def fixExamplesMakefile(self, src):
15826e8aaceSBarry Smith    '''Change ././${PETSC_ARCH} in makefile in root petsc directory with ${PETSC_DIR}'''
15926e8aaceSBarry Smith    lines   = []
16026e8aaceSBarry Smith    oldFile = open(src, 'r')
161e551db17SScott Kruger    alllines=oldFile.read()
16226e8aaceSBarry Smith    oldFile.close()
163e551db17SScott Kruger    newlines=alllines.split('\n')[0]+'\n'  # Firstline
164e551db17SScott Kruger    # Hardcode PETSC_DIR and PETSC_ARCH to avoid users doing the worng thing
165e551db17SScott Kruger    newlines+='PETSC_DIR='+self.installDir+'\n'
166e551db17SScott Kruger    newlines+='PETSC_ARCH=\n'
167e551db17SScott Kruger    for line in alllines.split('\n')[1:]:
1684ff3c6a1SScott Kruger      if line.startswith('TESTLOGFILE'):
169c173c275SScott Kruger        newlines+='TESTLOGFILE = $(TESTDIR)/examples-install.log\n'
170e551db17SScott Kruger      elif line.startswith('CONFIGDIR'):
171e551db17SScott Kruger        newlines+='CONFIGDIR:=$(PETSC_DIR)/$(PETSC_ARCH)/share/petsc/examples/config\n'
1724ff3c6a1SScott Kruger      elif line.startswith('EXAMPLESDIR'):
173e551db17SScott Kruger        newlines+='EXAMPLESDIR:=$(PETSC_DIR)/$(PETSC_ARCH)/share/petsc/examples\n'
174fc46264cSScott Kruger      elif line.startswith('$(generatedtest)') and 'petscvariables' in line:
175c173c275SScott Kruger        newlines+='all: test\n\n'+line+'\n'
176e551db17SScott Kruger      else:
1774ff3c6a1SScott Kruger        newlines+=line+'\n'
17826e8aaceSBarry Smith    newFile = open(src, 'w')
179e551db17SScott Kruger    newFile.write(newlines)
18026e8aaceSBarry Smith    newFile.close()
18126e8aaceSBarry Smith    return
18226e8aaceSBarry Smith
183e551db17SScott Kruger  def copyConfig(self, src, dst):
184e551db17SScott Kruger    """Recursively copy the examples directories
185e551db17SScott Kruger    """
186e551db17SScott Kruger    if not os.path.isdir(dst):
187e551db17SScott Kruger      raise shutil.Error, 'Destination is not a directory'
188e551db17SScott Kruger
18949da0fa4SScott Kruger    self.copyfile('gmakefile.test',dst)
190e551db17SScott Kruger    newConfigDir=os.path.join(dst,'config')  # Am not renaming at present
191e551db17SScott Kruger    if not os.path.isdir(newConfigDir): os.mkdir(newConfigDir)
192e551db17SScott Kruger    testConfFiles="gmakegentest.py gmakegen.py testparse.py example_template.py".split()
193c173c275SScott Kruger    testConfFiles+="petsc_harness.sh report_tests.py watchtime.sh".split()
194c173c275SScott Kruger    testConfFiles+=["cmakegen.py"]
195e551db17SScott Kruger    for tf in testConfFiles:
196e551db17SScott Kruger      self.copyfile(os.path.join('config',tf),newConfigDir)
197e551db17SScott Kruger    return
198e551db17SScott Kruger
19926e8aaceSBarry Smith  def copyExamples(self, src, dst):
2004ff3c6a1SScott Kruger    """copy the examples directories
20126e8aaceSBarry Smith    """
2024ff3c6a1SScott Kruger    top=os.path.relpath(src,os.path.abspath(os.curdir))
2034ff3c6a1SScott Kruger    for root, dirs, files in os.walk(top, topdown=False):
2044ff3c6a1SScott Kruger        if not os.path.basename(root) == "examples": continue
2054ff3c6a1SScott Kruger        shutil.copytree(root, os.path.join(dst,root),
2064ff3c6a1SScott Kruger                        ignore=shutil.ignore_patterns('*.dSYM'))
20726e8aaceSBarry Smith
2084ff3c6a1SScott Kruger    return
2090ee81e68SLisandro Dalcin
2100ee81e68SLisandro Dalcin  def copytree(self, src, dst, symlinks = False, copyFunc = shutil.copy2, exclude = []):
2110ee81e68SLisandro Dalcin    """Recursively copy a directory tree using copyFunc, which defaults to shutil.copy2().
2120ee81e68SLisandro Dalcin
2130ee81e68SLisandro Dalcin       The copyFunc() you provide is only used on the top level, lower levels always use shutil.copy2
2140ee81e68SLisandro Dalcin
2150ee81e68SLisandro Dalcin    The destination directory must not already exist.
2160ee81e68SLisandro Dalcin    If exception(s) occur, an shutil.Error is raised with a list of reasons.
2170ee81e68SLisandro Dalcin
2180ee81e68SLisandro Dalcin    If the optional symlinks flag is true, symbolic links in the
2190ee81e68SLisandro Dalcin    source tree result in symbolic links in the destination tree; if
2200ee81e68SLisandro Dalcin    it is false, the contents of the files pointed to by symbolic
2210ee81e68SLisandro Dalcin    links are copied.
2220ee81e68SLisandro Dalcin    """
2230ee81e68SLisandro Dalcin    copies = []
2240ee81e68SLisandro Dalcin    names  = os.listdir(src)
2250ee81e68SLisandro Dalcin    if not os.path.exists(dst):
2260ee81e68SLisandro Dalcin      os.makedirs(dst)
2270ee81e68SLisandro Dalcin    elif not os.path.isdir(dst):
2280ee81e68SLisandro Dalcin      raise shutil.Error, 'Destination is not a directory'
2290ee81e68SLisandro Dalcin    errors = []
2300ee81e68SLisandro Dalcin    for name in names:
2310ee81e68SLisandro Dalcin      srcname = os.path.join(src, name)
2320ee81e68SLisandro Dalcin      dstname = os.path.join(dst, name)
2330ee81e68SLisandro Dalcin      try:
2340ee81e68SLisandro Dalcin        if symlinks and os.path.islink(srcname):
2350ee81e68SLisandro Dalcin          linkto = os.readlink(srcname)
2360ee81e68SLisandro Dalcin          os.symlink(linkto, dstname)
2370ee81e68SLisandro Dalcin        elif os.path.isdir(srcname):
2380ee81e68SLisandro Dalcin          copies.extend(self.copytree(srcname, dstname, symlinks,exclude = exclude))
2390ee81e68SLisandro Dalcin        elif not os.path.basename(srcname) in exclude:
2400ee81e68SLisandro Dalcin          copyFunc(srcname, dstname)
2410ee81e68SLisandro Dalcin          copies.append((srcname, dstname))
2420ee81e68SLisandro Dalcin        # XXX What about devices, sockets etc.?
2430ee81e68SLisandro Dalcin      except (IOError, os.error), why:
2440ee81e68SLisandro Dalcin        errors.append((srcname, dstname, str(why)))
2450ee81e68SLisandro Dalcin      # catch the Error from the recursive copytree so that we can
2460ee81e68SLisandro Dalcin      # continue with other files
2470ee81e68SLisandro Dalcin      except shutil.Error, err:
2480ee81e68SLisandro Dalcin        errors.extend((srcname,dstname,str(err.args[0])))
2490ee81e68SLisandro Dalcin    try:
2500ee81e68SLisandro Dalcin      shutil.copystat(src, dst)
2510ee81e68SLisandro Dalcin    except OSError, e:
2520ee81e68SLisandro Dalcin      if WindowsError is not None and isinstance(e, WindowsError):
2530ee81e68SLisandro Dalcin        # Copying file access times may fail on Windows
2540ee81e68SLisandro Dalcin        pass
2550ee81e68SLisandro Dalcin      else:
2560ee81e68SLisandro Dalcin        errors.extend((src, dst, str(e)))
2570ee81e68SLisandro Dalcin    if errors:
2580ee81e68SLisandro Dalcin      raise shutil.Error, errors
2590ee81e68SLisandro Dalcin    return copies
2600ee81e68SLisandro Dalcin
2610ee81e68SLisandro Dalcin
2620ee81e68SLisandro Dalcin  def fixConfFile(self, src):
2630ee81e68SLisandro Dalcin    lines   = []
2640ee81e68SLisandro Dalcin    oldFile = open(src, 'r')
2650ee81e68SLisandro Dalcin    for line in oldFile.readlines():
2660ee81e68SLisandro Dalcin      # paths generated by configure could be different link-path than whats used by user, so fix both
2670ee81e68SLisandro Dalcin      line = line.replace(os.path.join(self.rootDir, self.arch), self.installDir)
2680ee81e68SLisandro Dalcin      line = line.replace(os.path.realpath(os.path.join(self.rootDir, self.arch)), self.installDir)
2690ee81e68SLisandro Dalcin      line = line.replace(os.path.join(self.rootDir, 'bin'), self.installBinDir)
2700ee81e68SLisandro Dalcin      line = line.replace(os.path.realpath(os.path.join(self.rootDir, 'bin')), self.installBinDir)
2710ee81e68SLisandro Dalcin      line = line.replace(os.path.join(self.rootDir, 'include'), self.installIncludeDir)
2720ee81e68SLisandro Dalcin      line = line.replace(os.path.realpath(os.path.join(self.rootDir, 'include')), self.installIncludeDir)
2730ee81e68SLisandro Dalcin      # remove PETSC_DIR/PETSC_ARCH variables from conf-makefiles. They are no longer necessary
2740ee81e68SLisandro Dalcin      line = line.replace('${PETSC_DIR}/${PETSC_ARCH}', self.installDir)
2750ee81e68SLisandro Dalcin      line = line.replace('PETSC_ARCH=${PETSC_ARCH}', '')
2760ee81e68SLisandro Dalcin      line = line.replace('${PETSC_DIR}', self.installDir)
2770ee81e68SLisandro Dalcin      lines.append(line)
2780ee81e68SLisandro Dalcin    oldFile.close()
2790ee81e68SLisandro Dalcin    newFile = open(src, 'w')
2800ee81e68SLisandro Dalcin    newFile.write(''.join(lines))
2810ee81e68SLisandro Dalcin    newFile.close()
2820ee81e68SLisandro Dalcin    return
2830ee81e68SLisandro Dalcin
2840ee81e68SLisandro Dalcin  def fixConf(self):
2850ee81e68SLisandro Dalcin    import shutil
2860ee81e68SLisandro Dalcin    for file in ['rules', 'variables','petscrules', 'petscvariables']:
2870ee81e68SLisandro Dalcin      self.fixConfFile(os.path.join(self.destConfDir,file))
2880ee81e68SLisandro Dalcin    self.fixConfFile(os.path.join(self.destLibDir,'pkgconfig','PETSc.pc'))
2890ee81e68SLisandro Dalcin    return
2900ee81e68SLisandro Dalcin
2910ee81e68SLisandro Dalcin  def createUninstaller(self):
2920ee81e68SLisandro Dalcin    uninstallscript = os.path.join(self.destConfDir, 'uninstall.py')
2930ee81e68SLisandro Dalcin    f = open(uninstallscript, 'w')
2940ee81e68SLisandro Dalcin    # Could use the Python AST to do this
2950ee81e68SLisandro Dalcin    f.write('#!'+sys.executable+'\n')
2960ee81e68SLisandro Dalcin    f.write('import os\n')
2970ee81e68SLisandro Dalcin
2980ee81e68SLisandro Dalcin    f.write('copies = '+repr(self.copies).replace(self.destDir,self.installDir))
2990ee81e68SLisandro Dalcin    f.write('''
3000ee81e68SLisandro Dalcinfor src, dst in copies:
3010ee81e68SLisandro Dalcin  try:
3020ee81e68SLisandro Dalcin    os.remove(dst)
3030ee81e68SLisandro Dalcin  except:
3040ee81e68SLisandro Dalcin    pass
3050ee81e68SLisandro Dalcin''')
3060ee81e68SLisandro Dalcin    #TODO: need to delete libXXX.YYY.dylib.dSYM directory on Mac
307af0996ceSBarry Smith    dirs = [os.path.join('include','petsc','finclude'),os.path.join('include','petsc','mpiuni'),os.path.join('include','petsc','private'),os.path.join('bin'),os.path.join('lib','petsc','conf')]
3080ee81e68SLisandro Dalcin    newdirs = []
3090ee81e68SLisandro Dalcin    for dir in dirs: newdirs.append(os.path.join(self.installDir,dir))
3100ee81e68SLisandro Dalcin    f.write('dirs = '+str(newdirs))
3110ee81e68SLisandro Dalcin    f.write('''
3120ee81e68SLisandro Dalcinfor dir in dirs:
3130ee81e68SLisandro Dalcin  import shutil
3140ee81e68SLisandro Dalcin  try:
3150ee81e68SLisandro Dalcin    shutil.rmtree(dir)
3160ee81e68SLisandro Dalcin  except:
3170ee81e68SLisandro Dalcin    pass
3180ee81e68SLisandro Dalcin''')
3190ee81e68SLisandro Dalcin    f.close()
3200ee81e68SLisandro Dalcin    os.chmod(uninstallscript,0744)
3210ee81e68SLisandro Dalcin    return
3220ee81e68SLisandro Dalcin
3230ee81e68SLisandro Dalcin  def installIncludes(self):
3240ee81e68SLisandro Dalcin    # TODO: should exclude petsc-mpi.uni except for uni builds
325af0996ceSBarry Smith    # TODO: should exclude petsc/finclude except for fortran builds
3260ee81e68SLisandro Dalcin    self.copies.extend(self.copytree(self.rootIncludeDir, self.destIncludeDir,exclude = ['makefile']))
3270ee81e68SLisandro Dalcin    self.copies.extend(self.copytree(self.archIncludeDir, self.destIncludeDir))
3280ee81e68SLisandro Dalcin    return
3290ee81e68SLisandro Dalcin
3300ee81e68SLisandro Dalcin  def installConf(self):
3310ee81e68SLisandro Dalcin    self.copies.extend(self.copytree(self.rootConfDir, self.destConfDir))
3320ee81e68SLisandro Dalcin    self.copies.extend(self.copytree(self.archConfDir, self.destConfDir, exclude = ['sowing', 'configure.log.bkp']))
3330ee81e68SLisandro Dalcin    return
3340ee81e68SLisandro Dalcin
3350ee81e68SLisandro Dalcin  def installBin(self):
336af0996ceSBarry Smith    self.copies.extend(self.copytree(os.path.join(self.rootBinDir), os.path.join(self.destBinDir)))
3370ee81e68SLisandro Dalcin    # TODO: should copy over petsc-mpiexec.uni only for uni builds
3380ee81e68SLisandro Dalcin    self.copies.extend(self.copyfile(os.path.join(self.rootBinDir,'petsc-mpiexec.uni'), self.destBinDir))
3390ee81e68SLisandro Dalcin    self.copies.extend(self.copytree(self.archBinDir, self.destBinDir, exclude = ['bfort','bib2html','doc2lt','doctext','mapnames', 'pstogif','pstoxbm','tohtml']))
3400ee81e68SLisandro Dalcin    return
3410ee81e68SLisandro Dalcin
3420ee81e68SLisandro Dalcin  def installShare(self):
3430ee81e68SLisandro Dalcin    self.copies.extend(self.copytree(self.rootShareDir, self.destShareDir))
344c173c275SScott Kruger    examplesdir=os.path.join(self.destShareDir,'petsc','examples')
345c173c275SScott Kruger    if os.path.exists(examplesdir):
346c173c275SScott Kruger      shutil.rmtree(examplesdir)
347c173c275SScott Kruger    os.mkdir(examplesdir)
34879eaf171SScott Kruger    os.mkdir(os.path.join(examplesdir,'src'))
3494ff3c6a1SScott Kruger    self.copyExamples(self.rootSrcDir,examplesdir)
350c173c275SScott Kruger    self.copyConfig(self.rootDir,examplesdir)
351c173c275SScott Kruger    self.fixExamplesMakefile(os.path.join(examplesdir,'gmakefile.test'))
3520ee81e68SLisandro Dalcin    return
3530ee81e68SLisandro Dalcin
3540ee81e68SLisandro Dalcin  def copyLib(self, src, dst):
3550ee81e68SLisandro Dalcin    '''Run ranlib on the destination library if it is an archive. Also run install_name_tool on dylib on Mac'''
3560ee81e68SLisandro Dalcin    # Symlinks (assumed local) are recreated at dst
3570ee81e68SLisandro Dalcin    if os.path.islink(src):
3580ee81e68SLisandro Dalcin      linkto = os.readlink(src)
3590ee81e68SLisandro Dalcin      try:
3600ee81e68SLisandro Dalcin        os.remove(dst)            # In case it already exists
3610ee81e68SLisandro Dalcin      except OSError:
3620ee81e68SLisandro Dalcin        pass
3630ee81e68SLisandro Dalcin      os.symlink(linkto, dst)
3640ee81e68SLisandro Dalcin      return
3650ee81e68SLisandro Dalcin    # Do not install object files
3660ee81e68SLisandro Dalcin    if not os.path.splitext(src)[1] == '.o':
3670ee81e68SLisandro Dalcin      shutil.copy2(src, dst)
3680ee81e68SLisandro Dalcin    if os.path.splitext(dst)[1] == '.'+self.arLibSuffix:
3690ee81e68SLisandro Dalcin      self.executeShellCommand(self.ranlib+' '+dst)
3700ee81e68SLisandro Dalcin    if os.path.splitext(dst)[1] == '.dylib' and os.path.isfile('/usr/bin/install_name_tool'):
371af2c601bSBarry Smith      [output,err,flg] = self.executeShellCommand("otool -D "+src)
372af2c601bSBarry Smith      oldname = output[output.find("\n")+1:]
37302e047dfSSatish Balay      installName = oldname.replace(self.archDir, self.installDir)
3740ee81e68SLisandro Dalcin      self.executeShellCommand('/usr/bin/install_name_tool -id ' + installName + ' ' + dst)
3750ee81e68SLisandro Dalcin    # preserve the original timestamps - so that the .a vs .so time order is preserved
3760ee81e68SLisandro Dalcin    shutil.copystat(src,dst)
3770ee81e68SLisandro Dalcin    return
3780ee81e68SLisandro Dalcin
3790ee81e68SLisandro Dalcin  def installLib(self):
380696e41a9SLisandro Dalcin    self.copies.extend(self.copytree(self.archLibDir, self.destLibDir, copyFunc = self.copyLib, exclude = ['.DIR', 'sowing']))
3810ee81e68SLisandro Dalcin    return
3820ee81e68SLisandro Dalcin
3830ee81e68SLisandro Dalcin
3840ee81e68SLisandro Dalcin  def outputInstallDone(self):
3850ee81e68SLisandro Dalcin    print '''\
3860ee81e68SLisandro Dalcin====================================
3870ee81e68SLisandro DalcinInstall complete.
3880ee81e68SLisandro DalcinNow to check if the libraries are working do (in current directory):
3890ee81e68SLisandro Dalcinmake PETSC_DIR=%s PETSC_ARCH="" test
3900ee81e68SLisandro Dalcin====================================\
3910ee81e68SLisandro Dalcin''' % (self.installDir)
3920ee81e68SLisandro Dalcin    return
3930ee81e68SLisandro Dalcin
3940ee81e68SLisandro Dalcin  def outputDestDirDone(self):
3950ee81e68SLisandro Dalcin    print '''\
3960ee81e68SLisandro Dalcin====================================
3970ee81e68SLisandro DalcinCopy to DESTDIR %s is now complete.
3980ee81e68SLisandro DalcinBefore use - please copy/install over to specified prefix: %s
3990ee81e68SLisandro Dalcin====================================\
4000ee81e68SLisandro Dalcin''' % (self.destDir,self.installDir)
4010ee81e68SLisandro Dalcin    return
4020ee81e68SLisandro Dalcin
4030ee81e68SLisandro Dalcin  def runsetup(self):
4040ee81e68SLisandro Dalcin    self.setup()
4050ee81e68SLisandro Dalcin    self.setupDirectories()
4060ee81e68SLisandro Dalcin    self.checkPrefix()
4070ee81e68SLisandro Dalcin    self.checkDestdir()
4080ee81e68SLisandro Dalcin    return
4090ee81e68SLisandro Dalcin
4100ee81e68SLisandro Dalcin  def runcopy(self):
4110ee81e68SLisandro Dalcin    if self.destDir == self.installDir:
4120ee81e68SLisandro Dalcin      print '*** Installing PETSc at prefix location:',self.destDir, ' ***'
4130ee81e68SLisandro Dalcin    else:
4140ee81e68SLisandro Dalcin      print '*** Copying PETSc to DESTDIR location:',self.destDir, ' ***'
4150ee81e68SLisandro Dalcin    if not os.path.exists(self.destDir):
4160ee81e68SLisandro Dalcin      try:
4170ee81e68SLisandro Dalcin        os.makedirs(self.destDir)
4180ee81e68SLisandro Dalcin      except:
4190ee81e68SLisandro Dalcin        print '********************************************************************'
4200ee81e68SLisandro Dalcin        print 'Unable to create', self.destDir, 'Perhaps you need to do "sudo make install"'
4210ee81e68SLisandro Dalcin        print '********************************************************************'
4220ee81e68SLisandro Dalcin        sys.exit(1)
4230ee81e68SLisandro Dalcin    self.installIncludes()
4240ee81e68SLisandro Dalcin    self.installConf()
4250ee81e68SLisandro Dalcin    self.installBin()
4260ee81e68SLisandro Dalcin    self.installLib()
4270ee81e68SLisandro Dalcin    self.installShare()
4280ee81e68SLisandro Dalcin    return
4290ee81e68SLisandro Dalcin
4300ee81e68SLisandro Dalcin  def runfix(self):
4310ee81e68SLisandro Dalcin    self.fixConf()
4320ee81e68SLisandro Dalcin    return
4330ee81e68SLisandro Dalcin
4340ee81e68SLisandro Dalcin  def rundone(self):
4350ee81e68SLisandro Dalcin    self.createUninstaller()
4360ee81e68SLisandro Dalcin    if self.destDir == self.installDir:
4370ee81e68SLisandro Dalcin      self.outputInstallDone()
4380ee81e68SLisandro Dalcin    else:
4390ee81e68SLisandro Dalcin      self.outputDestDirDone()
4400ee81e68SLisandro Dalcin    return
4410ee81e68SLisandro Dalcin
4420ee81e68SLisandro Dalcin  def run(self):
4430ee81e68SLisandro Dalcin    self.runsetup()
4440ee81e68SLisandro Dalcin    self.runcopy()
4450ee81e68SLisandro Dalcin    self.runfix()
4460ee81e68SLisandro Dalcin    self.rundone()
4470ee81e68SLisandro Dalcin    return
4480ee81e68SLisandro Dalcin
4490ee81e68SLisandro Dalcinif __name__ == '__main__':
4500ee81e68SLisandro Dalcin  Installer(sys.argv[1:]).run()
4510ee81e68SLisandro Dalcin  # temporary hack - delete log files created by BuildSystem - when 'sudo make install' is invoked
45203e6d329SSatish Balay  delfiles=['RDict.db','RDict.log','buildsystem.log','default.log','buildsystem.log.bkp','default.log.bkp']
4530ee81e68SLisandro Dalcin  for delfile in delfiles:
4540ee81e68SLisandro Dalcin    if os.path.exists(delfile) and (os.stat(delfile).st_uid==0):
4550ee81e68SLisandro Dalcin      os.remove(delfile)
456