xref: /petsc/config/PETSc/Configure.py (revision c032e545d7d253c98b14f5627d1a560eeed5f90c)
1f8833479SBarry Smithimport config.base
2f8833479SBarry Smith
3f8833479SBarry Smithimport os
4f8833479SBarry Smithimport re
540277576SBarry Smithimport cPickle
6f8833479SBarry Smith
78bec23c5SJed Brown# The sorted() builtin is not available with python-2.3
88bec23c5SJed Browntry: sorted
98bec23c5SJed Brownexcept NameError:
108bec23c5SJed Brown  def sorted(lst):
118bec23c5SJed Brown    lst.sort()
128bec23c5SJed Brown    return lst
138bec23c5SJed Brown
14f8833479SBarry Smithclass Configure(config.base.Configure):
15f8833479SBarry Smith  def __init__(self, framework):
16f8833479SBarry Smith    config.base.Configure.__init__(self, framework)
17f8833479SBarry Smith    self.headerPrefix = 'PETSC'
18f8833479SBarry Smith    self.substPrefix  = 'PETSC'
19f8833479SBarry Smith    return
20f8833479SBarry Smith
217c939e48SSatish Balay  def __str2__(self):
227c939e48SSatish Balay    desc = []
23a0022257SSatish Balay    desc.append('xxx=========================================================================xxx')
24ac1d0f13SJed Brown    if self.make.getMakeMacro('MAKE_IS_GNUMAKE'):
259481793eSSatish Balay      build_type = 'gnumake build'
269481793eSSatish Balay    elif self.getMakeMacro('PETSC_BUILD_USING_CMAKE'):
27b3618d6dSSatish Balay      build_type = 'cmake build'
28b3618d6dSSatish Balay    else:
29b3618d6dSSatish Balay      build_type = 'legacy build'
30b3618d6dSSatish Balay    desc.append(' Configure stage complete. Now build PETSc libraries with (%s):' % build_type)
31b3618d6dSSatish Balay    desc.append('   make PETSC_DIR='+self.petscdir.dir+' PETSC_ARCH='+self.arch.arch+' all')
32a0022257SSatish Balay    desc.append('xxx=========================================================================xxx')
337c939e48SSatish Balay    return '\n'.join(desc)+'\n'
34f8833479SBarry Smith
35f8833479SBarry Smith  def setupHelp(self, help):
36f8833479SBarry Smith    import nargs
37ce0b2093SBarry Smith    help.addArgument('PETSc',  '-prefix=<dir>',                   nargs.Arg(None, '', 'Specifiy location to install PETSc (eg. /usr/local)'))
387deb5ab3SBarry Smith    help.addArgument('PETSc',  '-with-prefetch=<bool>',           nargs.ArgBool(None, 1,'Enable checking for prefetch instructions'))
39eed94e11SSatish Balay    help.addArgument('Windows','-with-windows-graphics=<bool>',   nargs.ArgBool(None, 1,'Enable check for Windows Graphics'))
40569865ddSSatish Balay    help.addArgument('PETSc', '-with-default-arch=<bool>',        nargs.ArgBool(None, 1, 'Allow using the last configured arch without setting PETSC_ARCH'))
4157cb31baSSatish Balay    help.addArgument('PETSc','-with-single-library=<bool>',       nargs.ArgBool(None, 1,'Put all PETSc code into the single -lpetsc library'))
42525d6f2eSBarry Smith    help.addArgument('PETSc', '-with-ios=<bool>',              nargs.ArgBool(None, 0, 'Build an iPhone/iPad version of PETSc library'))
438fd71741SJason 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'))
44752d89a4SSatish 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)'))
45752d89a4SSatish Balay
46f8833479SBarry Smith    return
47f8833479SBarry Smith
48f8833479SBarry Smith  def setupDependencies(self, framework):
49f8833479SBarry Smith    config.base.Configure.setupDependencies(self, framework)
50dca78d2bSSatish Balay    self.programs      = framework.require('config.programs',           self)
51f8833479SBarry Smith    self.setCompilers  = framework.require('config.setCompilers',       self)
5230b8aa07SMatthew G. Knepley    self.compilers     = framework.require('config.compilers',          self)
539d310bb7SBarry Smith    self.arch          = framework.require('PETSc.options.arch',        self.setCompilers)
549d310bb7SBarry Smith    self.petscdir      = framework.require('PETSc.options.petscdir',    self.arch)
559d310bb7SBarry Smith    self.installdir    = framework.require('PETSc.options.installDir',  self)
569d310bb7SBarry Smith    self.languages     = framework.require('PETSc.options.languages',   self.setCompilers)
575aab0b90SMatthew G. Knepley    self.debugging     = framework.require('PETSc.options.debugging',   self.compilers)
5830b8aa07SMatthew G. Knepley    self.indexTypes    = framework.require('PETSc.options.indexTypes',  self.compilers)
59f8833479SBarry Smith    self.compilers     = framework.require('config.compilers',          self)
60f8833479SBarry Smith    self.types         = framework.require('config.types',              self)
61f8833479SBarry Smith    self.headers       = framework.require('config.headers',            self)
62f8833479SBarry Smith    self.functions     = framework.require('config.functions',          self)
63f8833479SBarry Smith    self.libraries     = framework.require('config.libraries',          self)
64cd37d877SShri Abhyankar    self.atomics       = framework.require('config.atomics',            self)
659481793eSSatish Balay    self.make          = framework.require('config.packages.make',      self)
669552296fSBarry Smith    self.blasLapack    = framework.require('config.packages.BlasLapack',self)
6706e08bc7SBarry Smith    self.cmake         = framework.require('config.packages.cmake',self)
689d310bb7SBarry Smith    self.externalpackagesdir = framework.require('PETSc.options.externalpackagesdir',self)
69e6b0c433SBarry Smith    self.mpi           = framework.require('config.packages.MPI',self)
7049d43ecaSSatish Balay
719d310bb7SBarry Smith    for utility in os.listdir(os.path.join('config','PETSc','options')):
72f8833479SBarry Smith      (utilityName, ext) = os.path.splitext(utility)
73f8833479SBarry Smith      if not utilityName.startswith('.') and not utilityName.startswith('#') and ext == '.py' and not utilityName == '__init__':
749d310bb7SBarry Smith        utilityObj                    = self.framework.require('PETSc.options.'+utilityName, self)
759d310bb7SBarry Smith        utilityObj.headerPrefix       = self.headerPrefix
769d310bb7SBarry Smith        utilityObj.archProvider       = self.arch
779d310bb7SBarry Smith        utilityObj.languageProvider   = self.languages
789d310bb7SBarry Smith        utilityObj.installDirProvider = self.installdir
799d310bb7SBarry Smith        utilityObj.externalPackagesDirProvider = self.externalpackagesdir
809d310bb7SBarry Smith        setattr(self, utilityName.lower(), utilityObj)
819d310bb7SBarry Smith
829d310bb7SBarry Smith    for utility in os.listdir(os.path.join('config','BuildSystem','config','utilities')):
839d310bb7SBarry Smith      (utilityName, ext) = os.path.splitext(utility)
849d310bb7SBarry Smith      if not utilityName.startswith('.') and not utilityName.startswith('#') and ext == '.py' and not utilityName == '__init__':
859d310bb7SBarry Smith        utilityObj                    = self.framework.require('config.utilities.'+utilityName, self)
8606e08bc7SBarry Smith        utilityObj.headerPrefix       = self.headerPrefix
8706e08bc7SBarry Smith        utilityObj.archProvider       = self.arch
8806e08bc7SBarry Smith        utilityObj.languageProvider   = self.languages
8906e08bc7SBarry Smith        utilityObj.installDirProvider = self.installdir
9006e08bc7SBarry Smith        utilityObj.externalPackagesDirProvider = self.externalpackagesdir
9106e08bc7SBarry Smith        setattr(self, utilityName.lower(), utilityObj)
9206e08bc7SBarry Smith
9347c09f67SMatthew G. Knepley    if os.path.isdir(os.path.join('config', 'BuildSystem', 'config', 'packages')):
9447c09f67SMatthew G. Knepley      for package in os.listdir(os.path.join('config', 'BuildSystem', 'config', 'packages')):
9547c09f67SMatthew G. Knepley        (packageName, ext) = os.path.splitext(package)
9647c09f67SMatthew G. Knepley        if not packageName.startswith('.') and not packageName.startswith('#') and ext == '.py' and not packageName == '__init__' and not packageName == 'PETSc':
9747c09f67SMatthew G. Knepley          packageObj                    = framework.require('config.packages.'+packageName, self)
985aab0b90SMatthew G. Knepley          packageObj.headerPrefix       = self.headerPrefix
99d37554e4SMatthew G Knepley          packageObj.archProvider       = self.arch
100d37554e4SMatthew G Knepley          packageObj.languageProvider   = self.languages
101339e8cb0SBarry Smith          packageObj.precisionProvider  = self.scalartypes
10230b8aa07SMatthew G. Knepley          packageObj.indexProvider      = self.indexTypes
103edb8ae0cSSatish Balay          packageObj.installDirProvider = self.installdir
10449d43ecaSSatish Balay          packageObj.externalPackagesDirProvider = self.externalpackagesdir
10547c09f67SMatthew G. Knepley          setattr(self, packageName.lower(), packageObj)
1065faf1eacSMatthew G. Knepley    # Force blaslapack and opencl to depend on scalarType so precision is set before BlasLapack is built
1079d310bb7SBarry Smith    framework.require('PETSc.options.scalarTypes', self.f2cblaslapack)
1089d310bb7SBarry Smith    framework.require('PETSc.options.scalarTypes', self.fblaslapack)
1099d310bb7SBarry Smith    framework.require('PETSc.options.scalarTypes', self.blaslapack)
1105faf1eacSMatthew G. Knepley    framework.require('PETSc.options.scalarTypes', self.opencl)
1119d310bb7SBarry Smith    framework.require('PETSc.Regression', self)
112f8833479SBarry Smith
113dca78d2bSSatish Balay    self.programs.headerPrefix   = self.headerPrefix
114f8833479SBarry Smith    self.compilers.headerPrefix  = self.headerPrefix
115f8833479SBarry Smith    self.types.headerPrefix      = self.headerPrefix
116f8833479SBarry Smith    self.headers.headerPrefix    = self.headerPrefix
117f8833479SBarry Smith    self.functions.headerPrefix  = self.headerPrefix
118f8833479SBarry Smith    self.libraries.headerPrefix  = self.headerPrefix
119f8833479SBarry Smith    self.blaslapack.headerPrefix = self.headerPrefix
120f8833479SBarry Smith    self.mpi.headerPrefix        = self.headerPrefix
121a8b45ee7SBarry Smith    headersC = map(lambda name: name+'.h', ['setjmp','dos', 'endian', 'fcntl', 'float', 'io', 'limits', 'malloc', 'pwd', 'search', 'strings',
122ba61063dSBarry Smith                                            'unistd', 'sys/sysinfo', 'machine/endian', 'sys/param', 'sys/procfs', 'sys/resource',
123a3aaec0aSJed Brown                                            'sys/systeminfo', 'sys/times', 'sys/utsname','string', 'stdlib',
124f8833479SBarry Smith                                            'sys/socket','sys/wait','netinet/in','netdb','Direct','time','Ws2tcpip','sys/types',
125e4773d96SSatish Balay                                            'WindowsX', 'cxxabi','float','ieeefp','stdint','sched','pthread','mathimf'])
12645082d64SJed Brown    functions = ['access', '_access', 'clock', 'drand48', 'getcwd', '_getcwd', 'getdomainname', 'gethostname',
127f8833479SBarry Smith                 'gettimeofday', 'getwd', 'memalign', 'memmove', 'mkstemp', 'popen', 'PXFGETARG', 'rand', 'getpagesize',
12838ecfe64SSatish Balay                 'readlink', 'realpath',  'sigaction', 'signal', 'sigset', 'usleep', 'sleep', '_sleep', 'socket',
129473bb0d5SSatish Balay                 'times', 'gethostbyname', 'uname','snprintf','_snprintf','lseek','_lseek','time','fork','stricmp',
130ac7218bbSSatish Balay                 'strcasecmp', 'bzero', 'dlopen', 'dlsym', 'dlclose', 'dlerror','get_nprocs','sysctlbyname',
1310787ed6cSSatish Balay                 '_set_output_format','_mkdir']
132f8833479SBarry Smith    libraries1 = [(['socket', 'nsl'], 'socket'), (['fpe'], 'handle_sigfpes')]
133f8833479SBarry Smith    self.headers.headers.extend(headersC)
134f8833479SBarry Smith    self.functions.functions.extend(functions)
135f8833479SBarry Smith    self.libraries.libraries.extend(libraries1)
1367d421530SBarry Smith
137f8833479SBarry Smith    return
138f8833479SBarry Smith
139262119f8SBarry Smith  def DumpPkgconfig(self):
140262119f8SBarry Smith    ''' Create a pkg-config file '''
141262119f8SBarry Smith    if not os.path.exists(os.path.join(self.petscdir.dir,self.arch.arch,'lib','pkgconfig')):
142262119f8SBarry Smith      os.makedirs(os.path.join(self.petscdir.dir,self.arch.arch,'lib','pkgconfig'))
143262119f8SBarry Smith    fd = open(os.path.join(self.petscdir.dir,self.arch.arch,'lib','pkgconfig','PETSc.pc'),'w')
144262119f8SBarry Smith    if self.framework.argDB['prefix']:
1455bb5b263SMatthew G. Knepley      fd.write('prefix='+self.installdir.dir+'\n')
146262119f8SBarry Smith      fd.write('exec_prefix=${prefix}\n')
147262119f8SBarry Smith      fd.write('includedir=${prefix}/include\n')
148262119f8SBarry Smith    else:
149262119f8SBarry Smith      fd.write('prefix='+self.petscdir.dir+'\n')
150262119f8SBarry Smith      fd.write('exec_prefix=${prefix}\n')
151262119f8SBarry Smith      fd.write('includedir=${prefix}/include\n')
1525bb5b263SMatthew G. Knepley    fd.write('libdir='+os.path.join(self.installdir.dir,'lib')+'\n')
153262119f8SBarry Smith
154262119f8SBarry Smith    self.setCompilers.pushLanguage('C')
155262119f8SBarry Smith    fd.write('ccompiler='+self.setCompilers.getCompiler()+'\n')
156262119f8SBarry Smith    self.setCompilers.popLanguage()
157262119f8SBarry Smith    if hasattr(self.compilers, 'C++'):
158262119f8SBarry Smith      self.setCompilers.pushLanguage('C++')
159262119f8SBarry Smith      fd.write('cxxcompiler='+self.setCompilers.getCompiler()+'\n')
160262119f8SBarry Smith      self.setCompilers.popLanguage()
161262119f8SBarry Smith    if hasattr(self.compilers, 'FC'):
162262119f8SBarry Smith      self.setCompilers.pushLanguage('FC')
163262119f8SBarry Smith      fd.write('fcompiler='+self.setCompilers.getCompiler()+'\n')
164262119f8SBarry Smith      self.setCompilers.popLanguage()
1659552296fSBarry Smith    fd.write('blaslapacklibs='+self.libraries.toStringNoDupes(self.blaslapack.lib)+'\n')
166262119f8SBarry Smith
167262119f8SBarry Smith    fd.write('\n')
168262119f8SBarry Smith    fd.write('Name: PETSc\n')
169262119f8SBarry Smith    fd.write('Description: Library to solve ODEs and algebraic equations\n')
170351d3a41SMatthew G Knepley    fd.write('Version: %s\n' % self.petscdir.version)
171262119f8SBarry Smith
172262119f8SBarry Smith    fd.write('Cflags: '+self.allincludes+'\n')
173262119f8SBarry Smith
174473a3ab2SBarry Smith    plibs = self.libraries.toStringNoDupes(['-L'+os.path.join(self.petscdir.dir,self.arch.arch,'lib'),' -lpetsc'])
175262119f8SBarry Smith    if self.framework.argDB['prefix']:
176*c032e545SMatthew G. Knepley      fd.write('Libs: '+plibs.replace(os.path.join(self.petscdir.dir,self.arch.arch),self.installdir.dir)+'\n')
177262119f8SBarry Smith    else:
178473a3ab2SBarry Smith      fd.write('Libs: '+plibs+'\n')
1797ef6e71eSSatish Balay    fd.write('Libs.private: '+self.libraries.toStringNoDupes(self.packagelibs+self.libraries.math+self.compilers.flibs+self.compilers.cxxlibs)+' '+self.compilers.LIBS+'\n')
180473a3ab2SBarry Smith
181262119f8SBarry Smith    fd.close()
182262119f8SBarry Smith    return
183262119f8SBarry Smith
184351d3a41SMatthew G Knepley  def DumpModule(self):
185351d3a41SMatthew G Knepley    ''' Create a module file '''
186af0996ceSBarry Smith    if not os.path.exists(os.path.join(self.petscdir.dir,self.arch.arch,'lib','petsc','conf','modules')):
187af0996ceSBarry Smith      os.makedirs(os.path.join(self.petscdir.dir,self.arch.arch,'lib','petsc','conf','modules'))
188af0996ceSBarry Smith    if not os.path.exists(os.path.join(self.petscdir.dir,self.arch.arch,'lib','petsc','conf','modules','petsc')):
189af0996ceSBarry Smith      os.makedirs(os.path.join(self.petscdir.dir,self.arch.arch,'lib','petsc','conf','modules','petsc'))
190351d3a41SMatthew G Knepley    if self.framework.argDB['prefix']:
1915bb5b263SMatthew G. Knepley      installdir  = self.installdir.dir
19255d606a3SSatish Balay      installarch = ''
19355d606a3SSatish Balay      installpath = os.path.join(installdir,'bin')
194351d3a41SMatthew G Knepley    else:
195351d3a41SMatthew G Knepley      installdir  = self.petscdir.dir
19655d606a3SSatish Balay      installarch = self.arch.arch
19755d606a3SSatish Balay      installpath = os.path.join(installdir,installarch,'bin')+':'+os.path.join(installdir,'bin')
198af0996ceSBarry Smith    fd = open(os.path.join(self.petscdir.dir,self.arch.arch,'lib','petsc','conf','modules','petsc',self.petscdir.version),'w')
199351d3a41SMatthew G Knepley    fd.write('''\
200351d3a41SMatthew G Knepley#%%Module
201351d3a41SMatthew G Knepley
202351d3a41SMatthew G Knepleyproc ModulesHelp { } {
203351d3a41SMatthew G Knepley    puts stderr "This module sets the path and environment variables for petsc-%s"
204351d3a41SMatthew G Knepley    puts stderr "     see http://www.mcs.anl.gov/petsc/ for more information      "
205351d3a41SMatthew G Knepley    puts stderr ""
206351d3a41SMatthew G Knepley}
207351d3a41SMatthew G Knepleymodule-whatis "PETSc - Portable, Extensible Toolkit for Scientific Computation"
208351d3a41SMatthew G Knepley
209351d3a41SMatthew G Knepleyset petsc_dir   %s
210351d3a41SMatthew G Knepleyset petsc_arch  %s
211351d3a41SMatthew G Knepley
212351d3a41SMatthew G Knepleysetenv PETSC_ARCH $petsc_arch
213351d3a41SMatthew G Knepleysetenv PETSC_DIR $petsc_dir
21455d606a3SSatish Balayprepend-path PATH %s
21555d606a3SSatish Balay''' % (self.petscdir.version, installdir, installarch, installpath))
216351d3a41SMatthew G Knepley    fd.close()
217351d3a41SMatthew G Knepley    return
218351d3a41SMatthew G Knepley
219f8833479SBarry Smith  def Dump(self):
220f8833479SBarry Smith    ''' Actually put the values into the configuration files '''
221f8833479SBarry Smith    # eventually everything between -- should be gone
22217f368bcSBarry Smith    if self.mpi.usingMPIUni:
22317f368bcSBarry Smith      #
22417f368bcSBarry Smith      # Remove any MPI/MPICH include files that may have been put here by previous runs of ./configure
2257908f030SMatthew 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)
22617f368bcSBarry Smith
227f8833479SBarry Smith#-----------------------------------------------------------------------------------------------------
228f8833479SBarry Smith
229f8833479SBarry Smith    # Sometimes we need C compiler, even if built with C++
230f8833479SBarry Smith    self.setCompilers.pushLanguage('C')
231f8833479SBarry Smith    self.addMakeMacro('CC_FLAGS',self.setCompilers.getCompilerFlags())
232f8833479SBarry Smith    self.setCompilers.popLanguage()
233f8833479SBarry Smith
23434f774f6SJed Brown    # And sometimes we need a C++ compiler even when PETSc is built with C
23534f774f6SJed Brown    if hasattr(self.compilers, 'CXX'):
23634f774f6SJed Brown      self.setCompilers.pushLanguage('Cxx')
23734f774f6SJed Brown      self.addMakeMacro('CXX_FLAGS',self.setCompilers.getCompilerFlags())
23834f774f6SJed Brown      self.setCompilers.popLanguage()
23934f774f6SJed Brown
240f8833479SBarry Smith    # C preprocessor values
2411315f054SBarry Smith    self.addMakeMacro('CPP_FLAGS',self.setCompilers.CPPFLAGS)
242f8833479SBarry Smith
243f8833479SBarry Smith    # compiler values
244f8833479SBarry Smith    self.setCompilers.pushLanguage(self.languages.clanguage)
245f8833479SBarry Smith    self.addMakeMacro('PCC',self.setCompilers.getCompiler())
246f8833479SBarry Smith    self.addMakeMacro('PCC_FLAGS',self.setCompilers.getCompilerFlags())
247f8833479SBarry Smith    self.setCompilers.popLanguage()
248f8833479SBarry Smith    # .o or .obj
249f8833479SBarry Smith    self.addMakeMacro('CC_SUFFIX','o')
250f8833479SBarry Smith
251f8833479SBarry Smith    # executable linker values
252f8833479SBarry Smith    self.setCompilers.pushLanguage(self.languages.clanguage)
253f8833479SBarry Smith    pcc_linker = self.setCompilers.getLinker()
254f8833479SBarry Smith    self.addMakeMacro('PCC_LINKER',pcc_linker)
255c84a332bSSatish Balay    self.addMakeMacro('PCC_LINKER_FLAGS',self.setCompilers.getLinkerFlags())
256f8833479SBarry Smith    self.setCompilers.popLanguage()
257f8833479SBarry Smith    # '' for Unix, .exe for Windows
258f8833479SBarry Smith    self.addMakeMacro('CC_LINKER_SUFFIX','')
259f8833479SBarry Smith
260f8833479SBarry Smith    if hasattr(self.compilers, 'FC'):
261f8833479SBarry Smith      self.setCompilers.pushLanguage('FC')
262f8833479SBarry Smith      # need FPPFLAGS in config/setCompilers
263f8833479SBarry Smith      self.addDefine('HAVE_FORTRAN','1')
264f8833479SBarry Smith      self.addMakeMacro('FPP_FLAGS',self.setCompilers.CPPFLAGS)
265f8833479SBarry Smith
266f8833479SBarry Smith      # compiler values
267f8833479SBarry Smith      self.addMakeMacro('FC_FLAGS',self.setCompilers.getCompilerFlags())
268f8833479SBarry Smith      self.setCompilers.popLanguage()
269f8833479SBarry Smith      # .o or .obj
270f8833479SBarry Smith      self.addMakeMacro('FC_SUFFIX','o')
271f8833479SBarry Smith
272f8833479SBarry Smith      # executable linker values
273f8833479SBarry Smith      self.setCompilers.pushLanguage('FC')
274f8833479SBarry Smith      # Cannot have NAG f90 as the linker - so use pcc_linker as fc_linker
275f8833479SBarry Smith      fc_linker = self.setCompilers.getLinker()
2767fca349cSMatthew G. Knepley      if config.setCompilers.Configure.isNAG(fc_linker, self.log):
277f8833479SBarry Smith        self.addMakeMacro('FC_LINKER',pcc_linker)
278f8833479SBarry Smith      else:
279f8833479SBarry Smith        self.addMakeMacro('FC_LINKER',fc_linker)
2806d53d35eSSatish Balay      self.addMakeMacro('FC_LINKER_FLAGS',self.setCompilers.getLinkerFlags())
2813feacd00SBarry Smith      # apple requires this shared library linker flag on SOME versions of the os
2823feacd00SBarry Smith      if self.setCompilers.getLinkerFlags().find('-Wl,-commons,use_dylibs') > -1:
2833feacd00SBarry Smith        self.addMakeMacro('DARWIN_COMMONS_USE_DYLIBS',' -Wl,-commons,use_dylibs ')
284bb82cf9cSSatish Balay      self.setCompilers.popLanguage()
2855d631499SMatthew Knepley
2865d631499SMatthew Knepley      # F90 Modules
2875d631499SMatthew Knepley      if self.setCompilers.fortranModuleIncludeFlag:
2885d631499SMatthew Knepley        self.addMakeMacro('FC_MODULE_FLAG', self.setCompilers.fortranModuleIncludeFlag)
2896ddd6694SSatish Balay      else: # for non-f90 compilers like g77
2906ddd6694SSatish Balay        self.addMakeMacro('FC_MODULE_FLAG', '-I')
291a324c51cSMatthew G Knepley      if self.setCompilers.fortranModuleIncludeFlag:
292a324c51cSMatthew G Knepley        self.addMakeMacro('FC_MODULE_OUTPUT_FLAG', self.setCompilers.fortranModuleOutputFlag)
293f8833479SBarry Smith    else:
294f8833479SBarry Smith      self.addMakeMacro('FC','')
295f8833479SBarry Smith
29646a3958fSBarry Smith    if hasattr(self.compilers, 'CUDAC'):
2977ff2890cSSatish Balay      self.setCompilers.pushLanguage('CUDA')
298d93a25ecSSatish Balay      self.addMakeMacro('CUDAC_FLAGS',self.setCompilers.getCompilerFlags())
2997ff2890cSSatish Balay      self.setCompilers.popLanguage()
3007ff2890cSSatish Balay
301f8833479SBarry Smith    # shared library linker values
302f8833479SBarry Smith    self.setCompilers.pushLanguage(self.languages.clanguage)
303f8833479SBarry Smith    # need to fix BuildSystem to collect these separately
304f8833479SBarry Smith    self.addMakeMacro('SL_LINKER',self.setCompilers.getLinker())
30570db8aa6SSatish Balay    self.addMakeMacro('SL_LINKER_FLAGS','${PCC_LINKER_FLAGS}')
306f8833479SBarry Smith    self.setCompilers.popLanguage()
307f8833479SBarry Smith    # One of 'a', 'so', 'lib', 'dll', 'dylib' (perhaps others also?) depending on the library generator and architecture
308f8833479SBarry Smith    # Note: . is not included in this macro, consistent with AR_LIB_SUFFIX
309f8833479SBarry Smith    if self.setCompilers.sharedLibraryExt == self.setCompilers.AR_LIB_SUFFIX:
310f8833479SBarry Smith      self.addMakeMacro('SL_LINKER_SUFFIX', '')
31146bc77b6SBarry Smith      self.addDefine('SLSUFFIX','""')
312f8833479SBarry Smith    else:
313f8833479SBarry Smith      self.addMakeMacro('SL_LINKER_SUFFIX', self.setCompilers.sharedLibraryExt)
31446bc77b6SBarry Smith      self.addDefine('SLSUFFIX','"'+self.setCompilers.sharedLibraryExt+'"')
315bb82cf9cSSatish Balay
31623e93537SBarry Smith    self.addMakeMacro('SL_LINKER_LIBS','${PETSC_EXTERNAL_LIB_BASIC}')
317bb82cf9cSSatish Balay
318f8833479SBarry Smith#-----------------------------------------------------------------------------------------------------
319f8833479SBarry Smith
320f8833479SBarry Smith    # CONLY or CPP. We should change the PETSc makefiles to do this better
321f8833479SBarry Smith    if self.languages.clanguage == 'C': lang = 'CONLY'
322f8833479SBarry Smith    else: lang = 'CXXONLY'
323f8833479SBarry Smith    self.addMakeMacro('PETSC_LANGUAGE',lang)
324f8833479SBarry Smith
325f8833479SBarry Smith    # real or complex
326f8833479SBarry Smith    self.addMakeMacro('PETSC_SCALAR',self.scalartypes.scalartype)
327f8833479SBarry Smith    # double or float
328f8833479SBarry Smith    self.addMakeMacro('PETSC_PRECISION',self.scalartypes.precision)
329f8833479SBarry Smith
330f8833479SBarry Smith    if self.framework.argDB['with-batch']:
331f8833479SBarry Smith      self.addMakeMacro('PETSC_WITH_BATCH','1')
332f8833479SBarry Smith
333f8833479SBarry Smith    # Test for compiler-specific macros that need to be defined.
3347fca349cSMatthew G. Knepley    if self.setCompilers.isCrayVector('CC', self.log):
335b409243cSBarry Smith      self.addDefine('HAVE_CRAY_VECTOR','1')
336f8833479SBarry Smith
337f8833479SBarry Smith#-----------------------------------------------------------------------------------------------------
338df1a78b3SMatthew G Knepley    if self.functions.haveFunction('gethostbyname') and self.functions.haveFunction('socket') and self.headers.haveHeader('netinet/in.h'):
339f8833479SBarry Smith      self.addDefine('USE_SOCKET_VIEWER','1')
34080e3883bSBarry Smith      if self.checkCompile('#include <sys/socket.h>','setsockopt(0,SOL_SOCKET,SO_REUSEADDR,0,0)'):
34180e3883bSBarry Smith        self.addDefine('HAVE_SO_REUSEADDR','1')
342f8833479SBarry Smith
343f8833479SBarry Smith#-----------------------------------------------------------------------------------------------------
344a6cc6bb1SBarry Smith    # print include and lib for makefiles
345f8833479SBarry Smith    self.framework.packages.reverse()
346a6cc6bb1SBarry Smith    includes = [os.path.join(self.petscdir.dir,'include'),os.path.join(self.petscdir.dir,self.arch.arch,'include')]
347996b3231SBarry Smith    libs = []
348f8833479SBarry Smith    for i in self.framework.packages:
349898a086dSBarry Smith      if i.useddirectly:
350eeb16384SBarry Smith        self.addDefine('HAVE_'+i.PACKAGE.replace('-','_'), 1)  # ONLY list package if it is used directly by PETSc (and not only by another package)
351f8833479SBarry Smith      if not isinstance(i.lib, list):
352f8833479SBarry Smith        i.lib = [i.lib]
3534d02c0d4SBarry Smith      if i.linkedbypetsc: libs.extend(i.lib)
354eeb16384SBarry Smith      self.addMakeMacro(i.PACKAGE.replace('-','_')+'_LIB', self.libraries.toStringNoDupes(i.lib))
355f8833479SBarry Smith      if hasattr(i,'include'):
356f8833479SBarry Smith        if not isinstance(i.include,list):
357f8833479SBarry Smith          i.include = [i.include]
358ac9e4c42SSatish Balay        includes.extend(i.include)
359eeb16384SBarry Smith        self.addMakeMacro(i.PACKAGE.replace('-','_')+'_INCLUDE',self.headers.toStringNoDupes(i.include))
360473a3ab2SBarry Smith    self.packagelibs = libs
3612df986feSBarry Smith    if self.framework.argDB['with-single-library']:
3621315f054SBarry 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
363262119f8SBarry Smith      self.addMakeMacro('PETSC_WITH_EXTERNAL_LIB',self.alllibs)
36491bb3077SSatish Balay    else:
3651315f054SBarry 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
3661315f054SBarry Smith    self.PETSC_EXTERNAL_LIB_BASIC = self.libraries.toStringNoDupes(libs+self.libraries.math+self.compilers.flibs+self.compilers.cxxlibs)+' '+self.compilers.LIBS
3671026b6b4SSatish Balay    if self.framework.argDB['prefix'] and self.setCompilers.CSharedLinkerFlag not in ['-L']:
3685bb5b263SMatthew 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'))
3691026b6b4SSatish Balay    else:
3701026b6b4SSatish Balay      lib_basic = self.PETSC_EXTERNAL_LIB_BASIC
3711026b6b4SSatish Balay    self.addMakeMacro('PETSC_EXTERNAL_LIB_BASIC',lib_basic)
372262119f8SBarry Smith    self.allincludes = self.headers.toStringNoDupes(includes)
373262119f8SBarry Smith    self.addMakeMacro('PETSC_CC_INCLUDES',self.allincludes)
374262119f8SBarry Smith    self.PETSC_CC_INCLUDES = self.allincludes
375cbd5cc15SBarry Smith    if hasattr(self.compilers, 'FC'):
376208c3fd5SBarry Smith      if self.compilers.fortranIsF90:
37743a63bfbSSatish Balay        self.addMakeMacro('PETSC_FC_INCLUDES',self.headers.toStringNoDupes(includes,includes))
37830d43657SSatish Balay      else:
37930d43657SSatish Balay        self.addMakeMacro('PETSC_FC_INCLUDES',self.headers.toStringNoDupes(includes))
380f8833479SBarry Smith
3815bb5b263SMatthew G. Knepley    self.addMakeMacro('DESTDIR',self.installdir.dir)
3825bb5b263SMatthew G. Knepley    self.addDefine('LIB_DIR','"'+os.path.join(self.installdir.dir,'lib')+'"')
383f8833479SBarry Smith
3840f3b21c2SBarry Smith    if self.framework.argDB['with-single-library']:
3850f3b21c2SBarry Smith      # overrides the values set in conf/variables
3860f3b21c2SBarry Smith      self.addMakeMacro('LIBNAME','${INSTALL_LIB_DIR}/libpetsc.${AR_LIB_SUFFIX}')
38757cb31baSSatish Balay      self.addMakeMacro('SHLIBS','libpetsc')
388bccf1c12SBarry Smith      self.addMakeMacro('PETSC_LIB_BASIC','-lpetsc')
389797063a9SSatish Balay      self.addMakeMacro('PETSC_KSP_LIB_BASIC','-lpetsc')
390797063a9SSatish Balay      self.addMakeMacro('PETSC_TS_LIB_BASIC','-lpetsc')
391b0a7d7e7SSatish Balay      self.addMakeMacro('PETSC_TAO_LIB_BASIC','-lpetsc')
392bb84e0fdSBarry Smith      self.addDefine('USE_SINGLE_LIBRARY', '1')
3932df986feSBarry Smith      if self.sharedlibraries.useShared:
394ea820d49SSatish Balay        self.addMakeMacro('PETSC_SYS_LIB','${C_SH_LIB_PATH} ${PETSC_WITH_EXTERNAL_LIB}')
395ea820d49SSatish Balay        self.addMakeMacro('PETSC_VEC_LIB','${C_SH_LIB_PATH} ${PETSC_WITH_EXTERNAL_LIB}')
396ea820d49SSatish Balay        self.addMakeMacro('PETSC_MAT_LIB','${C_SH_LIB_PATH} ${PETSC_WITH_EXTERNAL_LIB}')
397ea820d49SSatish Balay        self.addMakeMacro('PETSC_DM_LIB','${C_SH_LIB_PATH} ${PETSC_WITH_EXTERNAL_LIB}')
398ea820d49SSatish Balay        self.addMakeMacro('PETSC_KSP_LIB','${C_SH_LIB_PATH} ${PETSC_WITH_EXTERNAL_LIB}')
399ea820d49SSatish Balay        self.addMakeMacro('PETSC_SNES_LIB','${C_SH_LIB_PATH} ${PETSC_WITH_EXTERNAL_LIB}')
400ea820d49SSatish Balay        self.addMakeMacro('PETSC_TS_LIB','${C_SH_LIB_PATH} ${PETSC_WITH_EXTERNAL_LIB}')
401b0a7d7e7SSatish Balay        self.addMakeMacro('PETSC_TAO_LIB','${C_SH_LIB_PATH} ${PETSC_WITH_EXTERNAL_LIB}')
402fdb87e33SJed Brown        self.addMakeMacro('PETSC_CHARACTERISTIC_LIB','${C_SH_LIB_PATH} ${PETSC_WITH_EXTERNAL_LIB}')
403ea820d49SSatish Balay        self.addMakeMacro('PETSC_LIB','${C_SH_LIB_PATH} ${PETSC_WITH_EXTERNAL_LIB}')
404ea820d49SSatish Balay        self.addMakeMacro('PETSC_CONTRIB','${C_SH_LIB_PATH} ${PETSC_WITH_EXTERNAL_LIB}')
4052df986feSBarry Smith      else:
406ea820d49SSatish Balay        self.addMakeMacro('PETSC_SYS_LIB','${PETSC_WITH_EXTERNAL_LIB}')
407ea820d49SSatish Balay        self.addMakeMacro('PETSC_VEC_LIB','${PETSC_WITH_EXTERNAL_LIB}')
408ea820d49SSatish Balay        self.addMakeMacro('PETSC_MAT_LIB','${PETSC_WITH_EXTERNAL_LIB}')
409ea820d49SSatish Balay        self.addMakeMacro('PETSC_DM_LIB','${PETSC_WITH_EXTERNAL_LIB}')
410ea820d49SSatish Balay        self.addMakeMacro('PETSC_KSP_LIB','${PETSC_WITH_EXTERNAL_LIB}')
411ea820d49SSatish Balay        self.addMakeMacro('PETSC_SNES_LIB','${PETSC_WITH_EXTERNAL_LIB}')
412ea820d49SSatish Balay        self.addMakeMacro('PETSC_TS_LIB','${PETSC_WITH_EXTERNAL_LIB}')
413b0a7d7e7SSatish Balay        self.addMakeMacro('PETSC_TAO_LIB','${PETSC_WITH_EXTERNAL_LIB}')
414fdb87e33SJed Brown        self.addMakeMacro('PETSC_CHARACTERISTIC_LIB','${PETSC_WITH_EXTERNAL_LIB}')
415ea820d49SSatish Balay        self.addMakeMacro('PETSC_LIB','${PETSC_WITH_EXTERNAL_LIB}')
416ea820d49SSatish Balay        self.addMakeMacro('PETSC_CONTRIB','${PETSC_WITH_EXTERNAL_LIB}')
4170f3b21c2SBarry Smith
418f8833479SBarry Smith    if not os.path.exists(os.path.join(self.petscdir.dir,self.arch.arch,'lib')):
419f8833479SBarry Smith      os.makedirs(os.path.join(self.petscdir.dir,self.arch.arch,'lib'))
420f8833479SBarry Smith
421f8833479SBarry Smith    # add a makefile entry for configure options
422f8833479SBarry Smith    self.addMakeMacro('CONFIGURE_OPTIONS', self.framework.getOptionsString(['configModules', 'optionsModule']).replace('\"','\\"'))
423f8833479SBarry Smith    return
424f8833479SBarry Smith
425f8833479SBarry Smith  def dumpConfigInfo(self):
426f8833479SBarry Smith    import time
427f8833479SBarry Smith    fd = file(os.path.join(self.arch.arch,'include','petscconfiginfo.h'),'w')
428f8833479SBarry Smith    fd.write('static const char *petscconfigureoptions = "'+self.framework.getOptionsString(['configModules', 'optionsModule']).replace('\"','\\"')+'";\n')
429f8833479SBarry Smith    fd.close()
430f8833479SBarry Smith    return
431f8833479SBarry Smith
4322a4161d9SMatthew G Knepley  def dumpMachineInfo(self):
4332a4161d9SMatthew G Knepley    import platform
4342a4161d9SMatthew G Knepley    import time
43540373944SSatish Balay    import script
436ca77dbeeSGeoffrey Irving    def escape(s):
437ca77dbeeSGeoffrey Irving      return s.replace('"',r'\"').replace(r'\ ',r'\\ ')
4382a4161d9SMatthew G Knepley    fd = file(os.path.join(self.arch.arch,'include','petscmachineinfo.h'),'w')
4392a4161d9SMatthew G Knepley    fd.write('static const char *petscmachineinfo = \"\\n\"\n')
4402a4161d9SMatthew G Knepley    fd.write('\"-----------------------------------------\\n\"\n')
4412a4161d9SMatthew G Knepley    fd.write('\"Libraries compiled on %s on %s \\n\"\n' % (time.ctime(time.time()), platform.node()))
44260acdfe7SSatish Balay    fd.write('\"Machine characteristics: %s\\n\"\n' % (platform.platform()))
443ca77dbeeSGeoffrey Irving    fd.write('\"Using PETSc directory: %s\\n\"\n' % (escape(self.petscdir.dir)))
444ca77dbeeSGeoffrey Irving    fd.write('\"Using PETSc arch: %s\\n\"\n' % (escape(self.arch.arch)))
445cdec380aSBarry Smith    fd.write('\"-----------------------------------------\\n\";\n')
4462a4161d9SMatthew G Knepley    fd.write('static const char *petsccompilerinfo = \"\\n\"\n')
4472a4161d9SMatthew G Knepley    self.setCompilers.pushLanguage(self.languages.clanguage)
448ca77dbeeSGeoffrey Irving    fd.write('\"Using C compiler: %s %s ${COPTFLAGS} ${CFLAGS}\\n\"\n' % (escape(self.setCompilers.getCompiler()), escape(self.setCompilers.getCompilerFlags())))
4492a4161d9SMatthew G Knepley    self.setCompilers.popLanguage()
4508782282cSMatthew G Knepley    if hasattr(self.compilers, 'FC'):
4512a4161d9SMatthew G Knepley      self.setCompilers.pushLanguage('FC')
452ca77dbeeSGeoffrey 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)))
4532a4161d9SMatthew G Knepley      self.setCompilers.popLanguage()
454cdec380aSBarry Smith    fd.write('\"-----------------------------------------\\n\";\n')
4552a4161d9SMatthew G Knepley    fd.write('static const char *petsccompilerflagsinfo = \"\\n\"\n')
456ca77dbeeSGeoffrey 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)))
457cdec380aSBarry Smith    fd.write('\"-----------------------------------------\\n\";\n')
4582a4161d9SMatthew G Knepley    fd.write('static const char *petsclinkerinfo = \"\\n\"\n')
4592a4161d9SMatthew G Knepley    self.setCompilers.pushLanguage(self.languages.clanguage)
460ca77dbeeSGeoffrey Irving    fd.write('\"Using C linker: %s\\n\"\n' % (escape(self.setCompilers.getLinker())))
4612a4161d9SMatthew G Knepley    self.setCompilers.popLanguage()
4628782282cSMatthew G Knepley    if hasattr(self.compilers, 'FC'):
4632a4161d9SMatthew G Knepley      self.setCompilers.pushLanguage('FC')
464ca77dbeeSGeoffrey Irving      fd.write('\"Using Fortran linker: %s\\n\"\n' % (escape(self.setCompilers.getLinker())))
4652a4161d9SMatthew G Knepley      self.setCompilers.popLanguage()
466ad782ac6SSatish Balay    if self.framework.argDB['with-single-library']:
467ad782ac6SSatish Balay      petsclib = '-lpetsc'
468ad782ac6SSatish Balay    else:
469ad782ac6SSatish Balay      petsclib = '-lpetscts -lpetscsnes -lpetscksp -lpetscdm -lpetscmat -lpetscvec -lpetscsys'
470ca77dbeeSGeoffrey 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)))
471cdec380aSBarry Smith    fd.write('\"-----------------------------------------\\n\";\n')
4722a4161d9SMatthew G Knepley    fd.close()
4732a4161d9SMatthew G Knepley    return
474b2843cf1SBarry Smith
475511a6afcSJed Brown  def dumpCMakeConfig(self):
476511a6afcSJed Brown    '''
477724dfae7SSatish Balay    Writes configuration-specific values to ${PETSC_ARCH}/lib/petsc/conf/PETScBuildInternal.cmake.
478511a6afcSJed Brown    This file is private to PETSc and should not be included by third parties
479511a6afcSJed Brown    (a suitable file can be produced later by CMake, but this is not it).
480511a6afcSJed Brown    '''
481511a6afcSJed Brown    def cmakeset(fd,key,val=True):
482511a6afcSJed Brown      if val == True: val = 'YES'
483511a6afcSJed Brown      if val == False: val = 'NO'
484511a6afcSJed Brown      fd.write('set (' + key + ' ' + val + ')\n')
485511a6afcSJed Brown    def ensurelist(a):
486826d9344SJed Brown      if isinstance(a,list):
487826d9344SJed Brown        return a
488826d9344SJed Brown      else:
489826d9344SJed Brown        return [a]
490511a6afcSJed Brown    def libpath(lib):
491511a6afcSJed Brown      'Returns a search path if that is what this item provides, else "" which will be cleaned out later'
4921b1c0b30SJed Brown      if not isinstance(lib,str): return ''
493511a6afcSJed Brown      if lib.startswith('-L'): return lib[2:]
494511a6afcSJed Brown      if lib.startswith('-R'): return lib[2:]
495511a6afcSJed Brown      if lib.startswith('-Wl,-rpath,'):
496511a6afcSJed Brown        # This case occurs when an external package needs a specific system library that is normally provided by the compiler.
497511a6afcSJed Brown        # In other words, the -L path is builtin to the wrapper or compiler, here we provide it so that CMake can locate the
498511a6afcSJed Brown        # corresponding library.
499511a6afcSJed Brown        return lib[len('-Wl,-rpath,'):]
500511a6afcSJed Brown      if lib.startswith('-'): return ''
501511a6afcSJed Brown      return os.path.dirname(lib)
502511a6afcSJed Brown    def cleanlib(lib):
503511a6afcSJed Brown      'Returns a library name if that is what this item provides, else "" which will be cleaned out later'
50442e8629dSMatthew G Knepley      if not isinstance(lib,str): return ''
505511a6afcSJed Brown      if lib.startswith('-l'):  return lib[2:]
506511a6afcSJed Brown      if lib.startswith('-Wl') or lib.startswith('-L'): return ''
507511a6afcSJed Brown      lib = os.path.splitext(os.path.basename(lib))[0]
508511a6afcSJed Brown      if lib.startswith('lib'): return lib[3:]
509511a6afcSJed Brown      return lib
510511a6afcSJed Brown    def nub(lst):
51106e8c1ddSJed Brown      'Return a list containing the first occurrence of each unique element'
512511a6afcSJed Brown      unique = []
513511a6afcSJed Brown      for elem in lst:
514511a6afcSJed Brown        if elem not in unique and elem != '':
515511a6afcSJed Brown          unique.append(elem)
516511a6afcSJed Brown      return unique
51750937898SJed Brown    try: reversed # reversed was added in Python-2.4
51850937898SJed Brown    except NameError:
51950937898SJed Brown      def reversed(lst): return lst[::-1]
52006e8c1ddSJed Brown    def nublast(lst):
52106e8c1ddSJed Brown      'Return a list containing the last occurrence of each unique entry in a list'
52250937898SJed Brown      return reversed(nub(reversed(lst)))
523511a6afcSJed Brown    def cmakeexpand(varname):
524511a6afcSJed Brown      return r'"${' + varname + r'}"'
525582751aaSJed Brown    def uniqextend(lst,new):
526511a6afcSJed Brown      for x in ensurelist(new):
527582751aaSJed Brown        if x not in lst:
528582751aaSJed Brown          lst.append(x)
529511a6afcSJed Brown    def notstandardinclude(path):
530040257f2SJed Brown      return path not in '/usr/include'.split() # /usr/local/include is not automatically included on FreeBSD
531511a6afcSJed Brown    def writeMacroDefinitions(fd):
532511a6afcSJed Brown      if self.mpi.usingMPIUni:
533511a6afcSJed Brown        cmakeset(fd,'PETSC_HAVE_MPIUNI')
534511a6afcSJed Brown      for pkg in self.framework.packages:
535511a6afcSJed Brown        if pkg.useddirectly:
536eeb16384SBarry Smith          cmakeset(fd,'PETSC_HAVE_' + pkg.PACKAGE.replace('-','_'))
537a23e9343SMatthew G Knepley        for pair in pkg.defines.items():
538440af75fSJed Brown          if pair[0].startswith('HAVE_') and pair[1]:
539a23e9343SMatthew G Knepley            cmakeset(fd, self.framework.getFullDefineName(pkg, pair[0]), pair[1])
540511a6afcSJed Brown      for name,val in self.functions.defines.items():
541511a6afcSJed Brown        cmakeset(fd,'PETSC_'+name,val)
542511a6afcSJed Brown      for dct in [self.defines, self.libraryoptions.defines]:
543511a6afcSJed Brown        for k,v in dct.items():
544511a6afcSJed Brown          if k.startswith('USE_'):
545511a6afcSJed Brown            cmakeset(fd,'PETSC_' + k, v)
546511a6afcSJed Brown      cmakeset(fd,'PETSC_USE_COMPLEX', self.scalartypes.scalartype == 'complex')
547ce63c4c1SBarry Smith      cmakeset(fd,'PETSC_USE_REAL_' + self.scalartypes.precision.upper())
548511a6afcSJed Brown      cmakeset(fd,'PETSC_CLANGUAGE_'+self.languages.clanguage)
549511a6afcSJed Brown      if hasattr(self.compilers, 'FC'):
550511a6afcSJed Brown        cmakeset(fd,'PETSC_HAVE_FORTRAN')
551511a6afcSJed Brown        if self.compilers.fortranIsF90:
552511a6afcSJed Brown          cmakeset(fd,'PETSC_USING_F90')
553876d5c60SBarry Smith        if self.compilers.fortranIsF2003:
554876d5c60SBarry Smith          cmakeset(fd,'PETSC_USING_F2003')
55513c0a95cSJed Brown      if hasattr(self.compilers, 'CXX'):
55613c0a95cSJed Brown        cmakeset(fd,'PETSC_HAVE_CXX')
557511a6afcSJed Brown      if self.sharedlibraries.useShared:
558511a6afcSJed Brown        cmakeset(fd,'BUILD_SHARED_LIBS')
559511a6afcSJed Brown    def writeBuildFlags(fd):
56006e8c1ddSJed Brown      def extendby(lib):
56106e8c1ddSJed Brown        libs = ensurelist(lib)
56206e8c1ddSJed Brown        lib_paths.extend(map(libpath,libs))
56306e8c1ddSJed Brown        lib_libs.extend(map(cleanlib,libs))
564511a6afcSJed Brown      lib_paths = []
565511a6afcSJed Brown      lib_libs  = []
566511a6afcSJed Brown      includes  = []
567511a6afcSJed Brown      libvars   = []
568511a6afcSJed Brown      for pkg in self.framework.packages:
5694d02c0d4SBarry Smith        if pkg.linkedbypetsc:
57006e8c1ddSJed Brown          extendby(pkg.lib)
571040257f2SJed Brown          uniqextend(includes,pkg.include)
57206e8c1ddSJed Brown      extendby(self.libraries.math)
57306e8c1ddSJed Brown      extendby(self.libraries.rt)
57406e8c1ddSJed Brown      extendby(self.compilers.flibs)
57506e8c1ddSJed Brown      extendby(self.compilers.cxxlibs)
57606e8c1ddSJed Brown      extendby(self.compilers.LIBS.split())
57706e8c1ddSJed Brown      for libname in nublast(lib_libs):
578511a6afcSJed Brown        libvar = 'PETSC_' + libname.upper() + '_LIB'
5794c0032a9SSatish Balay        addpath = ''
58006e8c1ddSJed Brown        for lpath in nublast(lib_paths):
5814c0032a9SSatish Balay          addpath += '"' + str(lpath) + '" '
5824c0032a9SSatish Balay        fd.write('find_library (' + libvar + ' ' + libname + ' HINTS ' + addpath + ')\n')
583511a6afcSJed Brown        libvars.append(libvar)
584511a6afcSJed Brown      fd.write('mark_as_advanced (' + ' '.join(libvars) + ')\n')
585511a6afcSJed Brown      fd.write('set (PETSC_PACKAGE_LIBS ' + ' '.join(map(cmakeexpand,libvars)) + ')\n')
586040257f2SJed Brown      includes = filter(notstandardinclude,includes)
587040257f2SJed Brown      fd.write('set (PETSC_PACKAGE_INCLUDES ' + ' '.join(map(lambda i: '"'+i+'"',includes)) + ')\n')
588724dfae7SSatish Balay    fd = open(os.path.join(self.arch.arch,'lib','petsc','conf','PETScBuildInternal.cmake'), 'w')
589511a6afcSJed Brown    writeMacroDefinitions(fd)
590511a6afcSJed Brown    writeBuildFlags(fd)
591511a6afcSJed Brown    fd.close()
592511a6afcSJed Brown    return
593511a6afcSJed Brown
5948b0282a9SJed Brown  def dumpCMakeLists(self):
5958b0282a9SJed Brown    import sys
596994b4dadSSatish Balay    if sys.version_info >= (2,4):
5978b0282a9SJed Brown      import cmakegen
5988b0282a9SJed Brown      try:
599a98e69d2SJed Brown        cmakegen.main(self.petscdir.dir, log=self.framework.log)
6008b0282a9SJed Brown      except (OSError), e:
6018b0282a9SJed Brown        self.framework.logPrint('Generating CMakeLists.txt failed:\n' + str(e))
602aac20692SSatish Balay    else:
603aac20692SSatish Balay      self.framework.logPrint('Skipping cmakegen due to old python version: ' +str(sys.version_info) )
6048b0282a9SJed Brown
6058b0282a9SJed Brown  def cmakeBoot(self):
6068b0282a9SJed Brown    import sys
607ae937f1dSJed Brown    self.cmakeboot_success = False
608994b4dadSSatish Balay    if sys.version_info >= (2,4) and hasattr(self.cmake,'cmake'):
6095a4feeedSSatish Balay      oldRead = self.argDB.readonly
6105a4feeedSSatish Balay      self.argDB.readonly = True
611356464bcSMatthew G Knepley      try:
6128b0282a9SJed Brown        import cmakeboot
613ae937f1dSJed Brown        self.cmakeboot_success = cmakeboot.main(petscdir=self.petscdir.dir,petscarch=self.arch.arch,argDB=self.argDB,framework=self.framework,log=self.framework.log)
6148b0282a9SJed Brown      except (OSError), e:
6158b0282a9SJed Brown        self.framework.logPrint('Booting CMake in PETSC_ARCH failed:\n' + str(e))
616356464bcSMatthew G Knepley      except (ImportError, KeyError), e:
617356464bcSMatthew G Knepley        self.framework.logPrint('Importing cmakeboot failed:\n' + str(e))
6185a4feeedSSatish Balay      self.argDB.readonly = oldRead
6199b12c9c7SJed Brown      if self.cmakeboot_success:
6202f730bc2SKarl Rupp        if hasattr(self.compilers, 'FC') and self.compilers.fortranIsF90 and not self.setCompilers.fortranModuleOutputFlag:
62191f9b906SSatish Balay          self.framework.logPrint('CMake configured successfully, but could not be used by default because of missing fortranModuleOutputFlag\n')
6229b12c9c7SJed Brown        else:
6239b12c9c7SJed Brown          self.framework.logPrint('CMake configured successfully, using as default build\n')
624f7b66a64SJed Brown          self.addMakeMacro('PETSC_BUILD_USING_CMAKE',1)
625aac20692SSatish Balay      else:
6269b12c9c7SJed Brown        self.framework.logPrint('CMake configuration was unsuccessful\n')
6279b12c9c7SJed Brown    else:
628aac20692SSatish Balay      self.framework.logPrint('Skipping cmakeboot due to old python version: ' +str(sys.version_info) )
629356464bcSMatthew G Knepley    return
6308b0282a9SJed Brown
631b2843cf1SBarry Smith  def configurePrefetch(self):
632b2843cf1SBarry Smith    '''Sees if there are any prefetch functions supported'''
6337fca349cSMatthew G. Knepley    if config.setCompilers.Configure.isSolaris(self.log) or self.framework.argDB['with-ios'] or not self.framework.argDB['with-prefetch']:
63493f78423SSatish Balay      self.addDefine('Prefetch(a,b,c)', ' ')
63593f78423SSatish Balay      return
636ec284106SBarry Smith    self.pushLanguage(self.languages.clanguage)
63710699583SJed Brown    if self.checkLink('#include <xmmintrin.h>', 'void *v = 0;_mm_prefetch((const char*)v,_MM_HINT_NTA);\n'):
63850d8bf02SJed Brown      # The Intel Intrinsics manual [1] specifies the prototype
63950d8bf02SJed Brown      #
64050d8bf02SJed Brown      #   void _mm_prefetch(char const *a, int sel);
64150d8bf02SJed Brown      #
64250d8bf02SJed Brown      # but other vendors seem to insist on using subtly different
64350d8bf02SJed Brown      # prototypes, including void* for the pointer, and an enum for
64450d8bf02SJed Brown      # sel.  These are both reasonable changes, but negatively impact
64550d8bf02SJed Brown      # portability.
64650d8bf02SJed Brown      #
64750d8bf02SJed Brown      # [1] http://software.intel.com/file/6373
64850d8bf02SJed Brown      self.addDefine('HAVE_XMMINTRIN_H', 1)
64950d8bf02SJed Brown      self.addDefine('Prefetch(a,b,c)', '_mm_prefetch((const char*)(a),(c))')
65050d8bf02SJed Brown      self.addDefine('PREFETCH_HINT_NTA', '_MM_HINT_NTA')
65150d8bf02SJed Brown      self.addDefine('PREFETCH_HINT_T0',  '_MM_HINT_T0')
65250d8bf02SJed Brown      self.addDefine('PREFETCH_HINT_T1',  '_MM_HINT_T1')
65350d8bf02SJed Brown      self.addDefine('PREFETCH_HINT_T2',  '_MM_HINT_T2')
65450d8bf02SJed Brown    elif self.checkLink('#include <xmmintrin.h>', 'void *v = 0;_mm_prefetch(v,_MM_HINT_NTA);\n'):
65550d8bf02SJed Brown      self.addDefine('HAVE_XMMINTRIN_H', 1)
65650d8bf02SJed Brown      self.addDefine('Prefetch(a,b,c)', '_mm_prefetch((const void*)(a),(c))')
65750d8bf02SJed Brown      self.addDefine('PREFETCH_HINT_NTA', '_MM_HINT_NTA')
65850d8bf02SJed Brown      self.addDefine('PREFETCH_HINT_T0',  '_MM_HINT_T0')
65950d8bf02SJed Brown      self.addDefine('PREFETCH_HINT_T1',  '_MM_HINT_T1')
66050d8bf02SJed Brown      self.addDefine('PREFETCH_HINT_T2',  '_MM_HINT_T2')
66110699583SJed Brown    elif self.checkLink('', 'void *v = 0;__builtin_prefetch(v,0,0);\n'):
66210699583SJed Brown      # From GCC docs: void __builtin_prefetch(const void *addr,int rw,int locality)
66310699583SJed Brown      #
66410699583SJed Brown      #   The value of rw is a compile-time constant one or zero; one
66510699583SJed Brown      #   means that the prefetch is preparing for a write to the memory
66610699583SJed Brown      #   address and zero, the default, means that the prefetch is
66710699583SJed Brown      #   preparing for a read. The value locality must be a compile-time
66810699583SJed Brown      #   constant integer between zero and three. A value of zero means
66910699583SJed Brown      #   that the data has no temporal locality, so it need not be left
67010699583SJed Brown      #   in the cache after the access. A value of three means that the
67110699583SJed Brown      #   data has a high degree of temporal locality and should be left
67210699583SJed Brown      #   in all levels of cache possible. Values of one and two mean,
67310699583SJed Brown      #   respectively, a low or moderate degree of temporal locality.
67410699583SJed Brown      #
67510699583SJed Brown      # Here we adopt Intel's x86/x86-64 naming scheme for the locality
67610699583SJed Brown      # hints.  Using macros for these values in necessary since some
67710699583SJed Brown      # compilers require an enum.
67810699583SJed Brown      self.addDefine('Prefetch(a,b,c)', '__builtin_prefetch((a),(b),(c))')
67910699583SJed Brown      self.addDefine('PREFETCH_HINT_NTA', '0')
68010699583SJed Brown      self.addDefine('PREFETCH_HINT_T0',  '3')
68110699583SJed Brown      self.addDefine('PREFETCH_HINT_T1',  '2')
68210699583SJed Brown      self.addDefine('PREFETCH_HINT_T2',  '1')
683b2843cf1SBarry Smith    else:
684b2843cf1SBarry Smith      self.addDefine('Prefetch(a,b,c)', ' ')
6857d490b44SBarry Smith    self.popLanguage()
686b2843cf1SBarry Smith
68709bc878fSSatish Balay  def configureAtoll(self):
68809bc878fSSatish Balay    '''Checks if atoll exists'''
689436b02dcSSatish 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")'):
69009bc878fSSatish Balay       self.addDefine('HAVE_ATOLL', '1')
69109bc878fSSatish Balay
6922400fdedSBarry Smith  def configureUnused(self):
6932400fdedSBarry Smith    '''Sees if __attribute((unused)) is supported'''
6941adaff47SSean Farley    if self.framework.argDB['with-ios']:
6952400fdedSBarry Smith      self.addDefine('UNUSED', ' ')
6962400fdedSBarry Smith      return
6972400fdedSBarry Smith    self.pushLanguage(self.languages.clanguage)
698edf21b64SSatish 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'):
6992400fdedSBarry Smith      self.addDefine('UNUSED', '__attribute((unused))')
7002400fdedSBarry Smith    else:
7012400fdedSBarry Smith      self.addDefine('UNUSED', ' ')
7022400fdedSBarry Smith    self.popLanguage()
7032400fdedSBarry Smith
70498ed35c3SBarry Smith  def configureIsatty(self):
70598ed35c3SBarry Smith    '''Check if the Unix C function isatty() works correctly
70698ed35c3SBarry Smith       Actually just assumes it does not work correctly on batch systems'''
70798ed35c3SBarry Smith    if not self.framework.argDB['with-batch']:
70898ed35c3SBarry Smith      self.addDefine('USE_ISATTY',1)
70998ed35c3SBarry Smith
7101ef8df7fSJed Brown  def configureDeprecated(self):
7111ef8df7fSJed Brown    '''Check if __attribute((deprecated)) is supported'''
7121ef8df7fSJed Brown    self.pushLanguage(self.languages.clanguage)
71359a26b54SJed Brown    ## Recent versions of gcc and clang support __attribute((deprecated("string argument"))), which is very useful, but
71459a26b54SJed Brown    ## Intel has conspired to make a supremely environment-sensitive compiler.  The Intel compiler looks at the gcc
71559a26b54SJed Brown    ## executable in the environment to determine the language compatibility that it should attempt to emulate.  Some
71659a26b54SJed Brown    ## important Cray installations have built PETSc using the Intel compiler, but with a newer gcc module loaded (e.g.,
717df3898eeSBarry Smith    ## 4.7).  Thus at PETSc configure time, the Intel compiler decides to support the string argument, but the gcc
71859a26b54SJed Brown    ## found in the default user environment is older and does not support the argument.  If GCC and Intel were cool
71959a26b54SJed Brown    ## like Clang and supported __has_attribute, we could avoid configure tests entirely, but they don't.  And that is
72059a26b54SJed Brown    ## why we can't have nice things.
72159a26b54SJed Brown    #
72259a26b54SJed Brown    # if self.checkCompile("""__attribute((deprecated("Why you shouldn't use myfunc"))) static int myfunc(void) { return 1;}""", ''):
72359a26b54SJed Brown    #   self.addDefine('DEPRECATED(why)', '__attribute((deprecated(why)))')
72459a26b54SJed Brown    if self.checkCompile("""__attribute((deprecated)) static int myfunc(void) { return 1;}""", ''):
72559a26b54SJed Brown      self.addDefine('DEPRECATED(why)', '__attribute((deprecated))')
7261ef8df7fSJed Brown    else:
72747644db9SJed Brown      self.addDefine('DEPRECATED(why)', ' ')
7281ef8df7fSJed Brown    self.popLanguage()
7291ef8df7fSJed Brown
73018f41590SBarry Smith  def configureAlign(self):
73118f41590SBarry Smith    '''Check if __attribute(align) is supported'''
732752d89a4SSatish Balay    filename = 'conftestalign'
733752d89a4SSatish Balay    includes = '''
734752d89a4SSatish Balay#include <sys/types.h>
735752d89a4SSatish Balay#if STDC_HEADERS
736752d89a4SSatish Balay#include <stdlib.h>
737752d89a4SSatish Balay#include <stdio.h>
738752d89a4SSatish Balay#include <stddef.h>
739752d89a4SSatish Balay#endif\n'''
740752d89a4SSatish Balay    body     = '''
741752d89a4SSatish Balaystruct mystruct {int myint;} __attribute((aligned(16)));
742752d89a4SSatish BalayFILE *f = fopen("'''+filename+'''", "w");
743752d89a4SSatish Balayif (!f) exit(1);
744752d89a4SSatish Balayfprintf(f, "%lu\\n", (unsigned long)sizeof(struct mystruct));
745752d89a4SSatish Balay'''
746752d89a4SSatish Balay    if 'known-has-attribute-aligned' in self.argDB:
747752d89a4SSatish Balay      if self.argDB['known-has-attribute-aligned']:
748752d89a4SSatish Balay        size = 16
74918f41590SBarry Smith      else:
750752d89a4SSatish Balay        size = -3
751752d89a4SSatish Balay    elif not self.argDB['with-batch']:
752752d89a4SSatish Balay      self.pushLanguage(self.languages.clanguage)
753752d89a4SSatish Balay      try:
754752d89a4SSatish Balay        if self.checkRun(includes, body) and os.path.exists(filename):
755752d89a4SSatish Balay          f    = file(filename)
756752d89a4SSatish Balay          size = int(f.read())
757752d89a4SSatish Balay          f.close()
758752d89a4SSatish Balay          os.remove(filename)
7590045a809SSatish Balay        else:
7600045a809SSatish Balay          size = -4
761752d89a4SSatish Balay      except:
762752d89a4SSatish Balay        size = -1
763752d89a4SSatish Balay        self.framework.logPrint('Error checking attribute(aligned)')
76418f41590SBarry Smith      self.popLanguage()
765752d89a4SSatish Balay    else:
766752d89a4SSatish Balay      self.framework.addBatchInclude(['#include <stdlib.h>', '#include <stdio.h>', '#include <sys/types.h>','struct mystruct {int myint;} __attribute((aligned(16)));'])
767752d89a4SSatish Balay      self.framework.addBatchBody('fprintf(output, "  \'--known-has-attribute-aligned=%d\',\\n", sizeof(struct mystruct)==16);')
768752d89a4SSatish Balay      size = -2
769752d89a4SSatish Balay    if size == 16:
770752d89a4SSatish Balay      self.addDefine('ATTRIBUTEALIGNED(size)', '__attribute((aligned (size)))')
771752d89a4SSatish Balay      self.addDefine('HAVE_ATTRIBUTEALIGNED', 1)
772752d89a4SSatish Balay    else:
773752d89a4SSatish Balay      self.framework.logPrint('incorrect alignment. Found alignment:'+ str(size))
774752d89a4SSatish Balay      self.addDefine('ATTRIBUTEALIGNED(size)', ' ')
775752d89a4SSatish Balay    return
77618f41590SBarry Smith
7779800092aSJed Brown  def configureExpect(self):
7789800092aSJed Brown    '''Sees if the __builtin_expect directive is supported'''
7799800092aSJed Brown    self.pushLanguage(self.languages.clanguage)
7809800092aSJed Brown    if self.checkLink('', 'if (__builtin_expect(0,1)) return 1;'):
7819800092aSJed Brown      self.addDefine('HAVE_BUILTIN_EXPECT', 1)
7829800092aSJed Brown    self.popLanguage()
7839800092aSJed Brown
78453c77d0aSJed Brown  def configureFunctionName(self):
7851ec50b02SJed Brown    '''Sees if the compiler supports __func__ or a variant.  Falls back
7861ec50b02SJed Brown    on __FUNCT__ which PETSc source defines, but most users do not, thus
7871ec50b02SJed Brown    stack traces through user code are better when the compiler's
7881ec50b02SJed Brown    variant is used.'''
7891ec50b02SJed Brown    def getFunctionName(lang):
7901ec50b02SJed Brown      name = '__FUNCT__'
7911ec50b02SJed Brown      self.pushLanguage(lang)
79253c77d0aSJed Brown      if self.checkLink('', "if (__func__[0] != 'm') return 1;"):
7931ec50b02SJed Brown        name = '__func__'
79453c77d0aSJed Brown      elif self.checkLink('', "if (__FUNCTION__[0] != 'm') return 1;"):
7951ec50b02SJed Brown        name = '__FUNCTION__'
7961ec50b02SJed Brown      self.popLanguage()
7971ec50b02SJed Brown      return name
7981ec50b02SJed Brown    langs = []
799628773c9SSatish Balay
800628773c9SSatish Balay    self.addDefine('FUNCTION_NAME_C', getFunctionName('C'))
801628773c9SSatish Balay    if hasattr(self.compilers, 'CXX'):
802628773c9SSatish Balay      self.addDefine('FUNCTION_NAME_CXX', getFunctionName('Cxx'))
80312607bf0SSatish Balay    else:
80412607bf0SSatish Balay      self.addDefine('FUNCTION_NAME_CXX', '__FUNCT__')
80553c77d0aSJed Brown
806753ebd1dSJed Brown  def configureIntptrt(self):
807753ebd1dSJed Brown    '''Determine what to use for uintptr_t'''
808753ebd1dSJed Brown    def staticAssertSizeMatchesVoidStar(inc,typename):
809753ebd1dSJed Brown      # The declaration is an error if either array size is negative.
810753ebd1dSJed 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
811d26187a0SJed Brown      return self.checkCompile(inc, ('#define STATIC_ASSERT(cond) char negative_length_if_false[2*(!!(cond))-1]\n'
812979939cdSSatish Balay                                     + 'STATIC_ASSERT(sizeof(void*) == sizeof(%s));'%typename))
813753ebd1dSJed Brown    self.pushLanguage(self.languages.clanguage)
814753ebd1dSJed Brown    if self.checkCompile('#include <stdint.h>', 'int x; uintptr_t i = (uintptr_t)&x;'):
815753ebd1dSJed Brown      self.addDefine('UINTPTR_T', 'uintptr_t')
816753ebd1dSJed Brown    elif staticAssertSizeMatchesVoidStar('','unsigned long long'):
817753ebd1dSJed Brown      self.addDefine('UINTPTR_T', 'unsigned long long')
818753ebd1dSJed Brown    elif staticAssertSizeMatchesVoidStar('#include <stdlib.h>','size_t') or staticAssertSizeMatchesVoidStar('#include <string.h>', 'size_t'):
819753ebd1dSJed Brown      self.addDefine('UINTPTR_T', 'size_t')
820c82284b1SJed Brown    elif staticAssertSizeMatchesVoidStar('','unsigned long'):
821c82284b1SJed Brown      self.addDefine('UINTPTR_T', 'unsigned long')
8222d1b7972SSatish Balay    elif staticAssertSizeMatchesVoidStar('','unsigned'):
823753ebd1dSJed Brown      self.addDefine('UINTPTR_T', 'unsigned')
824d26187a0SJed Brown    else:
825d26187a0SJed Brown      raise RuntimeError('Could not find any unsigned integer type matching void*')
826753ebd1dSJed Brown    self.popLanguage()
827753ebd1dSJed Brown
828ed938b00SJed Brown  def configureRTLDDefault(self):
829bfef2c86SBarry Smith    if self.checkCompile('#include <dlfcn.h>\n void *ptr =  RTLD_DEFAULT;'):
830bfef2c86SBarry Smith      self.addDefine('RTLD_DEFAULT','1')
831f8833479SBarry Smith    return
832f8833479SBarry Smith
833f8833479SBarry Smith  def configureSolaris(self):
834f8833479SBarry Smith    '''Solaris specific stuff'''
835f8833479SBarry Smith    if os.path.isdir(os.path.join('/usr','ucblib')):
836f8833479SBarry Smith      try:
837f8833479SBarry Smith        flag = getattr(self.setCompilers, self.language[-1]+'SharedLinkerFlag')
838f8833479SBarry Smith      except AttributeError:
839f8833479SBarry Smith        flag = None
840f8833479SBarry Smith      if flag is None:
841f8833479SBarry Smith        self.compilers.LIBS += ' -L/usr/ucblib'
842f8833479SBarry Smith      else:
843f8833479SBarry Smith        self.compilers.LIBS += ' '+flag+'/usr/ucblib'
844f8833479SBarry Smith    return
845f8833479SBarry Smith
846f8833479SBarry Smith  def configureLinux(self):
847f8833479SBarry Smith    '''Linux specific stuff'''
8489f15855cSMatthew G Knepley    # TODO: Test for this by mallocing an odd number of floats and checking the address
849f8833479SBarry Smith    self.addDefine('HAVE_DOUBLE_ALIGN_MALLOC', 1)
850f8833479SBarry Smith    return
851f8833479SBarry Smith
852f8833479SBarry Smith  def configureWin32(self):
853f8833479SBarry Smith    '''Win32 non-cygwin specific stuff'''
854f8833479SBarry Smith    kernel32=0
855f8833479SBarry Smith    if self.libraries.add('Kernel32.lib','GetComputerName',prototype='#include <Windows.h>', call='GetComputerName(NULL,NULL);'):
856f8833479SBarry Smith      self.addDefine('HAVE_WINDOWS_H',1)
857f8833479SBarry Smith      self.addDefine('HAVE_GETCOMPUTERNAME',1)
858f8833479SBarry Smith      kernel32=1
859f8833479SBarry Smith    elif self.libraries.add('kernel32','GetComputerName',prototype='#include <Windows.h>', call='GetComputerName(NULL,NULL);'):
860f8833479SBarry Smith      self.addDefine('HAVE_WINDOWS_H',1)
861f8833479SBarry Smith      self.addDefine('HAVE_GETCOMPUTERNAME',1)
862f8833479SBarry Smith      kernel32=1
863f8833479SBarry Smith    if kernel32:
864eed94e11SSatish Balay      if self.framework.argDB['with-windows-graphics']:
865eed94e11SSatish Balay        self.addDefine('USE_WINDOWS_GRAPHICS',1)
866f8833479SBarry Smith      if self.checkLink('#include <Windows.h>','LoadLibrary(0)'):
867f8833479SBarry Smith        self.addDefine('HAVE_LOADLIBRARY',1)
868b50f6d9eSLisandro Dalcin      if self.checkLink('#include <Windows.h>','GetProcAddress(0,0)'):
869b50f6d9eSLisandro Dalcin        self.addDefine('HAVE_GETPROCADDRESS',1)
870b50f6d9eSLisandro Dalcin      if self.checkLink('#include <Windows.h>','FreeLibrary(0)'):
871b50f6d9eSLisandro Dalcin        self.addDefine('HAVE_FREELIBRARY',1)
872a21658a3SLisandro Dalcin      if self.checkLink('#include <Windows.h>','GetLastError()'):
873a21658a3SLisandro Dalcin        self.addDefine('HAVE_GETLASTERROR',1)
874a21658a3SLisandro Dalcin      if self.checkLink('#include <Windows.h>','SetLastError(0)'):
875a21658a3SLisandro Dalcin        self.addDefine('HAVE_SETLASTERROR',1)
876f8833479SBarry Smith      if self.checkLink('#include <Windows.h>\n','QueryPerformanceCounter(0);\n'):
877bea725cfSBarry Smith        self.addDefine('USE_MICROSOFT_TIME',1)
878f8833479SBarry Smith    if self.libraries.add('Advapi32.lib','GetUserName',prototype='#include <Windows.h>', call='GetUserName(NULL,NULL);'):
879f8833479SBarry Smith      self.addDefine('HAVE_GET_USER_NAME',1)
880f8833479SBarry Smith    elif self.libraries.add('advapi32','GetUserName',prototype='#include <Windows.h>', call='GetUserName(NULL,NULL);'):
881f8833479SBarry Smith      self.addDefine('HAVE_GET_USER_NAME',1)
882f8833479SBarry Smith
883f8833479SBarry Smith    if not self.libraries.add('User32.lib','GetDC',prototype='#include <Windows.h>',call='GetDC(0);'):
884f8833479SBarry Smith      self.libraries.add('user32','GetDC',prototype='#include <Windows.h>',call='GetDC(0);')
885f8833479SBarry Smith    if not self.libraries.add('Gdi32.lib','CreateCompatibleDC',prototype='#include <Windows.h>',call='CreateCompatibleDC(0);'):
886f8833479SBarry Smith      self.libraries.add('gdi32','CreateCompatibleDC',prototype='#include <Windows.h>',call='CreateCompatibleDC(0);')
887f8833479SBarry Smith
888f8833479SBarry Smith    self.types.check('int32_t', 'int')
889f8833479SBarry Smith    if not self.checkCompile('#include <sys/types.h>\n','uid_t u;\n'):
890f8833479SBarry Smith      self.addTypedef('int', 'uid_t')
891f8833479SBarry Smith      self.addTypedef('int', 'gid_t')
892f8833479SBarry Smith    if not self.checkLink('#if defined(PETSC_HAVE_UNISTD_H)\n#include <unistd.h>\n#endif\n','int a=R_OK;\n'):
893f8833479SBarry Smith      self.framework.addDefine('R_OK', '04')
894f8833479SBarry Smith      self.framework.addDefine('W_OK', '02')
895f8833479SBarry Smith      self.framework.addDefine('X_OK', '01')
896f8833479SBarry Smith    if not self.checkLink('#include <sys/stat.h>\n','int a=0;\nif (S_ISDIR(a)){}\n'):
897f8833479SBarry Smith      self.framework.addDefine('S_ISREG(a)', '(((a)&_S_IFMT) == _S_IFREG)')
898f8833479SBarry Smith      self.framework.addDefine('S_ISDIR(a)', '(((a)&_S_IFMT) == _S_IFDIR)')
899f8833479SBarry Smith    if self.checkCompile('#include <Windows.h>\n','LARGE_INTEGER a;\nDWORD b=a.u.HighPart;\n'):
900f8833479SBarry Smith      self.addDefine('HAVE_LARGE_INTEGER_U',1)
901f8833479SBarry Smith
902f8833479SBarry Smith    # Windows requires a Binary file creation flag when creating/opening binary files.  Is a better test in order?
903ef2cfba3SSatish Balay    if self.checkCompile('#include <Windows.h>\n#include <fcntl.h>\n', 'int flags = O_BINARY;'):
904f8833479SBarry Smith      self.addDefine('HAVE_O_BINARY',1)
905f8833479SBarry Smith
906f8833479SBarry Smith    if self.compilers.CC.find('win32fe') >= 0:
907f8833479SBarry Smith      self.addDefine('PATH_SEPARATOR','\';\'')
908f8833479SBarry Smith      self.addDefine('DIR_SEPARATOR','\'\\\\\'')
909f8833479SBarry Smith      self.addDefine('REPLACE_DIR_SEPARATOR','\'/\'')
910f8833479SBarry Smith      self.addDefine('CANNOT_START_DEBUGGER',1)
9117908f030SMatthew G. Knepley      (petscdir,error,status) = self.executeShellCommand('cygpath -w '+self.petscdir.dir, log = self.log)
91234531a4dSSatish Balay      self.addDefine('DIR','"'+petscdir.replace('\\','\\\\')+'"')
913f8833479SBarry Smith    else:
914f8833479SBarry Smith      self.addDefine('PATH_SEPARATOR','\':\'')
915f8833479SBarry Smith      self.addDefine('REPLACE_DIR_SEPARATOR','\'\\\\\'')
916f8833479SBarry Smith      self.addDefine('DIR_SEPARATOR','\'/\'')
91734531a4dSSatish Balay      self.addDefine('DIR', '"'+self.petscdir.dir+'"')
918bfef2c86SBarry Smith
919f8833479SBarry Smith    return
920f8833479SBarry Smith
921f8833479SBarry Smith#-----------------------------------------------------------------------------------------------------
922b10d012aSSatish Balay  def configureCygwinBrokenPipe(self):
923b10d012aSSatish Balay    '''Cygwin version <= 1.7.18 had issues with pipes and long commands invoked from gnu-make
924b10d012aSSatish Balay    http://cygwin.com/ml/cygwin/2013-05/msg00340.html '''
9257fca349cSMatthew G. Knepley    if config.setCompilers.Configure.isCygwin(self.log):
926b10d012aSSatish Balay      import platform
927b10d012aSSatish Balay      import re
928b10d012aSSatish Balay      r=re.compile("([0-9]+).([0-9]+).([0-9]+)")
929b10d012aSSatish Balay      m=r.match(platform.release())
930b10d012aSSatish Balay      major=int(m.group(1))
931b10d012aSSatish Balay      minor=int(m.group(2))
932b10d012aSSatish Balay      subminor=int(m.group(3))
933b10d012aSSatish Balay      if ((major < 1) or (major == 1 and minor < 7) or (major == 1 and minor == 7 and subminor <= 18)):
934b10d012aSSatish Balay        self.addMakeMacro('PETSC_CYGWIN_BROKEN_PIPE','1')
935b10d012aSSatish Balay    return
936b10d012aSSatish Balay
937b10d012aSSatish Balay#-----------------------------------------------------------------------------------------------------
938569865ddSSatish Balay  def configureDefaultArch(self):
939af0996ceSBarry Smith    conffile = os.path.join('lib','petsc','conf', 'petscvariables')
940569865ddSSatish Balay    if self.framework.argDB['with-default-arch']:
941569865ddSSatish Balay      fd = file(conffile, 'w')
942569865ddSSatish Balay      fd.write('PETSC_ARCH='+self.arch.arch+'\n')
943da93591fSBarry Smith      fd.write('PETSC_DIR='+self.petscdir.dir+'\n')
944af0996ceSBarry Smith      fd.write('include '+os.path.join(self.petscdir.dir,self.arch.arch,'lib','petsc','conf','petscvariables')+'\n')
945569865ddSSatish Balay      fd.close()
946569865ddSSatish Balay      self.framework.actions.addArgument('PETSc', 'Build', 'Set default architecture to '+self.arch.arch+' in '+conffile)
947569865ddSSatish Balay    elif os.path.isfile(conffile):
948569865ddSSatish Balay      try:
949569865ddSSatish Balay        os.unlink(conffile)
950569865ddSSatish Balay      except:
951569865ddSSatish Balay        raise RuntimeError('Unable to remove file '+conffile+'. Did a different user create it?')
952569865ddSSatish Balay    return
953569865ddSSatish Balay
954569865ddSSatish Balay#-----------------------------------------------------------------------------------------------------
955f8833479SBarry Smith  def configureScript(self):
956f8833479SBarry Smith    '''Output a script in the conf directory which will reproduce the configuration'''
957f8833479SBarry Smith    import nargs
958495bf1a9SSatish Balay    import sys
959af0996ceSBarry Smith    scriptName = os.path.join(self.arch.arch,'lib','petsc','conf', 'reconfigure-'+self.arch.arch+'.py')
960f8833479SBarry Smith    args = dict([(nargs.Arg.parseArgument(arg)[0], arg) for arg in self.framework.clArgs])
961e97fc2efSSatish Balay    if 'with-clean' in args:
962e97fc2efSSatish Balay      del args['with-clean']
963f8833479SBarry Smith    if 'configModules' in args:
9641063a081SSatish Balay      if nargs.Arg.parseArgument(args['configModules'])[1] == 'PETSc.Configure':
965f8833479SBarry Smith        del args['configModules']
966f8833479SBarry Smith    if 'optionsModule' in args:
96723a19ef1SSatish Balay      if nargs.Arg.parseArgument(args['optionsModule'])[1] == 'config.compilerOptions':
968f8833479SBarry Smith        del args['optionsModule']
969f8833479SBarry Smith    if not 'PETSC_ARCH' in args:
9701063a081SSatish Balay      args['PETSC_ARCH'] = 'PETSC_ARCH='+str(self.arch.arch)
971f8833479SBarry Smith    f = file(scriptName, 'w')
972495bf1a9SSatish Balay    f.write('#!'+sys.executable+'\n')
973f8833479SBarry Smith    f.write('if __name__ == \'__main__\':\n')
974f8833479SBarry Smith    f.write('  import sys\n')
9757561c02cSSatish Balay    f.write('  import os\n')
9767561c02cSSatish Balay    f.write('  sys.path.insert(0, os.path.abspath(\'config\'))\n')
977f8833479SBarry Smith    f.write('  import configure\n')
9781063a081SSatish Balay    # pretty print repr(args.values())
9791063a081SSatish Balay    f.write('  configure_options = [\n')
9808bec23c5SJed Brown    for itm in sorted(args.values()):
9811063a081SSatish Balay      f.write('    \''+str(itm)+'\',\n')
9821063a081SSatish Balay    f.write('  ]\n')
983f8833479SBarry Smith    f.write('  configure.petsc_configure(configure_options)\n')
984f8833479SBarry Smith    f.close()
985f8833479SBarry Smith    try:
986f8833479SBarry Smith      os.chmod(scriptName, 0775)
987f8833479SBarry Smith    except OSError, e:
988f8833479SBarry Smith      self.framework.logPrint('Unable to make reconfigure script executable:\n'+str(e))
989f8833479SBarry Smith    self.framework.actions.addArgument('PETSc', 'File creation', 'Created '+scriptName+' for automatic reconfiguration')
990f8833479SBarry Smith    return
991f8833479SBarry Smith
992f8833479SBarry Smith  def configureInstall(self):
993f8833479SBarry Smith    '''Setup the directories for installation'''
994f8833479SBarry Smith    if self.framework.argDB['prefix']:
995824e893fSSatish Balay      self.addMakeRule('shared_install','',['-@echo "Now to install the libraries do:"',\
996d093bd8dSBarry Smith                                              '-@echo "'+self.installdir.installSudo+'make PETSC_DIR=${PETSC_DIR} PETSC_ARCH=${PETSC_ARCH} install"',\
997315b77e6SSatish Balay                                              '-@echo "========================================="'])
998f8833479SBarry Smith    else:
999824e893fSSatish Balay      self.addMakeRule('shared_install','',['-@echo "Now to check if the libraries are working do:"',\
1000824e893fSSatish Balay                                              '-@echo "make PETSC_DIR=${PETSC_DIR} PETSC_ARCH=${PETSC_ARCH} test"',\
1001315b77e6SSatish Balay                                              '-@echo "========================================="'])
1002f8833479SBarry Smith      return
1003f8833479SBarry Smith
1004f8833479SBarry Smith  def configureGCOV(self):
1005f8833479SBarry Smith    if self.framework.argDB['with-gcov']:
1006f8833479SBarry Smith      self.addDefine('USE_GCOV','1')
1007f8833479SBarry Smith    return
1008f8833479SBarry Smith
1009f8833479SBarry Smith  def configureFortranFlush(self):
1010f8833479SBarry Smith    if hasattr(self.compilers, 'FC'):
1011f8833479SBarry Smith      for baseName in ['flush','flush_']:
1012f8833479SBarry Smith        if self.libraries.check('', baseName, otherLibs = self.compilers.flibs, fortranMangle = 1):
1013f8833479SBarry Smith          self.addDefine('HAVE_'+baseName.upper(), 1)
1014f8833479SBarry Smith          return
1015f8833479SBarry Smith
101628bb2e72SSatish Balay  def postProcessPackages(self):
101728bb2e72SSatish Balay    postPackages=[]
101828bb2e72SSatish Balay    for i in self.framework.packages:
101928bb2e72SSatish Balay      if hasattr(i,'postProcess'): postPackages.append(i)
102028bb2e72SSatish Balay    if postPackages:
1021e64d19dfSSatish Balay      # ctetgen needs petsc conf files. so attempt to create them early
1022a77eb93bSSatish Balay      self.framework.dumpConfFiles()
1023d9293e7bSBarry Smith      # tacky fix for dependency of Aluimia on Pflotran; requested via petsc-dev Matt provide a correct fix
1024d9293e7bSBarry Smith      for i in postPackages:
1025d9293e7bSBarry Smith        if i.name.upper() in ['PFLOTRAN']:
1026d9293e7bSBarry Smith          i.postProcess()
1027d9293e7bSBarry Smith          postPackages.remove(i)
102828bb2e72SSatish Balay      for i in postPackages: i.postProcess()
102928bb2e72SSatish Balay    return
1030f8833479SBarry Smith
1031f8833479SBarry Smith  def configure(self):
1032f8833479SBarry Smith    if not os.path.samefile(self.petscdir.dir, os.getcwd()):
1033f8833479SBarry Smith      raise RuntimeError('Wrong PETSC_DIR option specified: '+str(self.petscdir.dir) + '\n  Configure invoked in: '+os.path.realpath(os.getcwd()))
1034550489e3SMatthew 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):
10353552d8fbSSatish Balay      raise RuntimeError('Incorrect option --prefix='+self.framework.argDB['prefix']+' specified. It cannot be same as PETSC_DIR!')
1036c16c35a9SSatish 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)):
1037c16c35a9SSatish Balay      raise RuntimeError('Incorrect option --prefix='+self.framework.argDB['prefix']+' specified. It cannot be same as PETSC_DIR/PETSC_ARCH!')
1038f16c1317SJed Brown    self.framework.header          = os.path.join(self.arch.arch,'include','petscconf.h')
1039f16c1317SJed Brown    self.framework.cHeader         = os.path.join(self.arch.arch,'include','petscfix.h')
1040af0996ceSBarry Smith    self.framework.makeMacroHeader = os.path.join(self.arch.arch,'lib','petsc','conf','petscvariables')
1041af0996ceSBarry Smith    self.framework.makeRuleHeader  = os.path.join(self.arch.arch,'lib','petsc','conf','petscrules')
1042f8833479SBarry Smith    if self.libraries.math is None:
1043f8833479SBarry Smith      raise RuntimeError('PETSc requires a functional math library. Please send configure.log to petsc-maint@mcs.anl.gov.')
1044f8833479SBarry Smith    if self.languages.clanguage == 'Cxx' and not hasattr(self.compilers, 'CXX'):
1045f8833479SBarry Smith      raise RuntimeError('Cannot set C language to C++ without a functional C++ compiler.')
1046ed938b00SJed Brown    self.executeTest(self.configureRTLDDefault)
1047b2843cf1SBarry Smith    self.executeTest(self.configurePrefetch)
10482400fdedSBarry Smith    self.executeTest(self.configureUnused)
10491ef8df7fSJed Brown    self.executeTest(self.configureDeprecated)
105098ed35c3SBarry Smith    self.executeTest(self.configureIsatty)
10519800092aSJed Brown    self.executeTest(self.configureExpect);
105218f41590SBarry Smith    self.executeTest(self.configureAlign);
105353c77d0aSJed Brown    self.executeTest(self.configureFunctionName);
1054753ebd1dSJed Brown    self.executeTest(self.configureIntptrt);
1055f8833479SBarry Smith    self.executeTest(self.configureSolaris)
1056f8833479SBarry Smith    self.executeTest(self.configureLinux)
1057f8833479SBarry Smith    self.executeTest(self.configureWin32)
1058b10d012aSSatish Balay    self.executeTest(self.configureCygwinBrokenPipe)
1059569865ddSSatish Balay    self.executeTest(self.configureDefaultArch)
1060f8833479SBarry Smith    self.executeTest(self.configureScript)
1061f8833479SBarry Smith    self.executeTest(self.configureInstall)
1062f8833479SBarry Smith    self.executeTest(self.configureGCOV)
1063f8833479SBarry Smith    self.executeTest(self.configureFortranFlush)
106409bc878fSSatish Balay    self.executeTest(self.configureAtoll)
1065f8833479SBarry Smith    # dummy rules, always needed except for remote builds
1066f8833479SBarry Smith    self.addMakeRule('remote','')
1067f8833479SBarry Smith    self.addMakeRule('remoteclean','')
1068f8833479SBarry Smith
1069f8833479SBarry Smith    self.Dump()
1070f8833479SBarry Smith    self.dumpConfigInfo()
10712a4161d9SMatthew G Knepley    self.dumpMachineInfo()
1072511a6afcSJed Brown    self.dumpCMakeConfig()
10738b0282a9SJed Brown    self.dumpCMakeLists()
107440277576SBarry Smith    # need to save the current state of BuildSystem so that postProcess() packages can read it in and perhaps run make install
107540277576SBarry Smith    self.framework.storeSubstitutions(self.framework.argDB)
107640277576SBarry Smith    self.framework.argDB['configureCache'] = cPickle.dumps(self.framework)
107740277576SBarry Smith    self.framework.argDB.save(force = True)
10788b0282a9SJed Brown    self.cmakeBoot()
1079262119f8SBarry Smith    self.DumpPkgconfig()
1080351d3a41SMatthew G Knepley    self.DumpModule()
1081f7ad81e1SBarry Smith    self.postProcessPackages()
1082f8833479SBarry Smith    self.framework.log.write('================================================================================\n')
1083f8833479SBarry Smith    self.logClear()
1084f8833479SBarry Smith    return
1085