xref: /petsc/config/PETSc/Configure.py (revision 27b0f280e5bf9c639a7764debf82fd1b76d8a199)
1f8833479SBarry Smithimport config.base
2f8833479SBarry Smith
3f8833479SBarry Smithimport os
46dd73af6SBarry Smithimport sys
5f8833479SBarry Smithimport re
640277576SBarry Smithimport cPickle
7f8833479SBarry Smith
88bec23c5SJed Brown# The sorted() builtin is not available with python-2.3
98bec23c5SJed Browntry: sorted
108bec23c5SJed Brownexcept NameError:
118bec23c5SJed Brown  def sorted(lst):
128bec23c5SJed Brown    lst.sort()
138bec23c5SJed Brown    return lst
148bec23c5SJed Brown
15f8833479SBarry Smithclass Configure(config.base.Configure):
16f8833479SBarry Smith  def __init__(self, framework):
17f8833479SBarry Smith    config.base.Configure.__init__(self, framework)
18f8833479SBarry Smith    self.headerPrefix = 'PETSC'
19f8833479SBarry Smith    self.substPrefix  = 'PETSC'
20f8833479SBarry Smith    return
21f8833479SBarry Smith
227c939e48SSatish Balay  def __str2__(self):
237c939e48SSatish Balay    desc = []
24a0022257SSatish Balay    desc.append('xxx=========================================================================xxx')
25ac1d0f13SJed Brown    if self.make.getMakeMacro('MAKE_IS_GNUMAKE'):
269481793eSSatish Balay      build_type = 'gnumake build'
279481793eSSatish Balay    elif self.getMakeMacro('PETSC_BUILD_USING_CMAKE'):
28b3618d6dSSatish Balay      build_type = 'cmake build'
29b3618d6dSSatish Balay    else:
30b3618d6dSSatish Balay      build_type = 'legacy build'
31b3618d6dSSatish Balay    desc.append(' Configure stage complete. Now build PETSc libraries with (%s):' % build_type)
32b3618d6dSSatish Balay    desc.append('   make PETSC_DIR='+self.petscdir.dir+' PETSC_ARCH='+self.arch.arch+' all')
33a0022257SSatish Balay    desc.append('xxx=========================================================================xxx')
347c939e48SSatish Balay    return '\n'.join(desc)+'\n'
35f8833479SBarry Smith
36f8833479SBarry Smith  def setupHelp(self, help):
37f8833479SBarry Smith    import nargs
38ce0b2093SBarry Smith    help.addArgument('PETSc',  '-prefix=<dir>',                   nargs.Arg(None, '', 'Specifiy location to install PETSc (eg. /usr/local)'))
397deb5ab3SBarry Smith    help.addArgument('PETSc',  '-with-prefetch=<bool>',           nargs.ArgBool(None, 1,'Enable checking for prefetch instructions'))
40eed94e11SSatish Balay    help.addArgument('Windows','-with-windows-graphics=<bool>',   nargs.ArgBool(None, 1,'Enable check for Windows Graphics'))
41569865ddSSatish Balay    help.addArgument('PETSc', '-with-default-arch=<bool>',        nargs.ArgBool(None, 1, 'Allow using the last configured arch without setting PETSC_ARCH'))
4257cb31baSSatish Balay    help.addArgument('PETSc','-with-single-library=<bool>',       nargs.ArgBool(None, 1,'Put all PETSc code into the single -lpetsc library'))
43525d6f2eSBarry Smith    help.addArgument('PETSc', '-with-ios=<bool>',              nargs.ArgBool(None, 0, 'Build an iPhone/iPad version of PETSc library'))
448fd71741SJason Sarich    help.addArgument('PETSc', '-with-xsdk-defaults', nargs.ArgBool(None, 0, 'Set the following as defaults for the xSDK standard: --enable-debug=1, --enable-shared=1, --with-precision=double, --with-index-size=32, locate blas/lapack automatically'))
45752d89a4SSatish Balay    help.addArgument('PETSc', '-known-has-attribute-aligned=<bool>',nargs.ArgBool(None, None, 'Indicates __attribute((aligned(16)) directive works (the usual test will be skipped)'))
46*27b0f280SBarry Smith    help.addArgument('PETSc','-with-viewfromoptions=<bool>',      nargs.ArgBool(None, 1,'Support XXXSetFromOptions() calls, for calls with many small solvers turn this off'))
47752d89a4SSatish Balay
48f8833479SBarry Smith    return
49f8833479SBarry Smith
506dd73af6SBarry Smith  def registerPythonFile(self,filename,directory):
516dd73af6SBarry Smith    ''' Add a python file to the framework and registers its headerprefix, ... externalpackagedir
526dd73af6SBarry Smith        directory is the directory where the file relative to the BuildSystem or config path in python notation with . '''
536dd73af6SBarry Smith    (utilityName, ext) = os.path.splitext(filename)
546dd73af6SBarry Smith    if not utilityName.startswith('.') and not utilityName.startswith('#') and ext == '.py' and not utilityName == '__init__':
556dd73af6SBarry Smith      if directory: directory = directory+'.'
566dd73af6SBarry Smith      utilityObj                             = self.framework.require(directory+utilityName, self)
576dd73af6SBarry Smith      utilityObj.headerPrefix                = self.headerPrefix
586dd73af6SBarry Smith      utilityObj.archProvider                = self.arch
596dd73af6SBarry Smith      utilityObj.languageProvider            = self.languages
606dd73af6SBarry Smith      utilityObj.installDirProvider          = self.installdir
616dd73af6SBarry Smith      utilityObj.externalPackagesDirProvider = self.externalpackagesdir
626dd73af6SBarry Smith      utilityObj.precisionProvider           = self.scalartypes
636dd73af6SBarry Smith      utilityObj.indexProvider               = self.indexTypes
646dd73af6SBarry Smith      setattr(self, utilityName.lower(), utilityObj)
656dd73af6SBarry Smith
66f8833479SBarry Smith  def setupDependencies(self, framework):
67f8833479SBarry Smith    config.base.Configure.setupDependencies(self, framework)
68dca78d2bSSatish Balay    self.programs      = framework.require('config.programs',           self)
69f8833479SBarry Smith    self.setCompilers  = framework.require('config.setCompilers',       self)
7030b8aa07SMatthew G. Knepley    self.compilers     = framework.require('config.compilers',          self)
719d310bb7SBarry Smith    self.arch          = framework.require('PETSc.options.arch',        self.setCompilers)
729d310bb7SBarry Smith    self.petscdir      = framework.require('PETSc.options.petscdir',    self.arch)
739d310bb7SBarry Smith    self.installdir    = framework.require('PETSc.options.installDir',  self)
746dd73af6SBarry Smith    self.scalartypes   = framework.require('PETSc.options.scalarTypes', self)
756dd73af6SBarry Smith    self.indexTypes    = framework.require('PETSc.options.indexTypes',  self)
769d310bb7SBarry Smith    self.languages     = framework.require('PETSc.options.languages',   self.setCompilers)
775aab0b90SMatthew G. Knepley    self.debugging     = framework.require('PETSc.options.debugging',   self.compilers)
7830b8aa07SMatthew G. Knepley    self.indexTypes    = framework.require('PETSc.options.indexTypes',  self.compilers)
79f8833479SBarry Smith    self.compilers     = framework.require('config.compilers',          self)
80f8833479SBarry Smith    self.types         = framework.require('config.types',              self)
81f8833479SBarry Smith    self.headers       = framework.require('config.headers',            self)
82f8833479SBarry Smith    self.functions     = framework.require('config.functions',          self)
83f8833479SBarry Smith    self.libraries     = framework.require('config.libraries',          self)
84cd37d877SShri Abhyankar    self.atomics       = framework.require('config.atomics',            self)
859481793eSSatish Balay    self.make          = framework.require('config.packages.make',      self)
869552296fSBarry Smith    self.blasLapack    = framework.require('config.packages.BlasLapack',self)
8706e08bc7SBarry Smith    self.cmake         = framework.require('config.packages.cmake',self)
889d310bb7SBarry Smith    self.externalpackagesdir = framework.require('PETSc.options.externalpackagesdir',self)
89e6b0c433SBarry Smith    self.mpi           = framework.require('config.packages.MPI',self)
9049d43ecaSSatish Balay
919d310bb7SBarry Smith    for utility in os.listdir(os.path.join('config','PETSc','options')):
926dd73af6SBarry Smith      self.registerPythonFile(utility,'PETSc.options')
939d310bb7SBarry Smith
949d310bb7SBarry Smith    for utility in os.listdir(os.path.join('config','BuildSystem','config','utilities')):
956dd73af6SBarry Smith      self.registerPythonFile(utility,'config.utilities')
9606e08bc7SBarry Smith
9747c09f67SMatthew G. Knepley    for package in os.listdir(os.path.join('config', 'BuildSystem', 'config', 'packages')):
986dd73af6SBarry Smith      self.registerPythonFile(package,'config.packages')
996dd73af6SBarry Smith
1005faf1eacSMatthew G. Knepley    # Force blaslapack and opencl to depend on scalarType so precision is set before BlasLapack is built
1019d310bb7SBarry Smith    framework.require('PETSc.options.scalarTypes', self.f2cblaslapack)
1029d310bb7SBarry Smith    framework.require('PETSc.options.scalarTypes', self.fblaslapack)
1039d310bb7SBarry Smith    framework.require('PETSc.options.scalarTypes', self.blaslapack)
1045faf1eacSMatthew G. Knepley    framework.require('PETSc.options.scalarTypes', self.opencl)
1059d310bb7SBarry Smith    framework.require('PETSc.Regression', self)
106f8833479SBarry Smith
107dca78d2bSSatish Balay    self.programs.headerPrefix   = self.headerPrefix
108f8833479SBarry Smith    self.compilers.headerPrefix  = self.headerPrefix
109f8833479SBarry Smith    self.types.headerPrefix      = self.headerPrefix
110f8833479SBarry Smith    self.headers.headerPrefix    = self.headerPrefix
111f8833479SBarry Smith    self.functions.headerPrefix  = self.headerPrefix
112f8833479SBarry Smith    self.libraries.headerPrefix  = self.headerPrefix
1136dd73af6SBarry Smith
1146dd73af6SBarry Smith    # Look for any user provided --download-xxx=directory packages
1156dd73af6SBarry Smith    for arg in sys.argv:
1166dd73af6SBarry Smith      if arg.startswith('--download-') and arg.find('=') > -1:
1176dd73af6SBarry Smith        pname = arg[11:arg.find('=')]
1186dd73af6SBarry Smith        if not hasattr(self,pname):
1196dd73af6SBarry Smith          dname = os.path.dirname(arg[arg.find('=')+1:])
120ab079e5dSBarry Smith          if os.path.isdir(dname) and not os.path.isfile(os.path.join(dname,pname+'.py')):
121ab079e5dSBarry Smith            self.framework.logPrint('User is registering a new package: '+arg)
1226dd73af6SBarry Smith            sys.path.append(dname)
1236dd73af6SBarry Smith            self.registerPythonFile(pname+'.py','')
1246dd73af6SBarry Smith
1256dd73af6SBarry Smith    # test for a variety of basic headers and functions
126a8b45ee7SBarry Smith    headersC = map(lambda name: name+'.h', ['setjmp','dos', 'endian', 'fcntl', 'float', 'io', 'limits', 'malloc', 'pwd', 'search', 'strings',
127ba61063dSBarry Smith                                            'unistd', 'sys/sysinfo', 'machine/endian', 'sys/param', 'sys/procfs', 'sys/resource',
128a3aaec0aSJed Brown                                            'sys/systeminfo', 'sys/times', 'sys/utsname','string', 'stdlib',
129f8833479SBarry Smith                                            'sys/socket','sys/wait','netinet/in','netdb','Direct','time','Ws2tcpip','sys/types',
130e4773d96SSatish Balay                                            'WindowsX', 'cxxabi','float','ieeefp','stdint','sched','pthread','mathimf'])
13145082d64SJed Brown    functions = ['access', '_access', 'clock', 'drand48', 'getcwd', '_getcwd', 'getdomainname', 'gethostname',
132f8833479SBarry Smith                 'gettimeofday', 'getwd', 'memalign', 'memmove', 'mkstemp', 'popen', 'PXFGETARG', 'rand', 'getpagesize',
13338ecfe64SSatish Balay                 'readlink', 'realpath',  'sigaction', 'signal', 'sigset', 'usleep', 'sleep', '_sleep', 'socket',
134473bb0d5SSatish Balay                 'times', 'gethostbyname', 'uname','snprintf','_snprintf','lseek','_lseek','time','fork','stricmp',
135ac7218bbSSatish Balay                 'strcasecmp', 'bzero', 'dlopen', 'dlsym', 'dlclose', 'dlerror','get_nprocs','sysctlbyname',
1360787ed6cSSatish Balay                 '_set_output_format','_mkdir']
137f8833479SBarry Smith    libraries1 = [(['socket', 'nsl'], 'socket'), (['fpe'], 'handle_sigfpes')]
138f8833479SBarry Smith    self.headers.headers.extend(headersC)
139f8833479SBarry Smith    self.functions.functions.extend(functions)
140f8833479SBarry Smith    self.libraries.libraries.extend(libraries1)
1417d421530SBarry Smith
142f8833479SBarry Smith    return
143f8833479SBarry Smith
144262119f8SBarry Smith  def DumpPkgconfig(self):
145262119f8SBarry Smith    ''' Create a pkg-config file '''
146262119f8SBarry Smith    if not os.path.exists(os.path.join(self.petscdir.dir,self.arch.arch,'lib','pkgconfig')):
147262119f8SBarry Smith      os.makedirs(os.path.join(self.petscdir.dir,self.arch.arch,'lib','pkgconfig'))
148262119f8SBarry Smith    fd = open(os.path.join(self.petscdir.dir,self.arch.arch,'lib','pkgconfig','PETSc.pc'),'w')
149262119f8SBarry Smith    if self.framework.argDB['prefix']:
1505bb5b263SMatthew G. Knepley      fd.write('prefix='+self.installdir.dir+'\n')
151262119f8SBarry Smith      fd.write('exec_prefix=${prefix}\n')
152262119f8SBarry Smith      fd.write('includedir=${prefix}/include\n')
153262119f8SBarry Smith    else:
154262119f8SBarry Smith      fd.write('prefix='+self.petscdir.dir+'\n')
155262119f8SBarry Smith      fd.write('exec_prefix=${prefix}\n')
156262119f8SBarry Smith      fd.write('includedir=${prefix}/include\n')
1575bb5b263SMatthew G. Knepley    fd.write('libdir='+os.path.join(self.installdir.dir,'lib')+'\n')
158262119f8SBarry Smith
159262119f8SBarry Smith    self.setCompilers.pushLanguage('C')
160262119f8SBarry Smith    fd.write('ccompiler='+self.setCompilers.getCompiler()+'\n')
161262119f8SBarry Smith    self.setCompilers.popLanguage()
162262119f8SBarry Smith    if hasattr(self.compilers, 'C++'):
163262119f8SBarry Smith      self.setCompilers.pushLanguage('C++')
164262119f8SBarry Smith      fd.write('cxxcompiler='+self.setCompilers.getCompiler()+'\n')
165262119f8SBarry Smith      self.setCompilers.popLanguage()
166262119f8SBarry Smith    if hasattr(self.compilers, 'FC'):
167262119f8SBarry Smith      self.setCompilers.pushLanguage('FC')
168262119f8SBarry Smith      fd.write('fcompiler='+self.setCompilers.getCompiler()+'\n')
169262119f8SBarry Smith      self.setCompilers.popLanguage()
1709552296fSBarry Smith    fd.write('blaslapacklibs='+self.libraries.toStringNoDupes(self.blaslapack.lib)+'\n')
171262119f8SBarry Smith
172262119f8SBarry Smith    fd.write('\n')
173262119f8SBarry Smith    fd.write('Name: PETSc\n')
174262119f8SBarry Smith    fd.write('Description: Library to solve ODEs and algebraic equations\n')
175351d3a41SMatthew G Knepley    fd.write('Version: %s\n' % self.petscdir.version)
176262119f8SBarry Smith
177262119f8SBarry Smith    fd.write('Cflags: '+self.allincludes+'\n')
178262119f8SBarry Smith
179473a3ab2SBarry Smith    plibs = self.libraries.toStringNoDupes(['-L'+os.path.join(self.petscdir.dir,self.arch.arch,'lib'),' -lpetsc'])
180262119f8SBarry Smith    if self.framework.argDB['prefix']:
181c032e545SMatthew G. Knepley      fd.write('Libs: '+plibs.replace(os.path.join(self.petscdir.dir,self.arch.arch),self.installdir.dir)+'\n')
182262119f8SBarry Smith    else:
183473a3ab2SBarry Smith      fd.write('Libs: '+plibs+'\n')
1847ef6e71eSSatish Balay    fd.write('Libs.private: '+self.libraries.toStringNoDupes(self.packagelibs+self.libraries.math+self.compilers.flibs+self.compilers.cxxlibs)+' '+self.compilers.LIBS+'\n')
185473a3ab2SBarry Smith
186262119f8SBarry Smith    fd.close()
187262119f8SBarry Smith    return
188262119f8SBarry Smith
189351d3a41SMatthew G Knepley  def DumpModule(self):
190351d3a41SMatthew G Knepley    ''' Create a module file '''
191af0996ceSBarry Smith    if not os.path.exists(os.path.join(self.petscdir.dir,self.arch.arch,'lib','petsc','conf','modules')):
192af0996ceSBarry Smith      os.makedirs(os.path.join(self.petscdir.dir,self.arch.arch,'lib','petsc','conf','modules'))
193af0996ceSBarry Smith    if not os.path.exists(os.path.join(self.petscdir.dir,self.arch.arch,'lib','petsc','conf','modules','petsc')):
194af0996ceSBarry Smith      os.makedirs(os.path.join(self.petscdir.dir,self.arch.arch,'lib','petsc','conf','modules','petsc'))
195351d3a41SMatthew G Knepley    if self.framework.argDB['prefix']:
1965bb5b263SMatthew G. Knepley      installdir  = self.installdir.dir
19755d606a3SSatish Balay      installarch = ''
19855d606a3SSatish Balay      installpath = os.path.join(installdir,'bin')
199351d3a41SMatthew G Knepley    else:
200351d3a41SMatthew G Knepley      installdir  = self.petscdir.dir
20155d606a3SSatish Balay      installarch = self.arch.arch
20255d606a3SSatish Balay      installpath = os.path.join(installdir,installarch,'bin')+':'+os.path.join(installdir,'bin')
203af0996ceSBarry Smith    fd = open(os.path.join(self.petscdir.dir,self.arch.arch,'lib','petsc','conf','modules','petsc',self.petscdir.version),'w')
204351d3a41SMatthew G Knepley    fd.write('''\
205351d3a41SMatthew G Knepley#%%Module
206351d3a41SMatthew G Knepley
207351d3a41SMatthew G Knepleyproc ModulesHelp { } {
208351d3a41SMatthew G Knepley    puts stderr "This module sets the path and environment variables for petsc-%s"
209351d3a41SMatthew G Knepley    puts stderr "     see http://www.mcs.anl.gov/petsc/ for more information      "
210351d3a41SMatthew G Knepley    puts stderr ""
211351d3a41SMatthew G Knepley}
212351d3a41SMatthew G Knepleymodule-whatis "PETSc - Portable, Extensible Toolkit for Scientific Computation"
213351d3a41SMatthew G Knepley
214351d3a41SMatthew G Knepleyset petsc_dir   %s
215351d3a41SMatthew G Knepleyset petsc_arch  %s
216351d3a41SMatthew G Knepley
217351d3a41SMatthew G Knepleysetenv PETSC_ARCH $petsc_arch
218351d3a41SMatthew G Knepleysetenv PETSC_DIR $petsc_dir
21955d606a3SSatish Balayprepend-path PATH %s
22055d606a3SSatish Balay''' % (self.petscdir.version, installdir, installarch, installpath))
221351d3a41SMatthew G Knepley    fd.close()
222351d3a41SMatthew G Knepley    return
223351d3a41SMatthew G Knepley
224f8833479SBarry Smith  def Dump(self):
225f8833479SBarry Smith    ''' Actually put the values into the configuration files '''
226f8833479SBarry Smith    # eventually everything between -- should be gone
22717f368bcSBarry Smith    if self.mpi.usingMPIUni:
22817f368bcSBarry Smith      #
22917f368bcSBarry Smith      # Remove any MPI/MPICH include files that may have been put here by previous runs of ./configure
2307908f030SMatthew G. Knepley      self.executeShellCommand('rm -rf  '+os.path.join(self.petscdir.dir,self.arch.arch,'include','mpi*')+' '+os.path.join(self.petscdir.dir,self.arch.arch,'include','opa*'), log = self.log)
23117f368bcSBarry Smith
232f8833479SBarry Smith#-----------------------------------------------------------------------------------------------------
233f8833479SBarry Smith
234f8833479SBarry Smith    # Sometimes we need C compiler, even if built with C++
235f8833479SBarry Smith    self.setCompilers.pushLanguage('C')
236f8833479SBarry Smith    self.addMakeMacro('CC_FLAGS',self.setCompilers.getCompilerFlags())
237f8833479SBarry Smith    self.setCompilers.popLanguage()
238f8833479SBarry Smith
23934f774f6SJed Brown    # And sometimes we need a C++ compiler even when PETSc is built with C
24034f774f6SJed Brown    if hasattr(self.compilers, 'CXX'):
24134f774f6SJed Brown      self.setCompilers.pushLanguage('Cxx')
24234f774f6SJed Brown      self.addMakeMacro('CXX_FLAGS',self.setCompilers.getCompilerFlags())
24334f774f6SJed Brown      self.setCompilers.popLanguage()
24434f774f6SJed Brown
245f8833479SBarry Smith    # C preprocessor values
2461315f054SBarry Smith    self.addMakeMacro('CPP_FLAGS',self.setCompilers.CPPFLAGS)
247f8833479SBarry Smith
248f8833479SBarry Smith    # compiler values
249f8833479SBarry Smith    self.setCompilers.pushLanguage(self.languages.clanguage)
250f8833479SBarry Smith    self.addMakeMacro('PCC',self.setCompilers.getCompiler())
251f8833479SBarry Smith    self.addMakeMacro('PCC_FLAGS',self.setCompilers.getCompilerFlags())
252f8833479SBarry Smith    self.setCompilers.popLanguage()
253f8833479SBarry Smith    # .o or .obj
254f8833479SBarry Smith    self.addMakeMacro('CC_SUFFIX','o')
255f8833479SBarry Smith
256f8833479SBarry Smith    # executable linker values
257f8833479SBarry Smith    self.setCompilers.pushLanguage(self.languages.clanguage)
258f8833479SBarry Smith    pcc_linker = self.setCompilers.getLinker()
259f8833479SBarry Smith    self.addMakeMacro('PCC_LINKER',pcc_linker)
260c84a332bSSatish Balay    self.addMakeMacro('PCC_LINKER_FLAGS',self.setCompilers.getLinkerFlags())
261f8833479SBarry Smith    self.setCompilers.popLanguage()
262f8833479SBarry Smith    # '' for Unix, .exe for Windows
263f8833479SBarry Smith    self.addMakeMacro('CC_LINKER_SUFFIX','')
264f8833479SBarry Smith
265f8833479SBarry Smith    if hasattr(self.compilers, 'FC'):
266f8833479SBarry Smith      self.setCompilers.pushLanguage('FC')
267f8833479SBarry Smith      # need FPPFLAGS in config/setCompilers
268f8833479SBarry Smith      self.addDefine('HAVE_FORTRAN','1')
269f8833479SBarry Smith      self.addMakeMacro('FPP_FLAGS',self.setCompilers.CPPFLAGS)
270f8833479SBarry Smith
271f8833479SBarry Smith      # compiler values
272f8833479SBarry Smith      self.addMakeMacro('FC_FLAGS',self.setCompilers.getCompilerFlags())
273f8833479SBarry Smith      self.setCompilers.popLanguage()
274f8833479SBarry Smith      # .o or .obj
275f8833479SBarry Smith      self.addMakeMacro('FC_SUFFIX','o')
276f8833479SBarry Smith
277f8833479SBarry Smith      # executable linker values
278f8833479SBarry Smith      self.setCompilers.pushLanguage('FC')
279f8833479SBarry Smith      # Cannot have NAG f90 as the linker - so use pcc_linker as fc_linker
280f8833479SBarry Smith      fc_linker = self.setCompilers.getLinker()
2817fca349cSMatthew G. Knepley      if config.setCompilers.Configure.isNAG(fc_linker, self.log):
282f8833479SBarry Smith        self.addMakeMacro('FC_LINKER',pcc_linker)
283f8833479SBarry Smith      else:
284f8833479SBarry Smith        self.addMakeMacro('FC_LINKER',fc_linker)
2856d53d35eSSatish Balay      self.addMakeMacro('FC_LINKER_FLAGS',self.setCompilers.getLinkerFlags())
2863feacd00SBarry Smith      # apple requires this shared library linker flag on SOME versions of the os
2873feacd00SBarry Smith      if self.setCompilers.getLinkerFlags().find('-Wl,-commons,use_dylibs') > -1:
2883feacd00SBarry Smith        self.addMakeMacro('DARWIN_COMMONS_USE_DYLIBS',' -Wl,-commons,use_dylibs ')
289bb82cf9cSSatish Balay      self.setCompilers.popLanguage()
2905d631499SMatthew Knepley
2915d631499SMatthew Knepley      # F90 Modules
2925d631499SMatthew Knepley      if self.setCompilers.fortranModuleIncludeFlag:
2935d631499SMatthew Knepley        self.addMakeMacro('FC_MODULE_FLAG', self.setCompilers.fortranModuleIncludeFlag)
2946ddd6694SSatish Balay      else: # for non-f90 compilers like g77
2956ddd6694SSatish Balay        self.addMakeMacro('FC_MODULE_FLAG', '-I')
296a324c51cSMatthew G Knepley      if self.setCompilers.fortranModuleIncludeFlag:
297a324c51cSMatthew G Knepley        self.addMakeMacro('FC_MODULE_OUTPUT_FLAG', self.setCompilers.fortranModuleOutputFlag)
298f8833479SBarry Smith    else:
299f8833479SBarry Smith      self.addMakeMacro('FC','')
300f8833479SBarry Smith
30146a3958fSBarry Smith    if hasattr(self.compilers, 'CUDAC'):
3027ff2890cSSatish Balay      self.setCompilers.pushLanguage('CUDA')
303d93a25ecSSatish Balay      self.addMakeMacro('CUDAC_FLAGS',self.setCompilers.getCompilerFlags())
3047ff2890cSSatish Balay      self.setCompilers.popLanguage()
3057ff2890cSSatish Balay
306f8833479SBarry Smith    # shared library linker values
307f8833479SBarry Smith    self.setCompilers.pushLanguage(self.languages.clanguage)
308f8833479SBarry Smith    # need to fix BuildSystem to collect these separately
309f8833479SBarry Smith    self.addMakeMacro('SL_LINKER',self.setCompilers.getLinker())
31070db8aa6SSatish Balay    self.addMakeMacro('SL_LINKER_FLAGS','${PCC_LINKER_FLAGS}')
311f8833479SBarry Smith    self.setCompilers.popLanguage()
312f8833479SBarry Smith    # One of 'a', 'so', 'lib', 'dll', 'dylib' (perhaps others also?) depending on the library generator and architecture
313f8833479SBarry Smith    # Note: . is not included in this macro, consistent with AR_LIB_SUFFIX
314f8833479SBarry Smith    if self.setCompilers.sharedLibraryExt == self.setCompilers.AR_LIB_SUFFIX:
315f8833479SBarry Smith      self.addMakeMacro('SL_LINKER_SUFFIX', '')
31646bc77b6SBarry Smith      self.addDefine('SLSUFFIX','""')
317f8833479SBarry Smith    else:
318f8833479SBarry Smith      self.addMakeMacro('SL_LINKER_SUFFIX', self.setCompilers.sharedLibraryExt)
31946bc77b6SBarry Smith      self.addDefine('SLSUFFIX','"'+self.setCompilers.sharedLibraryExt+'"')
320bb82cf9cSSatish Balay
32123e93537SBarry Smith    self.addMakeMacro('SL_LINKER_LIBS','${PETSC_EXTERNAL_LIB_BASIC}')
322bb82cf9cSSatish Balay
323f8833479SBarry Smith#-----------------------------------------------------------------------------------------------------
324f8833479SBarry Smith
325f8833479SBarry Smith    # CONLY or CPP. We should change the PETSc makefiles to do this better
326f8833479SBarry Smith    if self.languages.clanguage == 'C': lang = 'CONLY'
327f8833479SBarry Smith    else: lang = 'CXXONLY'
328f8833479SBarry Smith    self.addMakeMacro('PETSC_LANGUAGE',lang)
329f8833479SBarry Smith
330f8833479SBarry Smith    # real or complex
331f8833479SBarry Smith    self.addMakeMacro('PETSC_SCALAR',self.scalartypes.scalartype)
332f8833479SBarry Smith    # double or float
333f8833479SBarry Smith    self.addMakeMacro('PETSC_PRECISION',self.scalartypes.precision)
334f8833479SBarry Smith
335f8833479SBarry Smith    if self.framework.argDB['with-batch']:
336f8833479SBarry Smith      self.addMakeMacro('PETSC_WITH_BATCH','1')
337f8833479SBarry Smith
338f8833479SBarry Smith    # Test for compiler-specific macros that need to be defined.
3397fca349cSMatthew G. Knepley    if self.setCompilers.isCrayVector('CC', self.log):
340b409243cSBarry Smith      self.addDefine('HAVE_CRAY_VECTOR','1')
341f8833479SBarry Smith
342f8833479SBarry Smith#-----------------------------------------------------------------------------------------------------
343df1a78b3SMatthew G Knepley    if self.functions.haveFunction('gethostbyname') and self.functions.haveFunction('socket') and self.headers.haveHeader('netinet/in.h'):
344f8833479SBarry Smith      self.addDefine('USE_SOCKET_VIEWER','1')
34580e3883bSBarry Smith      if self.checkCompile('#include <sys/socket.h>','setsockopt(0,SOL_SOCKET,SO_REUSEADDR,0,0)'):
34680e3883bSBarry Smith        self.addDefine('HAVE_SO_REUSEADDR','1')
347f8833479SBarry Smith
348f8833479SBarry Smith#-----------------------------------------------------------------------------------------------------
349a6cc6bb1SBarry Smith    # print include and lib for makefiles
350f8833479SBarry Smith    self.framework.packages.reverse()
351a6cc6bb1SBarry Smith    includes = [os.path.join(self.petscdir.dir,'include'),os.path.join(self.petscdir.dir,self.arch.arch,'include')]
352996b3231SBarry Smith    libs = []
353f8833479SBarry Smith    for i in self.framework.packages:
354898a086dSBarry Smith      if i.useddirectly:
355eeb16384SBarry Smith        self.addDefine('HAVE_'+i.PACKAGE.replace('-','_'), 1)  # ONLY list package if it is used directly by PETSc (and not only by another package)
356f8833479SBarry Smith      if not isinstance(i.lib, list):
357f8833479SBarry Smith        i.lib = [i.lib]
3584d02c0d4SBarry Smith      if i.linkedbypetsc: libs.extend(i.lib)
359eeb16384SBarry Smith      self.addMakeMacro(i.PACKAGE.replace('-','_')+'_LIB', self.libraries.toStringNoDupes(i.lib))
360f8833479SBarry Smith      if hasattr(i,'include'):
361f8833479SBarry Smith        if not isinstance(i.include,list):
362f8833479SBarry Smith          i.include = [i.include]
363ac9e4c42SSatish Balay        includes.extend(i.include)
364eeb16384SBarry Smith        self.addMakeMacro(i.PACKAGE.replace('-','_')+'_INCLUDE',self.headers.toStringNoDupes(i.include))
365473a3ab2SBarry Smith    self.packagelibs = libs
3662df986feSBarry Smith    if self.framework.argDB['with-single-library']:
3671315f054SBarry Smith      self.alllibs = self.libraries.toStringNoDupes(['-L'+os.path.join(self.petscdir.dir,self.arch.arch,'lib'),' -lpetsc']+libs+self.libraries.math+self.compilers.flibs+self.compilers.cxxlibs)+' '+self.compilers.LIBS
368262119f8SBarry Smith      self.addMakeMacro('PETSC_WITH_EXTERNAL_LIB',self.alllibs)
36991bb3077SSatish Balay    else:
3701315f054SBarry Smith      self.alllibs = self.libraries.toStringNoDupes(['-L'+os.path.join(self.petscdir.dir,self.arch.arch,'lib'),'-lpetscts -lpetscsnes -lpetscksp -lpetscdm -lpetscmat -lpetscvec -lpetscsys']+libs+self.libraries.math+self.compilers.flibs+self.compilers.cxxlibs)+' '+self.compilers.LIBS
3711315f054SBarry Smith    self.PETSC_EXTERNAL_LIB_BASIC = self.libraries.toStringNoDupes(libs+self.libraries.math+self.compilers.flibs+self.compilers.cxxlibs)+' '+self.compilers.LIBS
3721026b6b4SSatish Balay    if self.framework.argDB['prefix'] and self.setCompilers.CSharedLinkerFlag not in ['-L']:
3735bb5b263SMatthew G. Knepley      lib_basic = self.PETSC_EXTERNAL_LIB_BASIC.replace(self.setCompilers.CSharedLinkerFlag+os.path.join(self.petscdir.dir,self.arch.arch,'lib'),self.setCompilers.CSharedLinkerFlag+os.path.join(self.installdir.dir,'lib'))
3741026b6b4SSatish Balay    else:
3751026b6b4SSatish Balay      lib_basic = self.PETSC_EXTERNAL_LIB_BASIC
3761026b6b4SSatish Balay    self.addMakeMacro('PETSC_EXTERNAL_LIB_BASIC',lib_basic)
377262119f8SBarry Smith    self.allincludes = self.headers.toStringNoDupes(includes)
378262119f8SBarry Smith    self.addMakeMacro('PETSC_CC_INCLUDES',self.allincludes)
379262119f8SBarry Smith    self.PETSC_CC_INCLUDES = self.allincludes
380cbd5cc15SBarry Smith    if hasattr(self.compilers, 'FC'):
381208c3fd5SBarry Smith      if self.compilers.fortranIsF90:
38243a63bfbSSatish Balay        self.addMakeMacro('PETSC_FC_INCLUDES',self.headers.toStringNoDupes(includes,includes))
38330d43657SSatish Balay      else:
38430d43657SSatish Balay        self.addMakeMacro('PETSC_FC_INCLUDES',self.headers.toStringNoDupes(includes))
385f8833479SBarry Smith
3865bb5b263SMatthew G. Knepley    self.addMakeMacro('DESTDIR',self.installdir.dir)
3875bb5b263SMatthew G. Knepley    self.addDefine('LIB_DIR','"'+os.path.join(self.installdir.dir,'lib')+'"')
388f8833479SBarry Smith
3890f3b21c2SBarry Smith    if self.framework.argDB['with-single-library']:
3900f3b21c2SBarry Smith      # overrides the values set in conf/variables
3910f3b21c2SBarry Smith      self.addMakeMacro('LIBNAME','${INSTALL_LIB_DIR}/libpetsc.${AR_LIB_SUFFIX}')
39257cb31baSSatish Balay      self.addMakeMacro('SHLIBS','libpetsc')
393bccf1c12SBarry Smith      self.addMakeMacro('PETSC_LIB_BASIC','-lpetsc')
394797063a9SSatish Balay      self.addMakeMacro('PETSC_KSP_LIB_BASIC','-lpetsc')
395797063a9SSatish Balay      self.addMakeMacro('PETSC_TS_LIB_BASIC','-lpetsc')
396b0a7d7e7SSatish Balay      self.addMakeMacro('PETSC_TAO_LIB_BASIC','-lpetsc')
397bb84e0fdSBarry Smith      self.addDefine('USE_SINGLE_LIBRARY', '1')
3982df986feSBarry Smith      if self.sharedlibraries.useShared:
399ea820d49SSatish Balay        self.addMakeMacro('PETSC_SYS_LIB','${C_SH_LIB_PATH} ${PETSC_WITH_EXTERNAL_LIB}')
400ea820d49SSatish Balay        self.addMakeMacro('PETSC_VEC_LIB','${C_SH_LIB_PATH} ${PETSC_WITH_EXTERNAL_LIB}')
401ea820d49SSatish Balay        self.addMakeMacro('PETSC_MAT_LIB','${C_SH_LIB_PATH} ${PETSC_WITH_EXTERNAL_LIB}')
402ea820d49SSatish Balay        self.addMakeMacro('PETSC_DM_LIB','${C_SH_LIB_PATH} ${PETSC_WITH_EXTERNAL_LIB}')
403ea820d49SSatish Balay        self.addMakeMacro('PETSC_KSP_LIB','${C_SH_LIB_PATH} ${PETSC_WITH_EXTERNAL_LIB}')
404ea820d49SSatish Balay        self.addMakeMacro('PETSC_SNES_LIB','${C_SH_LIB_PATH} ${PETSC_WITH_EXTERNAL_LIB}')
405ea820d49SSatish Balay        self.addMakeMacro('PETSC_TS_LIB','${C_SH_LIB_PATH} ${PETSC_WITH_EXTERNAL_LIB}')
406b0a7d7e7SSatish Balay        self.addMakeMacro('PETSC_TAO_LIB','${C_SH_LIB_PATH} ${PETSC_WITH_EXTERNAL_LIB}')
407fdb87e33SJed Brown        self.addMakeMacro('PETSC_CHARACTERISTIC_LIB','${C_SH_LIB_PATH} ${PETSC_WITH_EXTERNAL_LIB}')
408ea820d49SSatish Balay        self.addMakeMacro('PETSC_LIB','${C_SH_LIB_PATH} ${PETSC_WITH_EXTERNAL_LIB}')
409ea820d49SSatish Balay        self.addMakeMacro('PETSC_CONTRIB','${C_SH_LIB_PATH} ${PETSC_WITH_EXTERNAL_LIB}')
4102df986feSBarry Smith      else:
411ea820d49SSatish Balay        self.addMakeMacro('PETSC_SYS_LIB','${PETSC_WITH_EXTERNAL_LIB}')
412ea820d49SSatish Balay        self.addMakeMacro('PETSC_VEC_LIB','${PETSC_WITH_EXTERNAL_LIB}')
413ea820d49SSatish Balay        self.addMakeMacro('PETSC_MAT_LIB','${PETSC_WITH_EXTERNAL_LIB}')
414ea820d49SSatish Balay        self.addMakeMacro('PETSC_DM_LIB','${PETSC_WITH_EXTERNAL_LIB}')
415ea820d49SSatish Balay        self.addMakeMacro('PETSC_KSP_LIB','${PETSC_WITH_EXTERNAL_LIB}')
416ea820d49SSatish Balay        self.addMakeMacro('PETSC_SNES_LIB','${PETSC_WITH_EXTERNAL_LIB}')
417ea820d49SSatish Balay        self.addMakeMacro('PETSC_TS_LIB','${PETSC_WITH_EXTERNAL_LIB}')
418b0a7d7e7SSatish Balay        self.addMakeMacro('PETSC_TAO_LIB','${PETSC_WITH_EXTERNAL_LIB}')
419fdb87e33SJed Brown        self.addMakeMacro('PETSC_CHARACTERISTIC_LIB','${PETSC_WITH_EXTERNAL_LIB}')
420ea820d49SSatish Balay        self.addMakeMacro('PETSC_LIB','${PETSC_WITH_EXTERNAL_LIB}')
421ea820d49SSatish Balay        self.addMakeMacro('PETSC_CONTRIB','${PETSC_WITH_EXTERNAL_LIB}')
4220f3b21c2SBarry Smith
423f8833479SBarry Smith    if not os.path.exists(os.path.join(self.petscdir.dir,self.arch.arch,'lib')):
424f8833479SBarry Smith      os.makedirs(os.path.join(self.petscdir.dir,self.arch.arch,'lib'))
425f8833479SBarry Smith
426f8833479SBarry Smith    # add a makefile entry for configure options
427f8833479SBarry Smith    self.addMakeMacro('CONFIGURE_OPTIONS', self.framework.getOptionsString(['configModules', 'optionsModule']).replace('\"','\\"'))
428f8833479SBarry Smith    return
429f8833479SBarry Smith
430f8833479SBarry Smith  def dumpConfigInfo(self):
431f8833479SBarry Smith    import time
432f8833479SBarry Smith    fd = file(os.path.join(self.arch.arch,'include','petscconfiginfo.h'),'w')
433f8833479SBarry Smith    fd.write('static const char *petscconfigureoptions = "'+self.framework.getOptionsString(['configModules', 'optionsModule']).replace('\"','\\"')+'";\n')
434f8833479SBarry Smith    fd.close()
435f8833479SBarry Smith    return
436f8833479SBarry Smith
4372a4161d9SMatthew G Knepley  def dumpMachineInfo(self):
4382a4161d9SMatthew G Knepley    import platform
4392a4161d9SMatthew G Knepley    import time
44040373944SSatish Balay    import script
441ca77dbeeSGeoffrey Irving    def escape(s):
442ca77dbeeSGeoffrey Irving      return s.replace('"',r'\"').replace(r'\ ',r'\\ ')
4432a4161d9SMatthew G Knepley    fd = file(os.path.join(self.arch.arch,'include','petscmachineinfo.h'),'w')
4442a4161d9SMatthew G Knepley    fd.write('static const char *petscmachineinfo = \"\\n\"\n')
4452a4161d9SMatthew G Knepley    fd.write('\"-----------------------------------------\\n\"\n')
4462a4161d9SMatthew G Knepley    fd.write('\"Libraries compiled on %s on %s \\n\"\n' % (time.ctime(time.time()), platform.node()))
44760acdfe7SSatish Balay    fd.write('\"Machine characteristics: %s\\n\"\n' % (platform.platform()))
448ca77dbeeSGeoffrey Irving    fd.write('\"Using PETSc directory: %s\\n\"\n' % (escape(self.petscdir.dir)))
449ca77dbeeSGeoffrey Irving    fd.write('\"Using PETSc arch: %s\\n\"\n' % (escape(self.arch.arch)))
450cdec380aSBarry Smith    fd.write('\"-----------------------------------------\\n\";\n')
4512a4161d9SMatthew G Knepley    fd.write('static const char *petsccompilerinfo = \"\\n\"\n')
4522a4161d9SMatthew G Knepley    self.setCompilers.pushLanguage(self.languages.clanguage)
453ca77dbeeSGeoffrey Irving    fd.write('\"Using C compiler: %s %s ${COPTFLAGS} ${CFLAGS}\\n\"\n' % (escape(self.setCompilers.getCompiler()), escape(self.setCompilers.getCompilerFlags())))
4542a4161d9SMatthew G Knepley    self.setCompilers.popLanguage()
4558782282cSMatthew G Knepley    if hasattr(self.compilers, 'FC'):
4562a4161d9SMatthew G Knepley      self.setCompilers.pushLanguage('FC')
457ca77dbeeSGeoffrey Irving      fd.write('\"Using Fortran compiler: %s %s ${FOPTFLAGS} ${FFLAGS} %s\\n\"\n' % (escape(self.setCompilers.getCompiler()), escape(self.setCompilers.getCompilerFlags()), escape(self.setCompilers.CPPFLAGS)))
4582a4161d9SMatthew G Knepley      self.setCompilers.popLanguage()
459cdec380aSBarry Smith    fd.write('\"-----------------------------------------\\n\";\n')
4602a4161d9SMatthew G Knepley    fd.write('static const char *petsccompilerflagsinfo = \"\\n\"\n')
461ca77dbeeSGeoffrey Irving    fd.write('\"Using include paths: %s %s %s\\n\"\n' % ('-I'+escape(os.path.join(self.petscdir.dir, self.arch.arch, 'include')), '-I'+escape(os.path.join(self.petscdir.dir, 'include')), escape(self.PETSC_CC_INCLUDES)))
462cdec380aSBarry Smith    fd.write('\"-----------------------------------------\\n\";\n')
4632a4161d9SMatthew G Knepley    fd.write('static const char *petsclinkerinfo = \"\\n\"\n')
4642a4161d9SMatthew G Knepley    self.setCompilers.pushLanguage(self.languages.clanguage)
465ca77dbeeSGeoffrey Irving    fd.write('\"Using C linker: %s\\n\"\n' % (escape(self.setCompilers.getLinker())))
4662a4161d9SMatthew G Knepley    self.setCompilers.popLanguage()
4678782282cSMatthew G Knepley    if hasattr(self.compilers, 'FC'):
4682a4161d9SMatthew G Knepley      self.setCompilers.pushLanguage('FC')
469ca77dbeeSGeoffrey Irving      fd.write('\"Using Fortran linker: %s\\n\"\n' % (escape(self.setCompilers.getLinker())))
4702a4161d9SMatthew G Knepley      self.setCompilers.popLanguage()
471ad782ac6SSatish Balay    if self.framework.argDB['with-single-library']:
472ad782ac6SSatish Balay      petsclib = '-lpetsc'
473ad782ac6SSatish Balay    else:
474ad782ac6SSatish Balay      petsclib = '-lpetscts -lpetscsnes -lpetscksp -lpetscdm -lpetscmat -lpetscvec -lpetscsys'
475ca77dbeeSGeoffrey Irving    fd.write('\"Using libraries: %s%s -L%s %s %s\\n\"\n' % (escape(self.setCompilers.CSharedLinkerFlag), escape(os.path.join(self.petscdir.dir, self.arch.arch, 'lib')), escape(os.path.join(self.petscdir.dir, self.arch.arch, 'lib')), escape(petsclib), escape(self.PETSC_EXTERNAL_LIB_BASIC)))
476cdec380aSBarry Smith    fd.write('\"-----------------------------------------\\n\";\n')
4772a4161d9SMatthew G Knepley    fd.close()
4782a4161d9SMatthew G Knepley    return
479b2843cf1SBarry Smith
480511a6afcSJed Brown  def dumpCMakeConfig(self):
481511a6afcSJed Brown    '''
482724dfae7SSatish Balay    Writes configuration-specific values to ${PETSC_ARCH}/lib/petsc/conf/PETScBuildInternal.cmake.
483511a6afcSJed Brown    This file is private to PETSc and should not be included by third parties
484511a6afcSJed Brown    (a suitable file can be produced later by CMake, but this is not it).
485511a6afcSJed Brown    '''
486511a6afcSJed Brown    def cmakeset(fd,key,val=True):
487511a6afcSJed Brown      if val == True: val = 'YES'
488511a6afcSJed Brown      if val == False: val = 'NO'
489511a6afcSJed Brown      fd.write('set (' + key + ' ' + val + ')\n')
490511a6afcSJed Brown    def ensurelist(a):
491826d9344SJed Brown      if isinstance(a,list):
492826d9344SJed Brown        return a
493826d9344SJed Brown      else:
494826d9344SJed Brown        return [a]
495511a6afcSJed Brown    def libpath(lib):
496511a6afcSJed Brown      'Returns a search path if that is what this item provides, else "" which will be cleaned out later'
4971b1c0b30SJed Brown      if not isinstance(lib,str): return ''
498511a6afcSJed Brown      if lib.startswith('-L'): return lib[2:]
499511a6afcSJed Brown      if lib.startswith('-R'): return lib[2:]
500511a6afcSJed Brown      if lib.startswith('-Wl,-rpath,'):
501511a6afcSJed Brown        # This case occurs when an external package needs a specific system library that is normally provided by the compiler.
502511a6afcSJed Brown        # In other words, the -L path is builtin to the wrapper or compiler, here we provide it so that CMake can locate the
503511a6afcSJed Brown        # corresponding library.
504511a6afcSJed Brown        return lib[len('-Wl,-rpath,'):]
505511a6afcSJed Brown      if lib.startswith('-'): return ''
506511a6afcSJed Brown      return os.path.dirname(lib)
507511a6afcSJed Brown    def cleanlib(lib):
508511a6afcSJed Brown      'Returns a library name if that is what this item provides, else "" which will be cleaned out later'
50942e8629dSMatthew G Knepley      if not isinstance(lib,str): return ''
510511a6afcSJed Brown      if lib.startswith('-l'):  return lib[2:]
511511a6afcSJed Brown      if lib.startswith('-Wl') or lib.startswith('-L'): return ''
512511a6afcSJed Brown      lib = os.path.splitext(os.path.basename(lib))[0]
513511a6afcSJed Brown      if lib.startswith('lib'): return lib[3:]
514511a6afcSJed Brown      return lib
515511a6afcSJed Brown    def nub(lst):
51606e8c1ddSJed Brown      'Return a list containing the first occurrence of each unique element'
517511a6afcSJed Brown      unique = []
518511a6afcSJed Brown      for elem in lst:
519511a6afcSJed Brown        if elem not in unique and elem != '':
520511a6afcSJed Brown          unique.append(elem)
521511a6afcSJed Brown      return unique
52250937898SJed Brown    try: reversed # reversed was added in Python-2.4
52350937898SJed Brown    except NameError:
52450937898SJed Brown      def reversed(lst): return lst[::-1]
52506e8c1ddSJed Brown    def nublast(lst):
52606e8c1ddSJed Brown      'Return a list containing the last occurrence of each unique entry in a list'
52750937898SJed Brown      return reversed(nub(reversed(lst)))
528511a6afcSJed Brown    def cmakeexpand(varname):
529511a6afcSJed Brown      return r'"${' + varname + r'}"'
530582751aaSJed Brown    def uniqextend(lst,new):
531511a6afcSJed Brown      for x in ensurelist(new):
532582751aaSJed Brown        if x not in lst:
533582751aaSJed Brown          lst.append(x)
534511a6afcSJed Brown    def notstandardinclude(path):
535040257f2SJed Brown      return path not in '/usr/include'.split() # /usr/local/include is not automatically included on FreeBSD
536511a6afcSJed Brown    def writeMacroDefinitions(fd):
537511a6afcSJed Brown      if self.mpi.usingMPIUni:
538511a6afcSJed Brown        cmakeset(fd,'PETSC_HAVE_MPIUNI')
539511a6afcSJed Brown      for pkg in self.framework.packages:
540511a6afcSJed Brown        if pkg.useddirectly:
541eeb16384SBarry Smith          cmakeset(fd,'PETSC_HAVE_' + pkg.PACKAGE.replace('-','_'))
542a23e9343SMatthew G Knepley        for pair in pkg.defines.items():
543440af75fSJed Brown          if pair[0].startswith('HAVE_') and pair[1]:
544a23e9343SMatthew G Knepley            cmakeset(fd, self.framework.getFullDefineName(pkg, pair[0]), pair[1])
545511a6afcSJed Brown      for name,val in self.functions.defines.items():
546511a6afcSJed Brown        cmakeset(fd,'PETSC_'+name,val)
547511a6afcSJed Brown      for dct in [self.defines, self.libraryoptions.defines]:
548511a6afcSJed Brown        for k,v in dct.items():
549511a6afcSJed Brown          if k.startswith('USE_'):
550511a6afcSJed Brown            cmakeset(fd,'PETSC_' + k, v)
551511a6afcSJed Brown      cmakeset(fd,'PETSC_USE_COMPLEX', self.scalartypes.scalartype == 'complex')
552ce63c4c1SBarry Smith      cmakeset(fd,'PETSC_USE_REAL_' + self.scalartypes.precision.upper())
553511a6afcSJed Brown      cmakeset(fd,'PETSC_CLANGUAGE_'+self.languages.clanguage)
554511a6afcSJed Brown      if hasattr(self.compilers, 'FC'):
555511a6afcSJed Brown        cmakeset(fd,'PETSC_HAVE_FORTRAN')
556511a6afcSJed Brown        if self.compilers.fortranIsF90:
557511a6afcSJed Brown          cmakeset(fd,'PETSC_USING_F90')
558876d5c60SBarry Smith        if self.compilers.fortranIsF2003:
559876d5c60SBarry Smith          cmakeset(fd,'PETSC_USING_F2003')
56013c0a95cSJed Brown      if hasattr(self.compilers, 'CXX'):
56113c0a95cSJed Brown        cmakeset(fd,'PETSC_HAVE_CXX')
562511a6afcSJed Brown      if self.sharedlibraries.useShared:
563511a6afcSJed Brown        cmakeset(fd,'BUILD_SHARED_LIBS')
564511a6afcSJed Brown    def writeBuildFlags(fd):
56506e8c1ddSJed Brown      def extendby(lib):
56606e8c1ddSJed Brown        libs = ensurelist(lib)
56706e8c1ddSJed Brown        lib_paths.extend(map(libpath,libs))
56806e8c1ddSJed Brown        lib_libs.extend(map(cleanlib,libs))
569511a6afcSJed Brown      lib_paths = []
570511a6afcSJed Brown      lib_libs  = []
571511a6afcSJed Brown      includes  = []
572511a6afcSJed Brown      libvars   = []
573511a6afcSJed Brown      for pkg in self.framework.packages:
5744d02c0d4SBarry Smith        if pkg.linkedbypetsc:
57506e8c1ddSJed Brown          extendby(pkg.lib)
576040257f2SJed Brown          uniqextend(includes,pkg.include)
57706e8c1ddSJed Brown      extendby(self.libraries.math)
57806e8c1ddSJed Brown      extendby(self.libraries.rt)
57906e8c1ddSJed Brown      extendby(self.compilers.flibs)
58006e8c1ddSJed Brown      extendby(self.compilers.cxxlibs)
58106e8c1ddSJed Brown      extendby(self.compilers.LIBS.split())
58206e8c1ddSJed Brown      for libname in nublast(lib_libs):
583511a6afcSJed Brown        libvar = 'PETSC_' + libname.upper() + '_LIB'
5844c0032a9SSatish Balay        addpath = ''
58506e8c1ddSJed Brown        for lpath in nublast(lib_paths):
5864c0032a9SSatish Balay          addpath += '"' + str(lpath) + '" '
5874c0032a9SSatish Balay        fd.write('find_library (' + libvar + ' ' + libname + ' HINTS ' + addpath + ')\n')
588511a6afcSJed Brown        libvars.append(libvar)
589511a6afcSJed Brown      fd.write('mark_as_advanced (' + ' '.join(libvars) + ')\n')
590511a6afcSJed Brown      fd.write('set (PETSC_PACKAGE_LIBS ' + ' '.join(map(cmakeexpand,libvars)) + ')\n')
591040257f2SJed Brown      includes = filter(notstandardinclude,includes)
592040257f2SJed Brown      fd.write('set (PETSC_PACKAGE_INCLUDES ' + ' '.join(map(lambda i: '"'+i+'"',includes)) + ')\n')
593724dfae7SSatish Balay    fd = open(os.path.join(self.arch.arch,'lib','petsc','conf','PETScBuildInternal.cmake'), 'w')
594511a6afcSJed Brown    writeMacroDefinitions(fd)
595511a6afcSJed Brown    writeBuildFlags(fd)
596511a6afcSJed Brown    fd.close()
597511a6afcSJed Brown    return
598511a6afcSJed Brown
5998b0282a9SJed Brown  def dumpCMakeLists(self):
6008b0282a9SJed Brown    import sys
601994b4dadSSatish Balay    if sys.version_info >= (2,4):
6028b0282a9SJed Brown      import cmakegen
6038b0282a9SJed Brown      try:
604a98e69d2SJed Brown        cmakegen.main(self.petscdir.dir, log=self.framework.log)
6058b0282a9SJed Brown      except (OSError), e:
6068b0282a9SJed Brown        self.framework.logPrint('Generating CMakeLists.txt failed:\n' + str(e))
607aac20692SSatish Balay    else:
608aac20692SSatish Balay      self.framework.logPrint('Skipping cmakegen due to old python version: ' +str(sys.version_info) )
6098b0282a9SJed Brown
6108b0282a9SJed Brown  def cmakeBoot(self):
6118b0282a9SJed Brown    import sys
612ae937f1dSJed Brown    self.cmakeboot_success = False
613994b4dadSSatish Balay    if sys.version_info >= (2,4) and hasattr(self.cmake,'cmake'):
6145a4feeedSSatish Balay      oldRead = self.argDB.readonly
6155a4feeedSSatish Balay      self.argDB.readonly = True
616356464bcSMatthew G Knepley      try:
6178b0282a9SJed Brown        import cmakeboot
618ae937f1dSJed Brown        self.cmakeboot_success = cmakeboot.main(petscdir=self.petscdir.dir,petscarch=self.arch.arch,argDB=self.argDB,framework=self.framework,log=self.framework.log)
6198b0282a9SJed Brown      except (OSError), e:
6208b0282a9SJed Brown        self.framework.logPrint('Booting CMake in PETSC_ARCH failed:\n' + str(e))
621356464bcSMatthew G Knepley      except (ImportError, KeyError), e:
622356464bcSMatthew G Knepley        self.framework.logPrint('Importing cmakeboot failed:\n' + str(e))
6235a4feeedSSatish Balay      self.argDB.readonly = oldRead
6249b12c9c7SJed Brown      if self.cmakeboot_success:
6252f730bc2SKarl Rupp        if hasattr(self.compilers, 'FC') and self.compilers.fortranIsF90 and not self.setCompilers.fortranModuleOutputFlag:
62691f9b906SSatish Balay          self.framework.logPrint('CMake configured successfully, but could not be used by default because of missing fortranModuleOutputFlag\n')
6279b12c9c7SJed Brown        else:
6289b12c9c7SJed Brown          self.framework.logPrint('CMake configured successfully, using as default build\n')
629f7b66a64SJed Brown          self.addMakeMacro('PETSC_BUILD_USING_CMAKE',1)
630aac20692SSatish Balay      else:
6319b12c9c7SJed Brown        self.framework.logPrint('CMake configuration was unsuccessful\n')
6329b12c9c7SJed Brown    else:
633aac20692SSatish Balay      self.framework.logPrint('Skipping cmakeboot due to old python version: ' +str(sys.version_info) )
634356464bcSMatthew G Knepley    return
6358b0282a9SJed Brown
636b2843cf1SBarry Smith  def configurePrefetch(self):
637b2843cf1SBarry Smith    '''Sees if there are any prefetch functions supported'''
6387fca349cSMatthew G. Knepley    if config.setCompilers.Configure.isSolaris(self.log) or self.framework.argDB['with-ios'] or not self.framework.argDB['with-prefetch']:
63993f78423SSatish Balay      self.addDefine('Prefetch(a,b,c)', ' ')
64093f78423SSatish Balay      return
641ec284106SBarry Smith    self.pushLanguage(self.languages.clanguage)
64210699583SJed Brown    if self.checkLink('#include <xmmintrin.h>', 'void *v = 0;_mm_prefetch((const char*)v,_MM_HINT_NTA);\n'):
64350d8bf02SJed Brown      # The Intel Intrinsics manual [1] specifies the prototype
64450d8bf02SJed Brown      #
64550d8bf02SJed Brown      #   void _mm_prefetch(char const *a, int sel);
64650d8bf02SJed Brown      #
64750d8bf02SJed Brown      # but other vendors seem to insist on using subtly different
64850d8bf02SJed Brown      # prototypes, including void* for the pointer, and an enum for
64950d8bf02SJed Brown      # sel.  These are both reasonable changes, but negatively impact
65050d8bf02SJed Brown      # portability.
65150d8bf02SJed Brown      #
65250d8bf02SJed Brown      # [1] http://software.intel.com/file/6373
65350d8bf02SJed Brown      self.addDefine('HAVE_XMMINTRIN_H', 1)
65450d8bf02SJed Brown      self.addDefine('Prefetch(a,b,c)', '_mm_prefetch((const char*)(a),(c))')
65550d8bf02SJed Brown      self.addDefine('PREFETCH_HINT_NTA', '_MM_HINT_NTA')
65650d8bf02SJed Brown      self.addDefine('PREFETCH_HINT_T0',  '_MM_HINT_T0')
65750d8bf02SJed Brown      self.addDefine('PREFETCH_HINT_T1',  '_MM_HINT_T1')
65850d8bf02SJed Brown      self.addDefine('PREFETCH_HINT_T2',  '_MM_HINT_T2')
65950d8bf02SJed Brown    elif self.checkLink('#include <xmmintrin.h>', 'void *v = 0;_mm_prefetch(v,_MM_HINT_NTA);\n'):
66050d8bf02SJed Brown      self.addDefine('HAVE_XMMINTRIN_H', 1)
66150d8bf02SJed Brown      self.addDefine('Prefetch(a,b,c)', '_mm_prefetch((const void*)(a),(c))')
66250d8bf02SJed Brown      self.addDefine('PREFETCH_HINT_NTA', '_MM_HINT_NTA')
66350d8bf02SJed Brown      self.addDefine('PREFETCH_HINT_T0',  '_MM_HINT_T0')
66450d8bf02SJed Brown      self.addDefine('PREFETCH_HINT_T1',  '_MM_HINT_T1')
66550d8bf02SJed Brown      self.addDefine('PREFETCH_HINT_T2',  '_MM_HINT_T2')
66610699583SJed Brown    elif self.checkLink('', 'void *v = 0;__builtin_prefetch(v,0,0);\n'):
66710699583SJed Brown      # From GCC docs: void __builtin_prefetch(const void *addr,int rw,int locality)
66810699583SJed Brown      #
66910699583SJed Brown      #   The value of rw is a compile-time constant one or zero; one
67010699583SJed Brown      #   means that the prefetch is preparing for a write to the memory
67110699583SJed Brown      #   address and zero, the default, means that the prefetch is
67210699583SJed Brown      #   preparing for a read. The value locality must be a compile-time
67310699583SJed Brown      #   constant integer between zero and three. A value of zero means
67410699583SJed Brown      #   that the data has no temporal locality, so it need not be left
67510699583SJed Brown      #   in the cache after the access. A value of three means that the
67610699583SJed Brown      #   data has a high degree of temporal locality and should be left
67710699583SJed Brown      #   in all levels of cache possible. Values of one and two mean,
67810699583SJed Brown      #   respectively, a low or moderate degree of temporal locality.
67910699583SJed Brown      #
68010699583SJed Brown      # Here we adopt Intel's x86/x86-64 naming scheme for the locality
68110699583SJed Brown      # hints.  Using macros for these values in necessary since some
68210699583SJed Brown      # compilers require an enum.
68310699583SJed Brown      self.addDefine('Prefetch(a,b,c)', '__builtin_prefetch((a),(b),(c))')
68410699583SJed Brown      self.addDefine('PREFETCH_HINT_NTA', '0')
68510699583SJed Brown      self.addDefine('PREFETCH_HINT_T0',  '3')
68610699583SJed Brown      self.addDefine('PREFETCH_HINT_T1',  '2')
68710699583SJed Brown      self.addDefine('PREFETCH_HINT_T2',  '1')
688b2843cf1SBarry Smith    else:
689b2843cf1SBarry Smith      self.addDefine('Prefetch(a,b,c)', ' ')
6907d490b44SBarry Smith    self.popLanguage()
691b2843cf1SBarry Smith
69209bc878fSSatish Balay  def configureAtoll(self):
69309bc878fSSatish Balay    '''Checks if atoll exists'''
694436b02dcSSatish Balay    if self.checkLink('#define _POSIX_C_SOURCE 200112L\n#include <stdlib.h>','long v = atoll("25")') or self.checkLink ('#include <stdlib.h>','long v = atoll("25")'):
69509bc878fSSatish Balay       self.addDefine('HAVE_ATOLL', '1')
69609bc878fSSatish Balay
6972400fdedSBarry Smith  def configureUnused(self):
6982400fdedSBarry Smith    '''Sees if __attribute((unused)) is supported'''
6991adaff47SSean Farley    if self.framework.argDB['with-ios']:
7002400fdedSBarry Smith      self.addDefine('UNUSED', ' ')
7012400fdedSBarry Smith      return
7022400fdedSBarry Smith    self.pushLanguage(self.languages.clanguage)
703edf21b64SSatish Balay    if self.checkLink('__attribute((unused)) static int myfunc(__attribute((unused)) void *name){ return 1;}', 'int i = 0;\nint j = myfunc(&i);\ntypedef void* atype;\n__attribute((unused))  atype a;\n'):
7042400fdedSBarry Smith      self.addDefine('UNUSED', '__attribute((unused))')
7052400fdedSBarry Smith    else:
7062400fdedSBarry Smith      self.addDefine('UNUSED', ' ')
7072400fdedSBarry Smith    self.popLanguage()
7082400fdedSBarry Smith
70998ed35c3SBarry Smith  def configureIsatty(self):
71098ed35c3SBarry Smith    '''Check if the Unix C function isatty() works correctly
71198ed35c3SBarry Smith       Actually just assumes it does not work correctly on batch systems'''
71298ed35c3SBarry Smith    if not self.framework.argDB['with-batch']:
71398ed35c3SBarry Smith      self.addDefine('USE_ISATTY',1)
71498ed35c3SBarry Smith
7151ef8df7fSJed Brown  def configureDeprecated(self):
7161ef8df7fSJed Brown    '''Check if __attribute((deprecated)) is supported'''
7171ef8df7fSJed Brown    self.pushLanguage(self.languages.clanguage)
71859a26b54SJed Brown    ## Recent versions of gcc and clang support __attribute((deprecated("string argument"))), which is very useful, but
71959a26b54SJed Brown    ## Intel has conspired to make a supremely environment-sensitive compiler.  The Intel compiler looks at the gcc
72059a26b54SJed Brown    ## executable in the environment to determine the language compatibility that it should attempt to emulate.  Some
72159a26b54SJed Brown    ## important Cray installations have built PETSc using the Intel compiler, but with a newer gcc module loaded (e.g.,
722df3898eeSBarry Smith    ## 4.7).  Thus at PETSc configure time, the Intel compiler decides to support the string argument, but the gcc
72359a26b54SJed Brown    ## found in the default user environment is older and does not support the argument.  If GCC and Intel were cool
72459a26b54SJed Brown    ## like Clang and supported __has_attribute, we could avoid configure tests entirely, but they don't.  And that is
72559a26b54SJed Brown    ## why we can't have nice things.
72659a26b54SJed Brown    #
72759a26b54SJed Brown    # if self.checkCompile("""__attribute((deprecated("Why you shouldn't use myfunc"))) static int myfunc(void) { return 1;}""", ''):
72859a26b54SJed Brown    #   self.addDefine('DEPRECATED(why)', '__attribute((deprecated(why)))')
72959a26b54SJed Brown    if self.checkCompile("""__attribute((deprecated)) static int myfunc(void) { return 1;}""", ''):
73059a26b54SJed Brown      self.addDefine('DEPRECATED(why)', '__attribute((deprecated))')
7311ef8df7fSJed Brown    else:
73247644db9SJed Brown      self.addDefine('DEPRECATED(why)', ' ')
7331ef8df7fSJed Brown    self.popLanguage()
7341ef8df7fSJed Brown
73518f41590SBarry Smith  def configureAlign(self):
73618f41590SBarry Smith    '''Check if __attribute(align) is supported'''
737752d89a4SSatish Balay    filename = 'conftestalign'
738752d89a4SSatish Balay    includes = '''
739752d89a4SSatish Balay#include <sys/types.h>
740752d89a4SSatish Balay#if STDC_HEADERS
741752d89a4SSatish Balay#include <stdlib.h>
742752d89a4SSatish Balay#include <stdio.h>
743752d89a4SSatish Balay#include <stddef.h>
744752d89a4SSatish Balay#endif\n'''
745752d89a4SSatish Balay    body     = '''
746752d89a4SSatish Balaystruct mystruct {int myint;} __attribute((aligned(16)));
747752d89a4SSatish BalayFILE *f = fopen("'''+filename+'''", "w");
748752d89a4SSatish Balayif (!f) exit(1);
749752d89a4SSatish Balayfprintf(f, "%lu\\n", (unsigned long)sizeof(struct mystruct));
750752d89a4SSatish Balay'''
751752d89a4SSatish Balay    if 'known-has-attribute-aligned' in self.argDB:
752752d89a4SSatish Balay      if self.argDB['known-has-attribute-aligned']:
753752d89a4SSatish Balay        size = 16
75418f41590SBarry Smith      else:
755752d89a4SSatish Balay        size = -3
756752d89a4SSatish Balay    elif not self.argDB['with-batch']:
757752d89a4SSatish Balay      self.pushLanguage(self.languages.clanguage)
758752d89a4SSatish Balay      try:
759752d89a4SSatish Balay        if self.checkRun(includes, body) and os.path.exists(filename):
760752d89a4SSatish Balay          f    = file(filename)
761752d89a4SSatish Balay          size = int(f.read())
762752d89a4SSatish Balay          f.close()
763752d89a4SSatish Balay          os.remove(filename)
7640045a809SSatish Balay        else:
7650045a809SSatish Balay          size = -4
766752d89a4SSatish Balay      except:
767752d89a4SSatish Balay        size = -1
768752d89a4SSatish Balay        self.framework.logPrint('Error checking attribute(aligned)')
76918f41590SBarry Smith      self.popLanguage()
770752d89a4SSatish Balay    else:
771752d89a4SSatish Balay      self.framework.addBatchInclude(['#include <stdlib.h>', '#include <stdio.h>', '#include <sys/types.h>','struct mystruct {int myint;} __attribute((aligned(16)));'])
772752d89a4SSatish Balay      self.framework.addBatchBody('fprintf(output, "  \'--known-has-attribute-aligned=%d\',\\n", sizeof(struct mystruct)==16);')
773752d89a4SSatish Balay      size = -2
774752d89a4SSatish Balay    if size == 16:
775752d89a4SSatish Balay      self.addDefine('ATTRIBUTEALIGNED(size)', '__attribute((aligned (size)))')
776752d89a4SSatish Balay      self.addDefine('HAVE_ATTRIBUTEALIGNED', 1)
777752d89a4SSatish Balay    else:
778752d89a4SSatish Balay      self.framework.logPrint('incorrect alignment. Found alignment:'+ str(size))
779752d89a4SSatish Balay      self.addDefine('ATTRIBUTEALIGNED(size)', ' ')
780752d89a4SSatish Balay    return
78118f41590SBarry Smith
7829800092aSJed Brown  def configureExpect(self):
7839800092aSJed Brown    '''Sees if the __builtin_expect directive is supported'''
7849800092aSJed Brown    self.pushLanguage(self.languages.clanguage)
7859800092aSJed Brown    if self.checkLink('', 'if (__builtin_expect(0,1)) return 1;'):
7869800092aSJed Brown      self.addDefine('HAVE_BUILTIN_EXPECT', 1)
7879800092aSJed Brown    self.popLanguage()
7889800092aSJed Brown
78953c77d0aSJed Brown  def configureFunctionName(self):
7901ec50b02SJed Brown    '''Sees if the compiler supports __func__ or a variant.  Falls back
7911ec50b02SJed Brown    on __FUNCT__ which PETSc source defines, but most users do not, thus
7921ec50b02SJed Brown    stack traces through user code are better when the compiler's
7931ec50b02SJed Brown    variant is used.'''
7941ec50b02SJed Brown    def getFunctionName(lang):
7951ec50b02SJed Brown      name = '__FUNCT__'
7961ec50b02SJed Brown      self.pushLanguage(lang)
79753c77d0aSJed Brown      if self.checkLink('', "if (__func__[0] != 'm') return 1;"):
7981ec50b02SJed Brown        name = '__func__'
79953c77d0aSJed Brown      elif self.checkLink('', "if (__FUNCTION__[0] != 'm') return 1;"):
8001ec50b02SJed Brown        name = '__FUNCTION__'
8011ec50b02SJed Brown      self.popLanguage()
8021ec50b02SJed Brown      return name
8031ec50b02SJed Brown    langs = []
804628773c9SSatish Balay
805628773c9SSatish Balay    self.addDefine('FUNCTION_NAME_C', getFunctionName('C'))
806628773c9SSatish Balay    if hasattr(self.compilers, 'CXX'):
807628773c9SSatish Balay      self.addDefine('FUNCTION_NAME_CXX', getFunctionName('Cxx'))
80812607bf0SSatish Balay    else:
80912607bf0SSatish Balay      self.addDefine('FUNCTION_NAME_CXX', '__FUNCT__')
81053c77d0aSJed Brown
811753ebd1dSJed Brown  def configureIntptrt(self):
812753ebd1dSJed Brown    '''Determine what to use for uintptr_t'''
813753ebd1dSJed Brown    def staticAssertSizeMatchesVoidStar(inc,typename):
814753ebd1dSJed Brown      # The declaration is an error if either array size is negative.
815753ebd1dSJed Brown      # It should be okay to use an int that is too large, but it would be very unlikely for this to be the case
816d26187a0SJed Brown      return self.checkCompile(inc, ('#define STATIC_ASSERT(cond) char negative_length_if_false[2*(!!(cond))-1]\n'
817979939cdSSatish Balay                                     + 'STATIC_ASSERT(sizeof(void*) == sizeof(%s));'%typename))
818753ebd1dSJed Brown    self.pushLanguage(self.languages.clanguage)
819753ebd1dSJed Brown    if self.checkCompile('#include <stdint.h>', 'int x; uintptr_t i = (uintptr_t)&x;'):
820753ebd1dSJed Brown      self.addDefine('UINTPTR_T', 'uintptr_t')
821753ebd1dSJed Brown    elif staticAssertSizeMatchesVoidStar('','unsigned long long'):
822753ebd1dSJed Brown      self.addDefine('UINTPTR_T', 'unsigned long long')
823753ebd1dSJed Brown    elif staticAssertSizeMatchesVoidStar('#include <stdlib.h>','size_t') or staticAssertSizeMatchesVoidStar('#include <string.h>', 'size_t'):
824753ebd1dSJed Brown      self.addDefine('UINTPTR_T', 'size_t')
825c82284b1SJed Brown    elif staticAssertSizeMatchesVoidStar('','unsigned long'):
826c82284b1SJed Brown      self.addDefine('UINTPTR_T', 'unsigned long')
8272d1b7972SSatish Balay    elif staticAssertSizeMatchesVoidStar('','unsigned'):
828753ebd1dSJed Brown      self.addDefine('UINTPTR_T', 'unsigned')
829d26187a0SJed Brown    else:
830d26187a0SJed Brown      raise RuntimeError('Could not find any unsigned integer type matching void*')
831753ebd1dSJed Brown    self.popLanguage()
832753ebd1dSJed Brown
833ed938b00SJed Brown  def configureRTLDDefault(self):
834bfef2c86SBarry Smith    if self.checkCompile('#include <dlfcn.h>\n void *ptr =  RTLD_DEFAULT;'):
835bfef2c86SBarry Smith      self.addDefine('RTLD_DEFAULT','1')
836f8833479SBarry Smith    return
837f8833479SBarry Smith
838f8833479SBarry Smith  def configureSolaris(self):
839f8833479SBarry Smith    '''Solaris specific stuff'''
840f8833479SBarry Smith    if os.path.isdir(os.path.join('/usr','ucblib')):
841f8833479SBarry Smith      try:
842f8833479SBarry Smith        flag = getattr(self.setCompilers, self.language[-1]+'SharedLinkerFlag')
843f8833479SBarry Smith      except AttributeError:
844f8833479SBarry Smith        flag = None
845f8833479SBarry Smith      if flag is None:
846f8833479SBarry Smith        self.compilers.LIBS += ' -L/usr/ucblib'
847f8833479SBarry Smith      else:
848f8833479SBarry Smith        self.compilers.LIBS += ' '+flag+'/usr/ucblib'
849f8833479SBarry Smith    return
850f8833479SBarry Smith
851f8833479SBarry Smith  def configureLinux(self):
852f8833479SBarry Smith    '''Linux specific stuff'''
8539f15855cSMatthew G Knepley    # TODO: Test for this by mallocing an odd number of floats and checking the address
854f8833479SBarry Smith    self.addDefine('HAVE_DOUBLE_ALIGN_MALLOC', 1)
855f8833479SBarry Smith    return
856f8833479SBarry Smith
857f8833479SBarry Smith  def configureWin32(self):
858f8833479SBarry Smith    '''Win32 non-cygwin specific stuff'''
859f8833479SBarry Smith    kernel32=0
860f8833479SBarry Smith    if self.libraries.add('Kernel32.lib','GetComputerName',prototype='#include <Windows.h>', call='GetComputerName(NULL,NULL);'):
861f8833479SBarry Smith      self.addDefine('HAVE_WINDOWS_H',1)
862f8833479SBarry Smith      self.addDefine('HAVE_GETCOMPUTERNAME',1)
863f8833479SBarry Smith      kernel32=1
864f8833479SBarry Smith    elif self.libraries.add('kernel32','GetComputerName',prototype='#include <Windows.h>', call='GetComputerName(NULL,NULL);'):
865f8833479SBarry Smith      self.addDefine('HAVE_WINDOWS_H',1)
866f8833479SBarry Smith      self.addDefine('HAVE_GETCOMPUTERNAME',1)
867f8833479SBarry Smith      kernel32=1
868f8833479SBarry Smith    if kernel32:
869eed94e11SSatish Balay      if self.framework.argDB['with-windows-graphics']:
870eed94e11SSatish Balay        self.addDefine('USE_WINDOWS_GRAPHICS',1)
871f8833479SBarry Smith      if self.checkLink('#include <Windows.h>','LoadLibrary(0)'):
872f8833479SBarry Smith        self.addDefine('HAVE_LOADLIBRARY',1)
873b50f6d9eSLisandro Dalcin      if self.checkLink('#include <Windows.h>','GetProcAddress(0,0)'):
874b50f6d9eSLisandro Dalcin        self.addDefine('HAVE_GETPROCADDRESS',1)
875b50f6d9eSLisandro Dalcin      if self.checkLink('#include <Windows.h>','FreeLibrary(0)'):
876b50f6d9eSLisandro Dalcin        self.addDefine('HAVE_FREELIBRARY',1)
877a21658a3SLisandro Dalcin      if self.checkLink('#include <Windows.h>','GetLastError()'):
878a21658a3SLisandro Dalcin        self.addDefine('HAVE_GETLASTERROR',1)
879a21658a3SLisandro Dalcin      if self.checkLink('#include <Windows.h>','SetLastError(0)'):
880a21658a3SLisandro Dalcin        self.addDefine('HAVE_SETLASTERROR',1)
881f8833479SBarry Smith      if self.checkLink('#include <Windows.h>\n','QueryPerformanceCounter(0);\n'):
882bea725cfSBarry Smith        self.addDefine('USE_MICROSOFT_TIME',1)
883f8833479SBarry Smith    if self.libraries.add('Advapi32.lib','GetUserName',prototype='#include <Windows.h>', call='GetUserName(NULL,NULL);'):
884f8833479SBarry Smith      self.addDefine('HAVE_GET_USER_NAME',1)
885f8833479SBarry Smith    elif self.libraries.add('advapi32','GetUserName',prototype='#include <Windows.h>', call='GetUserName(NULL,NULL);'):
886f8833479SBarry Smith      self.addDefine('HAVE_GET_USER_NAME',1)
887f8833479SBarry Smith
888f8833479SBarry Smith    if not self.libraries.add('User32.lib','GetDC',prototype='#include <Windows.h>',call='GetDC(0);'):
889f8833479SBarry Smith      self.libraries.add('user32','GetDC',prototype='#include <Windows.h>',call='GetDC(0);')
890f8833479SBarry Smith    if not self.libraries.add('Gdi32.lib','CreateCompatibleDC',prototype='#include <Windows.h>',call='CreateCompatibleDC(0);'):
891f8833479SBarry Smith      self.libraries.add('gdi32','CreateCompatibleDC',prototype='#include <Windows.h>',call='CreateCompatibleDC(0);')
892f8833479SBarry Smith
893f8833479SBarry Smith    self.types.check('int32_t', 'int')
894f8833479SBarry Smith    if not self.checkCompile('#include <sys/types.h>\n','uid_t u;\n'):
895f8833479SBarry Smith      self.addTypedef('int', 'uid_t')
896f8833479SBarry Smith      self.addTypedef('int', 'gid_t')
897f8833479SBarry Smith    if not self.checkLink('#if defined(PETSC_HAVE_UNISTD_H)\n#include <unistd.h>\n#endif\n','int a=R_OK;\n'):
898f8833479SBarry Smith      self.framework.addDefine('R_OK', '04')
899f8833479SBarry Smith      self.framework.addDefine('W_OK', '02')
900f8833479SBarry Smith      self.framework.addDefine('X_OK', '01')
901f8833479SBarry Smith    if not self.checkLink('#include <sys/stat.h>\n','int a=0;\nif (S_ISDIR(a)){}\n'):
902f8833479SBarry Smith      self.framework.addDefine('S_ISREG(a)', '(((a)&_S_IFMT) == _S_IFREG)')
903f8833479SBarry Smith      self.framework.addDefine('S_ISDIR(a)', '(((a)&_S_IFMT) == _S_IFDIR)')
904f8833479SBarry Smith    if self.checkCompile('#include <Windows.h>\n','LARGE_INTEGER a;\nDWORD b=a.u.HighPart;\n'):
905f8833479SBarry Smith      self.addDefine('HAVE_LARGE_INTEGER_U',1)
906f8833479SBarry Smith
907f8833479SBarry Smith    # Windows requires a Binary file creation flag when creating/opening binary files.  Is a better test in order?
908ef2cfba3SSatish Balay    if self.checkCompile('#include <Windows.h>\n#include <fcntl.h>\n', 'int flags = O_BINARY;'):
909f8833479SBarry Smith      self.addDefine('HAVE_O_BINARY',1)
910f8833479SBarry Smith
911f8833479SBarry Smith    if self.compilers.CC.find('win32fe') >= 0:
912f8833479SBarry Smith      self.addDefine('PATH_SEPARATOR','\';\'')
913f8833479SBarry Smith      self.addDefine('DIR_SEPARATOR','\'\\\\\'')
914f8833479SBarry Smith      self.addDefine('REPLACE_DIR_SEPARATOR','\'/\'')
915f8833479SBarry Smith      self.addDefine('CANNOT_START_DEBUGGER',1)
9167908f030SMatthew G. Knepley      (petscdir,error,status) = self.executeShellCommand('cygpath -w '+self.petscdir.dir, log = self.log)
91734531a4dSSatish Balay      self.addDefine('DIR','"'+petscdir.replace('\\','\\\\')+'"')
918f8833479SBarry Smith    else:
919f8833479SBarry Smith      self.addDefine('PATH_SEPARATOR','\':\'')
920f8833479SBarry Smith      self.addDefine('REPLACE_DIR_SEPARATOR','\'\\\\\'')
921f8833479SBarry Smith      self.addDefine('DIR_SEPARATOR','\'/\'')
92234531a4dSSatish Balay      self.addDefine('DIR', '"'+self.petscdir.dir+'"')
923bfef2c86SBarry Smith
924f8833479SBarry Smith    return
925f8833479SBarry Smith
926f8833479SBarry Smith#-----------------------------------------------------------------------------------------------------
927b10d012aSSatish Balay  def configureCygwinBrokenPipe(self):
928b10d012aSSatish Balay    '''Cygwin version <= 1.7.18 had issues with pipes and long commands invoked from gnu-make
929b10d012aSSatish Balay    http://cygwin.com/ml/cygwin/2013-05/msg00340.html '''
9307fca349cSMatthew G. Knepley    if config.setCompilers.Configure.isCygwin(self.log):
931b10d012aSSatish Balay      import platform
932b10d012aSSatish Balay      import re
933b10d012aSSatish Balay      r=re.compile("([0-9]+).([0-9]+).([0-9]+)")
934b10d012aSSatish Balay      m=r.match(platform.release())
935b10d012aSSatish Balay      major=int(m.group(1))
936b10d012aSSatish Balay      minor=int(m.group(2))
937b10d012aSSatish Balay      subminor=int(m.group(3))
938b10d012aSSatish Balay      if ((major < 1) or (major == 1 and minor < 7) or (major == 1 and minor == 7 and subminor <= 18)):
939b10d012aSSatish Balay        self.addMakeMacro('PETSC_CYGWIN_BROKEN_PIPE','1')
940b10d012aSSatish Balay    return
941b10d012aSSatish Balay
942b10d012aSSatish Balay#-----------------------------------------------------------------------------------------------------
943569865ddSSatish Balay  def configureDefaultArch(self):
944af0996ceSBarry Smith    conffile = os.path.join('lib','petsc','conf', 'petscvariables')
945569865ddSSatish Balay    if self.framework.argDB['with-default-arch']:
946569865ddSSatish Balay      fd = file(conffile, 'w')
947569865ddSSatish Balay      fd.write('PETSC_ARCH='+self.arch.arch+'\n')
948da93591fSBarry Smith      fd.write('PETSC_DIR='+self.petscdir.dir+'\n')
949af0996ceSBarry Smith      fd.write('include '+os.path.join(self.petscdir.dir,self.arch.arch,'lib','petsc','conf','petscvariables')+'\n')
950569865ddSSatish Balay      fd.close()
951569865ddSSatish Balay      self.framework.actions.addArgument('PETSc', 'Build', 'Set default architecture to '+self.arch.arch+' in '+conffile)
952569865ddSSatish Balay    elif os.path.isfile(conffile):
953569865ddSSatish Balay      try:
954569865ddSSatish Balay        os.unlink(conffile)
955569865ddSSatish Balay      except:
956569865ddSSatish Balay        raise RuntimeError('Unable to remove file '+conffile+'. Did a different user create it?')
957569865ddSSatish Balay    return
958569865ddSSatish Balay
959569865ddSSatish Balay#-----------------------------------------------------------------------------------------------------
960f8833479SBarry Smith  def configureScript(self):
961f8833479SBarry Smith    '''Output a script in the conf directory which will reproduce the configuration'''
962f8833479SBarry Smith    import nargs
963495bf1a9SSatish Balay    import sys
964af0996ceSBarry Smith    scriptName = os.path.join(self.arch.arch,'lib','petsc','conf', 'reconfigure-'+self.arch.arch+'.py')
965f8833479SBarry Smith    args = dict([(nargs.Arg.parseArgument(arg)[0], arg) for arg in self.framework.clArgs])
966e97fc2efSSatish Balay    if 'with-clean' in args:
967e97fc2efSSatish Balay      del args['with-clean']
968f8833479SBarry Smith    if 'configModules' in args:
9691063a081SSatish Balay      if nargs.Arg.parseArgument(args['configModules'])[1] == 'PETSc.Configure':
970f8833479SBarry Smith        del args['configModules']
971f8833479SBarry Smith    if 'optionsModule' in args:
97223a19ef1SSatish Balay      if nargs.Arg.parseArgument(args['optionsModule'])[1] == 'config.compilerOptions':
973f8833479SBarry Smith        del args['optionsModule']
974f8833479SBarry Smith    if not 'PETSC_ARCH' in args:
9751063a081SSatish Balay      args['PETSC_ARCH'] = 'PETSC_ARCH='+str(self.arch.arch)
976f8833479SBarry Smith    f = file(scriptName, 'w')
977495bf1a9SSatish Balay    f.write('#!'+sys.executable+'\n')
978f8833479SBarry Smith    f.write('if __name__ == \'__main__\':\n')
979f8833479SBarry Smith    f.write('  import sys\n')
9807561c02cSSatish Balay    f.write('  import os\n')
9817561c02cSSatish Balay    f.write('  sys.path.insert(0, os.path.abspath(\'config\'))\n')
982f8833479SBarry Smith    f.write('  import configure\n')
9831063a081SSatish Balay    # pretty print repr(args.values())
9841063a081SSatish Balay    f.write('  configure_options = [\n')
9858bec23c5SJed Brown    for itm in sorted(args.values()):
9861063a081SSatish Balay      f.write('    \''+str(itm)+'\',\n')
9871063a081SSatish Balay    f.write('  ]\n')
988f8833479SBarry Smith    f.write('  configure.petsc_configure(configure_options)\n')
989f8833479SBarry Smith    f.close()
990f8833479SBarry Smith    try:
991f8833479SBarry Smith      os.chmod(scriptName, 0775)
992f8833479SBarry Smith    except OSError, e:
993f8833479SBarry Smith      self.framework.logPrint('Unable to make reconfigure script executable:\n'+str(e))
994f8833479SBarry Smith    self.framework.actions.addArgument('PETSc', 'File creation', 'Created '+scriptName+' for automatic reconfiguration')
995f8833479SBarry Smith    return
996f8833479SBarry Smith
997f8833479SBarry Smith  def configureInstall(self):
998f8833479SBarry Smith    '''Setup the directories for installation'''
999f8833479SBarry Smith    if self.framework.argDB['prefix']:
1000824e893fSSatish Balay      self.addMakeRule('shared_install','',['-@echo "Now to install the libraries do:"',\
1001d093bd8dSBarry Smith                                              '-@echo "'+self.installdir.installSudo+'make PETSC_DIR=${PETSC_DIR} PETSC_ARCH=${PETSC_ARCH} install"',\
1002315b77e6SSatish Balay                                              '-@echo "========================================="'])
1003f8833479SBarry Smith    else:
1004824e893fSSatish Balay      self.addMakeRule('shared_install','',['-@echo "Now to check if the libraries are working do:"',\
1005824e893fSSatish Balay                                              '-@echo "make PETSC_DIR=${PETSC_DIR} PETSC_ARCH=${PETSC_ARCH} test"',\
1006315b77e6SSatish Balay                                              '-@echo "========================================="'])
1007f8833479SBarry Smith      return
1008f8833479SBarry Smith
1009f8833479SBarry Smith  def configureGCOV(self):
1010f8833479SBarry Smith    if self.framework.argDB['with-gcov']:
1011f8833479SBarry Smith      self.addDefine('USE_GCOV','1')
1012f8833479SBarry Smith    return
1013f8833479SBarry Smith
1014f8833479SBarry Smith  def configureFortranFlush(self):
1015f8833479SBarry Smith    if hasattr(self.compilers, 'FC'):
1016f8833479SBarry Smith      for baseName in ['flush','flush_']:
1017f8833479SBarry Smith        if self.libraries.check('', baseName, otherLibs = self.compilers.flibs, fortranMangle = 1):
1018f8833479SBarry Smith          self.addDefine('HAVE_'+baseName.upper(), 1)
1019f8833479SBarry Smith          return
1020f8833479SBarry Smith
1021*27b0f280SBarry Smith  def configureViewFromOptions(self):
1022*27b0f280SBarry Smith    if not self.framework.argDB['with-viewfromoptions']:
1023*27b0f280SBarry Smith      self.addDefine('SKIP_VIEWFROMOPTIONS',1)
1024*27b0f280SBarry Smith
102528bb2e72SSatish Balay  def postProcessPackages(self):
102628bb2e72SSatish Balay    postPackages=[]
102728bb2e72SSatish Balay    for i in self.framework.packages:
102828bb2e72SSatish Balay      if hasattr(i,'postProcess'): postPackages.append(i)
102928bb2e72SSatish Balay    if postPackages:
1030e64d19dfSSatish Balay      # ctetgen needs petsc conf files. so attempt to create them early
1031a77eb93bSSatish Balay      self.framework.dumpConfFiles()
1032d9293e7bSBarry Smith      # tacky fix for dependency of Aluimia on Pflotran; requested via petsc-dev Matt provide a correct fix
1033d9293e7bSBarry Smith      for i in postPackages:
1034d9293e7bSBarry Smith        if i.name.upper() in ['PFLOTRAN']:
1035d9293e7bSBarry Smith          i.postProcess()
1036d9293e7bSBarry Smith          postPackages.remove(i)
103728bb2e72SSatish Balay      for i in postPackages: i.postProcess()
103828bb2e72SSatish Balay    return
1039f8833479SBarry Smith
1040f8833479SBarry Smith  def configure(self):
1041f8833479SBarry Smith    if not os.path.samefile(self.petscdir.dir, os.getcwd()):
1042f8833479SBarry Smith      raise RuntimeError('Wrong PETSC_DIR option specified: '+str(self.petscdir.dir) + '\n  Configure invoked in: '+os.path.realpath(os.getcwd()))
1043550489e3SMatthew G Knepley    if self.framework.argDB['prefix'] and os.path.isdir(self.framework.argDB['prefix']) and os.path.samefile(self.framework.argDB['prefix'],self.petscdir.dir):
10443552d8fbSSatish Balay      raise RuntimeError('Incorrect option --prefix='+self.framework.argDB['prefix']+' specified. It cannot be same as PETSC_DIR!')
1045c16c35a9SSatish Balay    if self.framework.argDB['prefix'] and os.path.isdir(self.framework.argDB['prefix']) and os.path.samefile(self.framework.argDB['prefix'],os.path.join(self.petscdir.dir,self.arch.arch)):
1046c16c35a9SSatish Balay      raise RuntimeError('Incorrect option --prefix='+self.framework.argDB['prefix']+' specified. It cannot be same as PETSC_DIR/PETSC_ARCH!')
1047f16c1317SJed Brown    self.framework.header          = os.path.join(self.arch.arch,'include','petscconf.h')
1048f16c1317SJed Brown    self.framework.cHeader         = os.path.join(self.arch.arch,'include','petscfix.h')
1049af0996ceSBarry Smith    self.framework.makeMacroHeader = os.path.join(self.arch.arch,'lib','petsc','conf','petscvariables')
1050af0996ceSBarry Smith    self.framework.makeRuleHeader  = os.path.join(self.arch.arch,'lib','petsc','conf','petscrules')
1051f8833479SBarry Smith    if self.libraries.math is None:
1052f8833479SBarry Smith      raise RuntimeError('PETSc requires a functional math library. Please send configure.log to petsc-maint@mcs.anl.gov.')
1053f8833479SBarry Smith    if self.languages.clanguage == 'Cxx' and not hasattr(self.compilers, 'CXX'):
1054f8833479SBarry Smith      raise RuntimeError('Cannot set C language to C++ without a functional C++ compiler.')
1055ed938b00SJed Brown    self.executeTest(self.configureRTLDDefault)
1056b2843cf1SBarry Smith    self.executeTest(self.configurePrefetch)
10572400fdedSBarry Smith    self.executeTest(self.configureUnused)
10581ef8df7fSJed Brown    self.executeTest(self.configureDeprecated)
105998ed35c3SBarry Smith    self.executeTest(self.configureIsatty)
10609800092aSJed Brown    self.executeTest(self.configureExpect);
106118f41590SBarry Smith    self.executeTest(self.configureAlign);
106253c77d0aSJed Brown    self.executeTest(self.configureFunctionName);
1063753ebd1dSJed Brown    self.executeTest(self.configureIntptrt);
1064f8833479SBarry Smith    self.executeTest(self.configureSolaris)
1065f8833479SBarry Smith    self.executeTest(self.configureLinux)
1066f8833479SBarry Smith    self.executeTest(self.configureWin32)
1067b10d012aSSatish Balay    self.executeTest(self.configureCygwinBrokenPipe)
1068569865ddSSatish Balay    self.executeTest(self.configureDefaultArch)
1069f8833479SBarry Smith    self.executeTest(self.configureScript)
1070f8833479SBarry Smith    self.executeTest(self.configureInstall)
1071f8833479SBarry Smith    self.executeTest(self.configureGCOV)
1072f8833479SBarry Smith    self.executeTest(self.configureFortranFlush)
107309bc878fSSatish Balay    self.executeTest(self.configureAtoll)
1074*27b0f280SBarry Smith    self.executeTest(self.configureViewFromOptions)
1075f8833479SBarry Smith    # dummy rules, always needed except for remote builds
1076f8833479SBarry Smith    self.addMakeRule('remote','')
1077f8833479SBarry Smith    self.addMakeRule('remoteclean','')
1078f8833479SBarry Smith
1079f8833479SBarry Smith    self.Dump()
1080f8833479SBarry Smith    self.dumpConfigInfo()
10812a4161d9SMatthew G Knepley    self.dumpMachineInfo()
1082511a6afcSJed Brown    self.dumpCMakeConfig()
10838b0282a9SJed Brown    self.dumpCMakeLists()
108440277576SBarry Smith    # need to save the current state of BuildSystem so that postProcess() packages can read it in and perhaps run make install
108540277576SBarry Smith    self.framework.storeSubstitutions(self.framework.argDB)
108640277576SBarry Smith    self.framework.argDB['configureCache'] = cPickle.dumps(self.framework)
108740277576SBarry Smith    self.framework.argDB.save(force = True)
10888b0282a9SJed Brown    self.cmakeBoot()
1089262119f8SBarry Smith    self.DumpPkgconfig()
1090351d3a41SMatthew G Knepley    self.DumpModule()
1091f7ad81e1SBarry Smith    self.postProcessPackages()
1092f8833479SBarry Smith    self.framework.log.write('================================================================================\n')
1093f8833479SBarry Smith    self.logClear()
1094f8833479SBarry Smith    return
1095