xref: /petsc/config/PETSc/Configure.py (revision 7b7fc14bd8f0b740a95830e49598c8c8fec13c20)
1f8833479SBarry Smithimport config.base
2f8833479SBarry Smith
3f8833479SBarry Smithimport os
46dd73af6SBarry Smithimport sys
5f8833479SBarry Smithimport re
6492432c8SJed Brownimport pickle
7f8833479SBarry Smith
8f8833479SBarry Smithclass Configure(config.base.Configure):
9f8833479SBarry Smith  def __init__(self, framework):
10f8833479SBarry Smith    config.base.Configure.__init__(self, framework)
11f8833479SBarry Smith    self.headerPrefix = 'PETSC'
12f8833479SBarry Smith    self.substPrefix  = 'PETSC'
13aa5c8b8eSBarry Smith    self.installed = 0 # 1 indicates that Configure itself has already compiled and installed PETSc
14f8833479SBarry Smith    return
15f8833479SBarry Smith
167c939e48SSatish Balay  def __str2__(self):
177c939e48SSatish Balay    desc = []
18aa5c8b8eSBarry Smith    if not self.installed:
19a0022257SSatish Balay      desc.append('xxx=========================================================================xxx')
20ac1d0f13SJed Brown      if self.make.getMakeMacro('MAKE_IS_GNUMAKE'):
219481793eSSatish Balay        build_type = 'gnumake build'
229481793eSSatish Balay      elif self.getMakeMacro('PETSC_BUILD_USING_CMAKE'):
23b3618d6dSSatish Balay        build_type = 'cmake build'
24b3618d6dSSatish Balay      else:
25b3618d6dSSatish Balay        build_type = 'legacy build'
26b3618d6dSSatish Balay      desc.append(' Configure stage complete. Now build PETSc libraries with (%s):' % build_type)
27b3618d6dSSatish Balay      desc.append('   make PETSC_DIR='+self.petscdir.dir+' PETSC_ARCH='+self.arch.arch+' all')
28a0022257SSatish Balay      desc.append('xxx=========================================================================xxx')
29aa5c8b8eSBarry Smith    else:
30aa5c8b8eSBarry Smith      desc.append('xxx=========================================================================xxx')
31aa5c8b8eSBarry Smith      desc.append(' Installation complete. You do not need to run make to compile or install the software')
32aa5c8b8eSBarry Smith      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'))
42cb297985SSatish Balay    help.addArgument('PETSc','-with-fortran-bindings=<bool>',     nargs.ArgBool(None, 1,'Build PETSc fortran bindings in the library and corresponding module files'))
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'))
4513f9d092SSatish Balay    help.addArgument('PETSc', '-with-display=<x11display>',       nargs.Arg(None, '', 'Specifiy DISPLAY env variable for use with matlab test)'))
462c30b4dfSSatish Balay    help.addArgument('PETSc', '-with-package-scripts=<pyscripts>',nargs.ArgFileList(None,None,'Specify configure package scripts for user provided packages'))
47f8833479SBarry Smith    return
48f8833479SBarry Smith
496dd73af6SBarry Smith  def registerPythonFile(self,filename,directory):
506dd73af6SBarry Smith    ''' Add a python file to the framework and registers its headerprefix, ... externalpackagedir
516dd73af6SBarry Smith        directory is the directory where the file relative to the BuildSystem or config path in python notation with . '''
526dd73af6SBarry Smith    (utilityName, ext) = os.path.splitext(filename)
536dd73af6SBarry Smith    if not utilityName.startswith('.') and not utilityName.startswith('#') and ext == '.py' and not utilityName == '__init__':
546dd73af6SBarry Smith      if directory: directory = directory+'.'
556dd73af6SBarry Smith      utilityObj                             = self.framework.require(directory+utilityName, self)
566dd73af6SBarry Smith      utilityObj.headerPrefix                = self.headerPrefix
576dd73af6SBarry Smith      utilityObj.archProvider                = self.arch
586dd73af6SBarry Smith      utilityObj.languageProvider            = self.languages
596dd73af6SBarry Smith      utilityObj.installDirProvider          = self.installdir
606dd73af6SBarry Smith      utilityObj.externalPackagesDirProvider = self.externalpackagesdir
616dd73af6SBarry Smith      utilityObj.precisionProvider           = self.scalartypes
626dd73af6SBarry Smith      utilityObj.indexProvider               = self.indexTypes
636dd73af6SBarry Smith      setattr(self, utilityName.lower(), utilityObj)
6451294b80SMatthew G. Knepley      return utilityObj
6551294b80SMatthew G. Knepley    return None
666dd73af6SBarry Smith
67f8833479SBarry Smith  def setupDependencies(self, framework):
68f8833479SBarry Smith    config.base.Configure.setupDependencies(self, framework)
69dca78d2bSSatish Balay    self.programs      = framework.require('config.programs',           self)
70f8833479SBarry Smith    self.setCompilers  = framework.require('config.setCompilers',       self)
7130b8aa07SMatthew G. Knepley    self.compilers     = framework.require('config.compilers',          self)
729d310bb7SBarry Smith    self.arch          = framework.require('PETSc.options.arch',        self.setCompilers)
739d310bb7SBarry Smith    self.petscdir      = framework.require('PETSc.options.petscdir',    self.arch)
749d310bb7SBarry Smith    self.installdir    = framework.require('PETSc.options.installDir',  self)
756dd73af6SBarry Smith    self.scalartypes   = framework.require('PETSc.options.scalarTypes', self)
766dd73af6SBarry Smith    self.indexTypes    = framework.require('PETSc.options.indexTypes',  self)
779d310bb7SBarry Smith    self.languages     = framework.require('PETSc.options.languages',   self.setCompilers)
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
9109a6cbfcSBernhard M. Wiedemann    for utility in sorted(os.listdir(os.path.join('config','PETSc','options'))):
926dd73af6SBarry Smith      self.registerPythonFile(utility,'PETSc.options')
939d310bb7SBarry Smith
9409a6cbfcSBernhard M. Wiedemann    for utility in sorted(os.listdir(os.path.join('config','BuildSystem','config','utilities'))):
956dd73af6SBarry Smith      self.registerPythonFile(utility,'config.utilities')
9606e08bc7SBarry Smith
9709a6cbfcSBernhard M. Wiedemann    for package in sorted(os.listdir(os.path.join('config', 'BuildSystem', 'config', 'packages'))):
9851294b80SMatthew G. Knepley      obj = self.registerPythonFile(package,'config.packages')
9951294b80SMatthew G. Knepley      if obj:
10051294b80SMatthew G. Knepley        obj.archProvider                = self.framework.requireModule(obj.archProvider, obj)
10151294b80SMatthew G. Knepley        obj.languageProvider            = self.framework.requireModule(obj.languageProvider, obj)
10251294b80SMatthew G. Knepley        obj.installDirProvider          = self.framework.requireModule(obj.installDirProvider, obj)
10351294b80SMatthew G. Knepley        obj.externalPackagesDirProvider = self.framework.requireModule(obj.externalPackagesDirProvider, obj)
10451294b80SMatthew G. Knepley        obj.precisionProvider           = self.framework.requireModule(obj.precisionProvider, obj)
10551294b80SMatthew G. Knepley        obj.indexProvider               = self.framework.requireModule(obj.indexProvider, obj)
1066dd73af6SBarry Smith
1075faf1eacSMatthew G. Knepley    # Force blaslapack and opencl to depend on scalarType so precision is set before BlasLapack is built
1089d310bb7SBarry Smith    framework.require('PETSc.options.scalarTypes', self.f2cblaslapack)
1099d310bb7SBarry Smith    framework.require('PETSc.options.scalarTypes', self.fblaslapack)
1109d310bb7SBarry Smith    framework.require('PETSc.options.scalarTypes', self.blaslapack)
1115faf1eacSMatthew G. Knepley    framework.require('PETSc.options.scalarTypes', self.opencl)
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
1196dd73af6SBarry Smith
1202c30b4dfSSatish Balay    # Register user provided package scripts
1212c30b4dfSSatish Balay    if 'with-package-scripts' in self.framework.argDB:
1222c30b4dfSSatish Balay      for script in self.framework.argDB['with-package-scripts']:
1232c30b4dfSSatish Balay        if os.path.splitext(script)[1] != '.py':
1242c30b4dfSSatish Balay          raise RuntimeError('Only python scripts compatible with configure package script format should be specified! Invalid option -with-package-scripts='+script)
1252c30b4dfSSatish Balay        self.framework.logPrint('User is registering a new package script: '+script)
1262c30b4dfSSatish Balay        dname,fname = os.path.split(script)
1272c30b4dfSSatish Balay        if dname: sys.path.append(dname)
1282c30b4dfSSatish Balay        self.registerPythonFile(fname,'')
1296dd73af6SBarry Smith
1306dd73af6SBarry Smith    # test for a variety of basic headers and functions
131a8b45ee7SBarry Smith    headersC = map(lambda name: name+'.h', ['setjmp','dos', 'endian', 'fcntl', 'float', 'io', 'limits', 'malloc', 'pwd', 'search', 'strings',
132ba61063dSBarry Smith                                            'unistd', 'sys/sysinfo', 'machine/endian', 'sys/param', 'sys/procfs', 'sys/resource',
133a3aaec0aSJed Brown                                            'sys/systeminfo', 'sys/times', 'sys/utsname','string', 'stdlib',
134f8833479SBarry Smith                                            'sys/socket','sys/wait','netinet/in','netdb','Direct','time','Ws2tcpip','sys/types',
135ffebf242SSatish Balay                                            'WindowsX', 'cxxabi','float','ieeefp','stdint','sched','pthread','inttypes','immintrin','zmmintrin'])
13645082d64SJed Brown    functions = ['access', '_access', 'clock', 'drand48', 'getcwd', '_getcwd', 'getdomainname', 'gethostname',
1379dcccc9aSSatish Balay                 'gettimeofday', 'getwd', 'memalign', 'mkstemp', 'popen', 'PXFGETARG', 'rand', 'getpagesize',
13838ecfe64SSatish Balay                 'readlink', 'realpath',  'sigaction', 'signal', 'sigset', 'usleep', 'sleep', '_sleep', 'socket',
139473bb0d5SSatish Balay                 'times', 'gethostbyname', 'uname','snprintf','_snprintf','lseek','_lseek','time','fork','stricmp',
140ac7218bbSSatish Balay                 'strcasecmp', 'bzero', 'dlopen', 'dlsym', 'dlclose', 'dlerror','get_nprocs','sysctlbyname',
1410787ed6cSSatish Balay                 '_set_output_format','_mkdir']
142f8833479SBarry Smith    libraries1 = [(['socket', 'nsl'], 'socket'), (['fpe'], 'handle_sigfpes')]
143f8833479SBarry Smith    self.headers.headers.extend(headersC)
144f8833479SBarry Smith    self.functions.functions.extend(functions)
145f8833479SBarry Smith    self.libraries.libraries.extend(libraries1)
1467d421530SBarry Smith
147f8833479SBarry Smith    return
148f8833479SBarry Smith
149262119f8SBarry Smith  def DumpPkgconfig(self):
150262119f8SBarry Smith    ''' Create a pkg-config file '''
151262119f8SBarry Smith    if not os.path.exists(os.path.join(self.petscdir.dir,self.arch.arch,'lib','pkgconfig')):
152262119f8SBarry Smith      os.makedirs(os.path.join(self.petscdir.dir,self.arch.arch,'lib','pkgconfig'))
153262119f8SBarry Smith    fd = open(os.path.join(self.petscdir.dir,self.arch.arch,'lib','pkgconfig','PETSc.pc'),'w')
1545e3311eeSJed Brown    cflags_inc = ['-I${includedir}']
155262119f8SBarry Smith    if self.framework.argDB['prefix']:
1565bb5b263SMatthew G. Knepley      fd.write('prefix='+self.installdir.dir+'\n')
157262119f8SBarry Smith    else:
158e1e675deSJed Brown      fd.write('prefix='+os.path.join(self.petscdir.dir, self.arch.arch)+'\n')
159e1e675deSJed Brown      cflags_inc.append('-I' + os.path.join(self.petscdir.dir, 'include'))
160262119f8SBarry Smith    fd.write('exec_prefix=${prefix}\n')
161262119f8SBarry Smith    fd.write('includedir=${prefix}/include\n')
1625e3311eeSJed Brown    fd.write('libdir=${prefix}/lib\n')
163262119f8SBarry Smith
164262119f8SBarry Smith    self.setCompilers.pushLanguage('C')
165262119f8SBarry Smith    fd.write('ccompiler='+self.setCompilers.getCompiler()+'\n')
166756c7f9fSJed Brown    fd.write('cflags_extra='+self.setCompilers.getCompilerFlags().strip()+'\n')
167756c7f9fSJed Brown    fd.write('cflags_dep='+self.compilers.dependenciesGenerationFlag.get('C','')+'\n')
168756c7f9fSJed Brown    fd.write('ldflag_rpath='+self.setCompilers.CSharedLinkerFlag+'\n')
169262119f8SBarry Smith    self.setCompilers.popLanguage()
17003e383c8SJed Brown    if hasattr(self.compilers, 'CXX'):
171262119f8SBarry Smith      self.setCompilers.pushLanguage('C++')
172262119f8SBarry Smith      fd.write('cxxcompiler='+self.setCompilers.getCompiler()+'\n')
173756c7f9fSJed Brown      fd.write('cxxflags_extra='+self.setCompilers.getCompilerFlags().strip()+'\n')
174262119f8SBarry Smith      self.setCompilers.popLanguage()
175262119f8SBarry Smith    if hasattr(self.compilers, 'FC'):
176262119f8SBarry Smith      self.setCompilers.pushLanguage('FC')
177262119f8SBarry Smith      fd.write('fcompiler='+self.setCompilers.getCompiler()+'\n')
178756c7f9fSJed Brown      fd.write('fflags_extra='+self.setCompilers.getCompilerFlags().strip()+'\n')
179262119f8SBarry Smith      self.setCompilers.popLanguage()
180262119f8SBarry Smith
181262119f8SBarry Smith    fd.write('\n')
182262119f8SBarry Smith    fd.write('Name: PETSc\n')
183262119f8SBarry Smith    fd.write('Description: Library to solve ODEs and algebraic equations\n')
184351d3a41SMatthew G Knepley    fd.write('Version: %s\n' % self.petscdir.version)
1855e3311eeSJed Brown    fd.write('Cflags: ' + ' '.join([self.setCompilers.CPPFLAGS] + cflags_inc) + '\n')
18637371b91SJed Brown    fd.write('Libs: '+self.libraries.toStringNoDupes(['-L${libdir}', self.petsclib], with_rpath=False)+'\n')
1878ebf8858SJed Brown    # Remove RPATH flags from library list.  User can add them using
1888ebf8858SJed Brown    # pkg-config --variable=ldflag_rpath and pkg-config --libs-only-L
189de8f682fSSatish Balay    fd.write('Libs.private: '+self.libraries.toStringNoDupes([f for f in self.packagelibs+self.complibs if not f.startswith(self.setCompilers.CSharedLinkerFlag)], with_rpath=False)+'\n')
190473a3ab2SBarry Smith
191262119f8SBarry Smith    fd.close()
192262119f8SBarry Smith    return
193262119f8SBarry Smith
194351d3a41SMatthew G Knepley  def DumpModule(self):
195351d3a41SMatthew G Knepley    ''' Create a module file '''
196af0996ceSBarry Smith    if not os.path.exists(os.path.join(self.petscdir.dir,self.arch.arch,'lib','petsc','conf','modules')):
197af0996ceSBarry Smith      os.makedirs(os.path.join(self.petscdir.dir,self.arch.arch,'lib','petsc','conf','modules'))
198af0996ceSBarry Smith    if not os.path.exists(os.path.join(self.petscdir.dir,self.arch.arch,'lib','petsc','conf','modules','petsc')):
199af0996ceSBarry Smith      os.makedirs(os.path.join(self.petscdir.dir,self.arch.arch,'lib','petsc','conf','modules','petsc'))
200351d3a41SMatthew G Knepley    if self.framework.argDB['prefix']:
2015bb5b263SMatthew G. Knepley      installdir  = self.installdir.dir
20255d606a3SSatish Balay      installarch = ''
20355d606a3SSatish Balay      installpath = os.path.join(installdir,'bin')
204351d3a41SMatthew G Knepley    else:
205351d3a41SMatthew G Knepley      installdir  = self.petscdir.dir
20655d606a3SSatish Balay      installarch = self.arch.arch
20755d606a3SSatish Balay      installpath = os.path.join(installdir,installarch,'bin')+':'+os.path.join(installdir,'bin')
208af0996ceSBarry Smith    fd = open(os.path.join(self.petscdir.dir,self.arch.arch,'lib','petsc','conf','modules','petsc',self.petscdir.version),'w')
209351d3a41SMatthew G Knepley    fd.write('''\
210351d3a41SMatthew G Knepley#%%Module
211351d3a41SMatthew G Knepley
212351d3a41SMatthew G Knepleyproc ModulesHelp { } {
213351d3a41SMatthew G Knepley    puts stderr "This module sets the path and environment variables for petsc-%s"
214a8d69d7bSBarry Smith    puts stderr "     see https://www.mcs.anl.gov/petsc/ for more information      "
215351d3a41SMatthew G Knepley    puts stderr ""
216351d3a41SMatthew G Knepley}
217351d3a41SMatthew G Knepleymodule-whatis "PETSc - Portable, Extensible Toolkit for Scientific Computation"
218351d3a41SMatthew G Knepley
219dd486775SJed Brownset petsc_dir   "%s"
220dd486775SJed Brownset petsc_arch  "%s"
221351d3a41SMatthew G Knepley
222dd486775SJed Brownsetenv PETSC_ARCH "$petsc_arch"
223dd486775SJed Brownsetenv PETSC_DIR "$petsc_dir"
224dd486775SJed Brownprepend-path PATH "%s"
22555d606a3SSatish Balay''' % (self.petscdir.version, installdir, installarch, installpath))
226351d3a41SMatthew G Knepley    fd.close()
227351d3a41SMatthew G Knepley    return
228351d3a41SMatthew G Knepley
229f8833479SBarry Smith  def Dump(self):
230f8833479SBarry Smith    ''' Actually put the values into the configuration files '''
231f8833479SBarry Smith    # eventually everything between -- should be gone
23217f368bcSBarry Smith    if self.mpi.usingMPIUni:
23317f368bcSBarry Smith      #
23417f368bcSBarry Smith      # Remove any MPI/MPICH include files that may have been put here by previous runs of ./configure
2357908f030SMatthew 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)
23617f368bcSBarry Smith
2375f27b2e0SBarry Smith    self.setCompilers.pushLanguage('C')
2385f27b2e0SBarry Smith    compiler = self.setCompilers.getCompiler()
239c9872b61SBarry Smith    if compiler.endswith('mpicc') or compiler.endswith('mpiicc'):
2405f27b2e0SBarry Smith      try:
2415f27b2e0SBarry Smith        output   = self.executeShellCommand(compiler + ' -show', log = self.log)[0]
2425f27b2e0SBarry Smith        compiler = output.split(' ')[0]
2435f27b2e0SBarry Smith        self.addDefine('MPICC_SHOW','"'+output.strip().replace('\n','\\\\n')+'"')
2445f27b2e0SBarry Smith      except:
245c9872b61SBarry Smith        self.addDefine('MPICC_SHOW','"Unavailable"')
246c9872b61SBarry Smith    else:
247c9872b61SBarry Smith      self.addDefine('MPICC_SHOW','"Unavailable"')
2485f27b2e0SBarry Smith    self.setCompilers.popLanguage()
249f8833479SBarry Smith#-----------------------------------------------------------------------------------------------------
250f8833479SBarry Smith
251f8833479SBarry Smith    # Sometimes we need C compiler, even if built with C++
252f8833479SBarry Smith    self.setCompilers.pushLanguage('C')
253f8833479SBarry Smith    self.addMakeMacro('CC_FLAGS',self.setCompilers.getCompilerFlags())
254f8833479SBarry Smith    self.setCompilers.popLanguage()
255f8833479SBarry Smith
25634f774f6SJed Brown    # And sometimes we need a C++ compiler even when PETSc is built with C
25734f774f6SJed Brown    if hasattr(self.compilers, 'CXX'):
25834f774f6SJed Brown      self.setCompilers.pushLanguage('Cxx')
25929921a8fSScott Kruger      self.addDefine('HAVE_CXX','1')
26034f774f6SJed Brown      self.addMakeMacro('CXX_FLAGS',self.setCompilers.getCompilerFlags())
2612f4326f3SSatish Balay      cxx_linker = self.setCompilers.getLinker()
2622f4326f3SSatish Balay      self.addMakeMacro('CXX_LINKER',cxx_linker)
2632f4326f3SSatish Balay      self.addMakeMacro('CXX_LINKER_FLAGS',self.setCompilers.getLinkerFlags())
26434f774f6SJed Brown      self.setCompilers.popLanguage()
26534f774f6SJed Brown
266f8833479SBarry Smith    # C preprocessor values
2671315f054SBarry Smith    self.addMakeMacro('CPP_FLAGS',self.setCompilers.CPPFLAGS)
268f8833479SBarry Smith
269f8833479SBarry Smith    # compiler values
270f8833479SBarry Smith    self.setCompilers.pushLanguage(self.languages.clanguage)
271f8833479SBarry Smith    self.addMakeMacro('PCC',self.setCompilers.getCompiler())
272f8833479SBarry Smith    self.addMakeMacro('PCC_FLAGS',self.setCompilers.getCompilerFlags())
273f8833479SBarry Smith    self.setCompilers.popLanguage()
274f8833479SBarry Smith    # .o or .obj
275f8833479SBarry Smith    self.addMakeMacro('CC_SUFFIX','o')
276f8833479SBarry Smith
277f8833479SBarry Smith    # executable linker values
278f8833479SBarry Smith    self.setCompilers.pushLanguage(self.languages.clanguage)
279f8833479SBarry Smith    pcc_linker = self.setCompilers.getLinker()
280f8833479SBarry Smith    self.addMakeMacro('PCC_LINKER',pcc_linker)
281c84a332bSSatish Balay    self.addMakeMacro('PCC_LINKER_FLAGS',self.setCompilers.getLinkerFlags())
282f8833479SBarry Smith    self.setCompilers.popLanguage()
283f8833479SBarry Smith    # '' for Unix, .exe for Windows
284f8833479SBarry Smith    self.addMakeMacro('CC_LINKER_SUFFIX','')
285f8833479SBarry Smith
286f8833479SBarry Smith    if hasattr(self.compilers, 'FC'):
287cb297985SSatish Balay      if self.framework.argDB['with-fortran-bindings']:
288cb297985SSatish Balay        self.addDefine('HAVE_FORTRAN','1')
289f8833479SBarry Smith      self.setCompilers.pushLanguage('FC')
290f8833479SBarry Smith      # need FPPFLAGS in config/setCompilers
291f8833479SBarry Smith      self.addMakeMacro('FPP_FLAGS',self.setCompilers.CPPFLAGS)
292f8833479SBarry Smith
293f8833479SBarry Smith      # compiler values
294f8833479SBarry Smith      self.addMakeMacro('FC_FLAGS',self.setCompilers.getCompilerFlags())
295f8833479SBarry Smith      self.setCompilers.popLanguage()
296f8833479SBarry Smith      # .o or .obj
297f8833479SBarry Smith      self.addMakeMacro('FC_SUFFIX','o')
298f8833479SBarry Smith
299f8833479SBarry Smith      # executable linker values
300f8833479SBarry Smith      self.setCompilers.pushLanguage('FC')
301f8833479SBarry Smith      # Cannot have NAG f90 as the linker - so use pcc_linker as fc_linker
302f8833479SBarry Smith      fc_linker = self.setCompilers.getLinker()
3037fca349cSMatthew G. Knepley      if config.setCompilers.Configure.isNAG(fc_linker, self.log):
304f8833479SBarry Smith        self.addMakeMacro('FC_LINKER',pcc_linker)
305f8833479SBarry Smith      else:
306f8833479SBarry Smith        self.addMakeMacro('FC_LINKER',fc_linker)
3076d53d35eSSatish Balay      self.addMakeMacro('FC_LINKER_FLAGS',self.setCompilers.getLinkerFlags())
3083feacd00SBarry Smith      # apple requires this shared library linker flag on SOME versions of the os
3093feacd00SBarry Smith      if self.setCompilers.getLinkerFlags().find('-Wl,-commons,use_dylibs') > -1:
3103feacd00SBarry Smith        self.addMakeMacro('DARWIN_COMMONS_USE_DYLIBS',' -Wl,-commons,use_dylibs ')
311bb82cf9cSSatish Balay      self.setCompilers.popLanguage()
3125d631499SMatthew Knepley
3135d631499SMatthew Knepley      # F90 Modules
3145d631499SMatthew Knepley      if self.setCompilers.fortranModuleIncludeFlag:
3155d631499SMatthew Knepley        self.addMakeMacro('FC_MODULE_FLAG', self.setCompilers.fortranModuleIncludeFlag)
3166ddd6694SSatish Balay      else: # for non-f90 compilers like g77
3176ddd6694SSatish Balay        self.addMakeMacro('FC_MODULE_FLAG', '-I')
318a324c51cSMatthew G Knepley      if self.setCompilers.fortranModuleIncludeFlag:
319a324c51cSMatthew G Knepley        self.addMakeMacro('FC_MODULE_OUTPUT_FLAG', self.setCompilers.fortranModuleOutputFlag)
320f8833479SBarry Smith    else:
321f8833479SBarry Smith      self.addMakeMacro('FC','')
322f8833479SBarry Smith
32346a3958fSBarry Smith    if hasattr(self.compilers, 'CUDAC'):
3247ff2890cSSatish Balay      self.setCompilers.pushLanguage('CUDA')
325d93a25ecSSatish Balay      self.addMakeMacro('CUDAC_FLAGS',self.setCompilers.getCompilerFlags())
3267ff2890cSSatish Balay      self.setCompilers.popLanguage()
3277ff2890cSSatish Balay
328f8833479SBarry Smith    # shared library linker values
329f8833479SBarry Smith    self.setCompilers.pushLanguage(self.languages.clanguage)
330f8833479SBarry Smith    # need to fix BuildSystem to collect these separately
331f8833479SBarry Smith    self.addMakeMacro('SL_LINKER',self.setCompilers.getLinker())
33270db8aa6SSatish Balay    self.addMakeMacro('SL_LINKER_FLAGS','${PCC_LINKER_FLAGS}')
333f8833479SBarry Smith    self.setCompilers.popLanguage()
334f8833479SBarry Smith    # One of 'a', 'so', 'lib', 'dll', 'dylib' (perhaps others also?) depending on the library generator and architecture
335f8833479SBarry Smith    # Note: . is not included in this macro, consistent with AR_LIB_SUFFIX
336f8833479SBarry Smith    if self.setCompilers.sharedLibraryExt == self.setCompilers.AR_LIB_SUFFIX:
337f8833479SBarry Smith      self.addMakeMacro('SL_LINKER_SUFFIX', '')
33846bc77b6SBarry Smith      self.addDefine('SLSUFFIX','""')
339f8833479SBarry Smith    else:
340f8833479SBarry Smith      self.addMakeMacro('SL_LINKER_SUFFIX', self.setCompilers.sharedLibraryExt)
34146bc77b6SBarry Smith      self.addDefine('SLSUFFIX','"'+self.setCompilers.sharedLibraryExt+'"')
342bb82cf9cSSatish Balay
34323e93537SBarry Smith    self.addMakeMacro('SL_LINKER_LIBS','${PETSC_EXTERNAL_LIB_BASIC}')
344bb82cf9cSSatish Balay
345f8833479SBarry Smith#-----------------------------------------------------------------------------------------------------
346f8833479SBarry Smith
347f8833479SBarry Smith    # CONLY or CPP. We should change the PETSc makefiles to do this better
348f8833479SBarry Smith    if self.languages.clanguage == 'C': lang = 'CONLY'
349f8833479SBarry Smith    else: lang = 'CXXONLY'
350f8833479SBarry Smith    self.addMakeMacro('PETSC_LANGUAGE',lang)
351f8833479SBarry Smith
352f8833479SBarry Smith    # real or complex
353f8833479SBarry Smith    self.addMakeMacro('PETSC_SCALAR',self.scalartypes.scalartype)
354f8833479SBarry Smith    # double or float
355f8833479SBarry Smith    self.addMakeMacro('PETSC_PRECISION',self.scalartypes.precision)
356f8833479SBarry Smith
357f8833479SBarry Smith    if self.framework.argDB['with-batch']:
358f8833479SBarry Smith      self.addMakeMacro('PETSC_WITH_BATCH','1')
359f8833479SBarry Smith
360f8833479SBarry Smith    # Test for compiler-specific macros that need to be defined.
3617fca349cSMatthew G. Knepley    if self.setCompilers.isCrayVector('CC', self.log):
362b409243cSBarry Smith      self.addDefine('HAVE_CRAY_VECTOR','1')
363f8833479SBarry Smith
364f8833479SBarry Smith#-----------------------------------------------------------------------------------------------------
365df1a78b3SMatthew G Knepley    if self.functions.haveFunction('gethostbyname') and self.functions.haveFunction('socket') and self.headers.haveHeader('netinet/in.h'):
366f8833479SBarry Smith      self.addDefine('USE_SOCKET_VIEWER','1')
36780e3883bSBarry Smith      if self.checkCompile('#include <sys/socket.h>','setsockopt(0,SOL_SOCKET,SO_REUSEADDR,0,0)'):
36880e3883bSBarry Smith        self.addDefine('HAVE_SO_REUSEADDR','1')
369f8833479SBarry Smith
370f8833479SBarry Smith#-----------------------------------------------------------------------------------------------------
371a6cc6bb1SBarry Smith    # print include and lib for makefiles
372f8833479SBarry Smith    self.framework.packages.reverse()
3735a21677cSJed Brown    petscincludes = [os.path.join(self.petscdir.dir,'include'),os.path.join(self.petscdir.dir,self.arch.arch,'include')]
3745a21677cSJed Brown    petscincludes_install = [os.path.join(self.installdir.dir, 'include')] if self.framework.argDB['prefix'] else petscincludes
3755a21677cSJed Brown    includes = []
376de8f682fSSatish Balay    self.packagelibs = []
377f8833479SBarry Smith    for i in self.framework.packages:
378898a086dSBarry Smith      if i.useddirectly:
379eeb16384SBarry Smith        self.addDefine('HAVE_'+i.PACKAGE.replace('-','_'), 1)  # ONLY list package if it is used directly by PETSc (and not only by another package)
380f8833479SBarry Smith      if not isinstance(i.lib, list):
381f8833479SBarry Smith        i.lib = [i.lib]
382de8f682fSSatish Balay      if i.linkedbypetsc: self.packagelibs.extend(i.lib)
383eeb16384SBarry Smith      self.addMakeMacro(i.PACKAGE.replace('-','_')+'_LIB', self.libraries.toStringNoDupes(i.lib))
384f8833479SBarry Smith      if hasattr(i,'include'):
385f8833479SBarry Smith        if not isinstance(i.include,list):
386f8833479SBarry Smith          i.include = [i.include]
387ac9e4c42SSatish Balay        includes.extend(i.include)
388eeb16384SBarry Smith        self.addMakeMacro(i.PACKAGE.replace('-','_')+'_INCLUDE',self.headers.toStringNoDupes(i.include))
3892df986feSBarry Smith    if self.framework.argDB['with-single-library']:
390e282ce78SJed Brown      self.petsclib = '-lpetsc'
39191bb3077SSatish Balay    else:
392e282ce78SJed Brown      self.petsclib = '-lpetscts -lpetscsnes -lpetscksp -lpetscdm -lpetscmat -lpetscvec -lpetscsys'
393de8f682fSSatish Balay    self.complibs = self.compilers.flibs+self.compilers.cxxlibs+self.compilers.LIBS.split()
3945a21677cSJed Brown    self.PETSC_WITH_EXTERNAL_LIB = self.libraries.toStringNoDupes(['-L${PETSC_DIR}/${PETSC_ARCH}/lib', self.petsclib]+self.packagelibs+self.complibs)
395de8f682fSSatish Balay    self.PETSC_EXTERNAL_LIB_BASIC = self.libraries.toStringNoDupes(self.packagelibs+self.complibs)
396de8f682fSSatish Balay
397de8f682fSSatish Balay    self.addMakeMacro('PETSC_EXTERNAL_LIB_BASIC',self.PETSC_EXTERNAL_LIB_BASIC)
3985a21677cSJed Brown    allincludes = petscincludes + includes
3995a21677cSJed Brown    allincludes_install = petscincludes_install + includes
4005a21677cSJed Brown    self.PETSC_CC_INCLUDES = self.headers.toStringNoDupes(allincludes)
4015a21677cSJed Brown    self.PETSC_CC_INCLUDES_INSTALL = self.headers.toStringNoDupes(allincludes_install)
4025a21677cSJed Brown    self.addMakeMacro('PETSC_CC_INCLUDES',self.PETSC_CC_INCLUDES)
4035a21677cSJed Brown    self.addMakeMacro('PETSC_CC_INCLUDES_INSTALL', self.PETSC_CC_INCLUDES_INSTALL)
404cbd5cc15SBarry Smith    if hasattr(self.compilers, 'FC'):
4055a21677cSJed Brown      def modinc(includes):
4065a21677cSJed Brown        return includes if self.compilers.fortranIsF90 else []
4075a21677cSJed Brown      self.addMakeMacro('PETSC_FC_INCLUDES',self.headers.toStringNoDupes(allincludes,modinc(allincludes)))
4085a21677cSJed Brown      self.addMakeMacro('PETSC_FC_INCLUDES_INSTALL',self.headers.toStringNoDupes(allincludes_install,modinc(allincludes_install)))
409f8833479SBarry Smith
4105bb5b263SMatthew G. Knepley    self.addDefine('LIB_DIR','"'+os.path.join(self.installdir.dir,'lib')+'"')
411f8833479SBarry Smith
4120f3b21c2SBarry Smith    if self.framework.argDB['with-single-library']:
4130f3b21c2SBarry Smith      # overrides the values set in conf/variables
4140f3b21c2SBarry Smith      self.addMakeMacro('LIBNAME','${INSTALL_LIB_DIR}/libpetsc.${AR_LIB_SUFFIX}')
41557cb31baSSatish Balay      self.addMakeMacro('SHLIBS','libpetsc')
416bccf1c12SBarry Smith      self.addMakeMacro('PETSC_LIB_BASIC','-lpetsc')
417797063a9SSatish Balay      self.addMakeMacro('PETSC_KSP_LIB_BASIC','-lpetsc')
418797063a9SSatish Balay      self.addMakeMacro('PETSC_TS_LIB_BASIC','-lpetsc')
419b0a7d7e7SSatish Balay      self.addMakeMacro('PETSC_TAO_LIB_BASIC','-lpetsc')
420de8f682fSSatish Balay      self.addMakeMacro('PETSC_WITH_EXTERNAL_LIB',self.PETSC_WITH_EXTERNAL_LIB)
421bb84e0fdSBarry Smith      self.addDefine('USE_SINGLE_LIBRARY', '1')
4222df986feSBarry Smith      if self.sharedlibraries.useShared:
423ea820d49SSatish Balay        self.addMakeMacro('PETSC_SYS_LIB','${C_SH_LIB_PATH} ${PETSC_WITH_EXTERNAL_LIB}')
424ea820d49SSatish Balay        self.addMakeMacro('PETSC_VEC_LIB','${C_SH_LIB_PATH} ${PETSC_WITH_EXTERNAL_LIB}')
425ea820d49SSatish Balay        self.addMakeMacro('PETSC_MAT_LIB','${C_SH_LIB_PATH} ${PETSC_WITH_EXTERNAL_LIB}')
426ea820d49SSatish Balay        self.addMakeMacro('PETSC_DM_LIB','${C_SH_LIB_PATH} ${PETSC_WITH_EXTERNAL_LIB}')
427ea820d49SSatish Balay        self.addMakeMacro('PETSC_KSP_LIB','${C_SH_LIB_PATH} ${PETSC_WITH_EXTERNAL_LIB}')
428ea820d49SSatish Balay        self.addMakeMacro('PETSC_SNES_LIB','${C_SH_LIB_PATH} ${PETSC_WITH_EXTERNAL_LIB}')
429ea820d49SSatish Balay        self.addMakeMacro('PETSC_TS_LIB','${C_SH_LIB_PATH} ${PETSC_WITH_EXTERNAL_LIB}')
430b0a7d7e7SSatish Balay        self.addMakeMacro('PETSC_TAO_LIB','${C_SH_LIB_PATH} ${PETSC_WITH_EXTERNAL_LIB}')
431fdb87e33SJed Brown        self.addMakeMacro('PETSC_CHARACTERISTIC_LIB','${C_SH_LIB_PATH} ${PETSC_WITH_EXTERNAL_LIB}')
432ea820d49SSatish Balay        self.addMakeMacro('PETSC_LIB','${C_SH_LIB_PATH} ${PETSC_WITH_EXTERNAL_LIB}')
433ea820d49SSatish Balay        self.addMakeMacro('PETSC_CONTRIB','${C_SH_LIB_PATH} ${PETSC_WITH_EXTERNAL_LIB}')
4342df986feSBarry Smith      else:
435ea820d49SSatish Balay        self.addMakeMacro('PETSC_SYS_LIB','${PETSC_WITH_EXTERNAL_LIB}')
436ea820d49SSatish Balay        self.addMakeMacro('PETSC_VEC_LIB','${PETSC_WITH_EXTERNAL_LIB}')
437ea820d49SSatish Balay        self.addMakeMacro('PETSC_MAT_LIB','${PETSC_WITH_EXTERNAL_LIB}')
438ea820d49SSatish Balay        self.addMakeMacro('PETSC_DM_LIB','${PETSC_WITH_EXTERNAL_LIB}')
439ea820d49SSatish Balay        self.addMakeMacro('PETSC_KSP_LIB','${PETSC_WITH_EXTERNAL_LIB}')
440ea820d49SSatish Balay        self.addMakeMacro('PETSC_SNES_LIB','${PETSC_WITH_EXTERNAL_LIB}')
441ea820d49SSatish Balay        self.addMakeMacro('PETSC_TS_LIB','${PETSC_WITH_EXTERNAL_LIB}')
442b0a7d7e7SSatish Balay        self.addMakeMacro('PETSC_TAO_LIB','${PETSC_WITH_EXTERNAL_LIB}')
443fdb87e33SJed Brown        self.addMakeMacro('PETSC_CHARACTERISTIC_LIB','${PETSC_WITH_EXTERNAL_LIB}')
444ea820d49SSatish Balay        self.addMakeMacro('PETSC_LIB','${PETSC_WITH_EXTERNAL_LIB}')
445ea820d49SSatish Balay        self.addMakeMacro('PETSC_CONTRIB','${PETSC_WITH_EXTERNAL_LIB}')
4460f3b21c2SBarry Smith
447f8833479SBarry Smith    if not os.path.exists(os.path.join(self.petscdir.dir,self.arch.arch,'lib')):
448f8833479SBarry Smith      os.makedirs(os.path.join(self.petscdir.dir,self.arch.arch,'lib'))
449f8833479SBarry Smith
45013f9d092SSatish Balay# add a makefile endtry for display
45113f9d092SSatish Balay    if self.framework.argDB['with-display']:
45213f9d092SSatish Balay      self.addMakeMacro('DISPLAY',self.framework.argDB['with-display'])
45313f9d092SSatish Balay
454f8833479SBarry Smith    # add a makefile entry for configure options
455f8833479SBarry Smith    self.addMakeMacro('CONFIGURE_OPTIONS', self.framework.getOptionsString(['configModules', 'optionsModule']).replace('\"','\\"'))
456f8833479SBarry Smith    return
457f8833479SBarry Smith
458f8833479SBarry Smith  def dumpConfigInfo(self):
459f8833479SBarry Smith    import time
460c6ef1b5bSJed Brown    fd = open(os.path.join(self.arch.arch,'include','petscconfiginfo.h'),'w')
461f8833479SBarry Smith    fd.write('static const char *petscconfigureoptions = "'+self.framework.getOptionsString(['configModules', 'optionsModule']).replace('\"','\\"')+'";\n')
462f8833479SBarry Smith    fd.close()
463f8833479SBarry Smith    return
464f8833479SBarry Smith
4652a4161d9SMatthew G Knepley  def dumpMachineInfo(self):
4662a4161d9SMatthew G Knepley    import platform
467a970bd74SBernhard M. Wiedemann    import datetime
4682a4161d9SMatthew G Knepley    import time
46940373944SSatish Balay    import script
470ca77dbeeSGeoffrey Irving    def escape(s):
471ca77dbeeSGeoffrey Irving      return s.replace('"',r'\"').replace(r'\ ',r'\\ ')
472c6ef1b5bSJed Brown    fd = open(os.path.join(self.arch.arch,'include','petscmachineinfo.h'),'w')
4732a4161d9SMatthew G Knepley    fd.write('static const char *petscmachineinfo = \"\\n\"\n')
4742a4161d9SMatthew G Knepley    fd.write('\"-----------------------------------------\\n\"\n')
475a970bd74SBernhard M. Wiedemann    buildhost = platform.node()
476a970bd74SBernhard M. Wiedemann    if os.environ.get('SOURCE_DATE_EPOCH'):
477a970bd74SBernhard M. Wiedemann      buildhost = "reproducible"
478a970bd74SBernhard M. Wiedemann    buildtime = datetime.datetime.utcfromtimestamp(int(os.environ.get('SOURCE_DATE_EPOCH', time.time())))
479a970bd74SBernhard M. Wiedemann    fd.write('\"Libraries compiled on %s on %s \\n\"\n' % (buildtime, buildhost))
48060acdfe7SSatish Balay    fd.write('\"Machine characteristics: %s\\n\"\n' % (platform.platform()))
4815188cb68SSatish Balay    fd.write('\"Using PETSc directory: %s\\n\"\n' % (escape(self.installdir.petscDir)))
4825188cb68SSatish Balay    fd.write('\"Using PETSc arch: %s\\n\"\n' % (escape(self.installdir.petscArch)))
483cdec380aSBarry Smith    fd.write('\"-----------------------------------------\\n\";\n')
4842a4161d9SMatthew G Knepley    fd.write('static const char *petsccompilerinfo = \"\\n\"\n')
4852a4161d9SMatthew G Knepley    self.setCompilers.pushLanguage(self.languages.clanguage)
4865f27b2e0SBarry Smith    fd.write('\"Using C compiler: %s %s \\n\"\n' % (escape(self.setCompilers.getCompiler()), escape(self.setCompilers.getCompilerFlags())))
4872a4161d9SMatthew G Knepley    self.setCompilers.popLanguage()
4888782282cSMatthew G Knepley    if hasattr(self.compilers, 'FC'):
4892a4161d9SMatthew G Knepley      self.setCompilers.pushLanguage('FC')
4905f27b2e0SBarry Smith      fd.write('\"Using Fortran compiler: %s %s  %s\\n\"\n' % (escape(self.setCompilers.getCompiler()), escape(self.setCompilers.getCompilerFlags()), escape(self.setCompilers.CPPFLAGS)))
4912a4161d9SMatthew G Knepley      self.setCompilers.popLanguage()
492cdec380aSBarry Smith    fd.write('\"-----------------------------------------\\n\";\n')
4932a4161d9SMatthew G Knepley    fd.write('static const char *petsccompilerflagsinfo = \"\\n\"\n')
4945a21677cSJed Brown    fd.write('\"Using include paths: %s\\n\"\n' % (escape(self.PETSC_CC_INCLUDES_INSTALL.replace('${PETSC_DIR}', self.installdir.petscDir))))
495cdec380aSBarry Smith    fd.write('\"-----------------------------------------\\n\";\n')
4962a4161d9SMatthew G Knepley    fd.write('static const char *petsclinkerinfo = \"\\n\"\n')
4972a4161d9SMatthew G Knepley    self.setCompilers.pushLanguage(self.languages.clanguage)
498ca77dbeeSGeoffrey Irving    fd.write('\"Using C linker: %s\\n\"\n' % (escape(self.setCompilers.getLinker())))
4992a4161d9SMatthew G Knepley    self.setCompilers.popLanguage()
5008782282cSMatthew G Knepley    if hasattr(self.compilers, 'FC'):
5012a4161d9SMatthew G Knepley      self.setCompilers.pushLanguage('FC')
502ca77dbeeSGeoffrey Irving      fd.write('\"Using Fortran linker: %s\\n\"\n' % (escape(self.setCompilers.getLinker())))
5032a4161d9SMatthew G Knepley      self.setCompilers.popLanguage()
5045188cb68SSatish Balay    fd.write('\"Using libraries: %s%s -L%s %s %s\\n\"\n' % (escape(self.setCompilers.CSharedLinkerFlag), escape(os.path.join(self.installdir.petscDir, self.installdir.petscArch, 'lib')), escape(os.path.join(self.installdir.petscDir, self.installdir.petscArch, 'lib')), escape(self.petsclib), escape(self.PETSC_EXTERNAL_LIB_BASIC)))
505cdec380aSBarry Smith    fd.write('\"-----------------------------------------\\n\";\n')
5062a4161d9SMatthew G Knepley    fd.close()
5072a4161d9SMatthew G Knepley    return
508b2843cf1SBarry Smith
509511a6afcSJed Brown  def dumpCMakeConfig(self):
510511a6afcSJed Brown    '''
511724dfae7SSatish Balay    Writes configuration-specific values to ${PETSC_ARCH}/lib/petsc/conf/PETScBuildInternal.cmake.
512511a6afcSJed Brown    This file is private to PETSc and should not be included by third parties
513511a6afcSJed Brown    (a suitable file can be produced later by CMake, but this is not it).
514511a6afcSJed Brown    '''
515511a6afcSJed Brown    def cmakeset(fd,key,val=True):
516511a6afcSJed Brown      if val == True: val = 'YES'
517511a6afcSJed Brown      if val == False: val = 'NO'
518511a6afcSJed Brown      fd.write('set (' + key + ' ' + val + ')\n')
519511a6afcSJed Brown    def ensurelist(a):
520826d9344SJed Brown      if isinstance(a,list):
521826d9344SJed Brown        return a
522826d9344SJed Brown      else:
523826d9344SJed Brown        return [a]
524511a6afcSJed Brown    def libpath(lib):
525511a6afcSJed Brown      'Returns a search path if that is what this item provides, else "" which will be cleaned out later'
5261b1c0b30SJed Brown      if not isinstance(lib,str): return ''
527511a6afcSJed Brown      if lib.startswith('-L'): return lib[2:]
528511a6afcSJed Brown      if lib.startswith('-R'): return lib[2:]
529511a6afcSJed Brown      if lib.startswith('-Wl,-rpath,'):
530511a6afcSJed Brown        # This case occurs when an external package needs a specific system library that is normally provided by the compiler.
531511a6afcSJed Brown        # In other words, the -L path is builtin to the wrapper or compiler, here we provide it so that CMake can locate the
532511a6afcSJed Brown        # corresponding library.
533511a6afcSJed Brown        return lib[len('-Wl,-rpath,'):]
534511a6afcSJed Brown      if lib.startswith('-'): return ''
535511a6afcSJed Brown      return os.path.dirname(lib)
536511a6afcSJed Brown    def cleanlib(lib):
537511a6afcSJed Brown      'Returns a library name if that is what this item provides, else "" which will be cleaned out later'
53842e8629dSMatthew G Knepley      if not isinstance(lib,str): return ''
539511a6afcSJed Brown      if lib.startswith('-l'):  return lib[2:]
540511a6afcSJed Brown      if lib.startswith('-Wl') or lib.startswith('-L'): return ''
541511a6afcSJed Brown      lib = os.path.splitext(os.path.basename(lib))[0]
542511a6afcSJed Brown      if lib.startswith('lib'): return lib[3:]
543511a6afcSJed Brown      return lib
544511a6afcSJed Brown    def nub(lst):
54506e8c1ddSJed Brown      'Return a list containing the first occurrence of each unique element'
546511a6afcSJed Brown      unique = []
547511a6afcSJed Brown      for elem in lst:
548511a6afcSJed Brown        if elem not in unique and elem != '':
549511a6afcSJed Brown          unique.append(elem)
550511a6afcSJed Brown      return unique
55106e8c1ddSJed Brown    def nublast(lst):
55206e8c1ddSJed Brown      'Return a list containing the last occurrence of each unique entry in a list'
55350937898SJed Brown      return reversed(nub(reversed(lst)))
554511a6afcSJed Brown    def cmakeexpand(varname):
555511a6afcSJed Brown      return r'"${' + varname + r'}"'
556582751aaSJed Brown    def uniqextend(lst,new):
557511a6afcSJed Brown      for x in ensurelist(new):
558582751aaSJed Brown        if x not in lst:
559582751aaSJed Brown          lst.append(x)
560511a6afcSJed Brown    def notstandardinclude(path):
561040257f2SJed Brown      return path not in '/usr/include'.split() # /usr/local/include is not automatically included on FreeBSD
562511a6afcSJed Brown    def writeMacroDefinitions(fd):
563511a6afcSJed Brown      if self.mpi.usingMPIUni:
564511a6afcSJed Brown        cmakeset(fd,'PETSC_HAVE_MPIUNI')
565511a6afcSJed Brown      for pkg in self.framework.packages:
566511a6afcSJed Brown        if pkg.useddirectly:
567eeb16384SBarry Smith          cmakeset(fd,'PETSC_HAVE_' + pkg.PACKAGE.replace('-','_'))
568a23e9343SMatthew G Knepley        for pair in pkg.defines.items():
569440af75fSJed Brown          if pair[0].startswith('HAVE_') and pair[1]:
570a23e9343SMatthew G Knepley            cmakeset(fd, self.framework.getFullDefineName(pkg, pair[0]), pair[1])
571511a6afcSJed Brown      for name,val in self.functions.defines.items():
572511a6afcSJed Brown        cmakeset(fd,'PETSC_'+name,val)
573511a6afcSJed Brown      for dct in [self.defines, self.libraryoptions.defines]:
574511a6afcSJed Brown        for k,v in dct.items():
575511a6afcSJed Brown          if k.startswith('USE_'):
576511a6afcSJed Brown            cmakeset(fd,'PETSC_' + k, v)
577511a6afcSJed Brown      cmakeset(fd,'PETSC_USE_COMPLEX', self.scalartypes.scalartype == 'complex')
578ce63c4c1SBarry Smith      cmakeset(fd,'PETSC_USE_REAL_' + self.scalartypes.precision.upper())
579511a6afcSJed Brown      cmakeset(fd,'PETSC_CLANGUAGE_'+self.languages.clanguage)
580511a6afcSJed Brown      if hasattr(self.compilers, 'FC'):
581511a6afcSJed Brown        cmakeset(fd,'PETSC_HAVE_FORTRAN')
582511a6afcSJed Brown        if self.compilers.fortranIsF90:
583511a6afcSJed Brown          cmakeset(fd,'PETSC_USING_F90')
584876d5c60SBarry Smith        if self.compilers.fortranIsF2003:
585876d5c60SBarry Smith          cmakeset(fd,'PETSC_USING_F2003')
58613c0a95cSJed Brown      if hasattr(self.compilers, 'CXX'):
58713c0a95cSJed Brown        cmakeset(fd,'PETSC_HAVE_CXX')
588511a6afcSJed Brown      if self.sharedlibraries.useShared:
589511a6afcSJed Brown        cmakeset(fd,'BUILD_SHARED_LIBS')
590511a6afcSJed Brown    def writeBuildFlags(fd):
59106e8c1ddSJed Brown      def extendby(lib):
59206e8c1ddSJed Brown        libs = ensurelist(lib)
59306e8c1ddSJed Brown        lib_paths.extend(map(libpath,libs))
59406e8c1ddSJed Brown        lib_libs.extend(map(cleanlib,libs))
595511a6afcSJed Brown      lib_paths = []
596511a6afcSJed Brown      lib_libs  = []
597511a6afcSJed Brown      includes  = []
598511a6afcSJed Brown      libvars   = []
599511a6afcSJed Brown      for pkg in self.framework.packages:
6004d02c0d4SBarry Smith        if pkg.linkedbypetsc:
60106e8c1ddSJed Brown          extendby(pkg.lib)
602040257f2SJed Brown          uniqextend(includes,pkg.include)
60306e8c1ddSJed Brown      extendby(self.libraries.math)
60406e8c1ddSJed Brown      extendby(self.libraries.rt)
60506e8c1ddSJed Brown      extendby(self.compilers.flibs)
60606e8c1ddSJed Brown      extendby(self.compilers.cxxlibs)
60706e8c1ddSJed Brown      extendby(self.compilers.LIBS.split())
60806e8c1ddSJed Brown      for libname in nublast(lib_libs):
609511a6afcSJed Brown        libvar = 'PETSC_' + libname.upper() + '_LIB'
6104c0032a9SSatish Balay        addpath = ''
61106e8c1ddSJed Brown        for lpath in nublast(lib_paths):
6124c0032a9SSatish Balay          addpath += '"' + str(lpath) + '" '
6134c0032a9SSatish Balay        fd.write('find_library (' + libvar + ' ' + libname + ' HINTS ' + addpath + ')\n')
614511a6afcSJed Brown        libvars.append(libvar)
615511a6afcSJed Brown      fd.write('mark_as_advanced (' + ' '.join(libvars) + ')\n')
616511a6afcSJed Brown      fd.write('set (PETSC_PACKAGE_LIBS ' + ' '.join(map(cmakeexpand,libvars)) + ')\n')
617bb3dd2f6SJed Brown      includes = list(filter(notstandardinclude,includes))
618040257f2SJed Brown      fd.write('set (PETSC_PACKAGE_INCLUDES ' + ' '.join(map(lambda i: '"'+i+'"',includes)) + ')\n')
619724dfae7SSatish Balay    fd = open(os.path.join(self.arch.arch,'lib','petsc','conf','PETScBuildInternal.cmake'), 'w')
620511a6afcSJed Brown    writeMacroDefinitions(fd)
621511a6afcSJed Brown    writeBuildFlags(fd)
622511a6afcSJed Brown    fd.close()
623511a6afcSJed Brown    return
624511a6afcSJed Brown
6258b0282a9SJed Brown  def dumpCMakeLists(self):
6268b0282a9SJed Brown    import sys
627994b4dadSSatish Balay    if sys.version_info >= (2,4):
6288b0282a9SJed Brown      import cmakegen
6298b0282a9SJed Brown      try:
630a98e69d2SJed Brown        cmakegen.main(self.petscdir.dir, log=self.framework.log)
6315b6bfdb9SJed Brown      except (OSError) as e:
6328b0282a9SJed Brown        self.framework.logPrint('Generating CMakeLists.txt failed:\n' + str(e))
633aac20692SSatish Balay    else:
634aac20692SSatish Balay      self.framework.logPrint('Skipping cmakegen due to old python version: ' +str(sys.version_info) )
6358b0282a9SJed Brown
6368b0282a9SJed Brown  def cmakeBoot(self):
6378b0282a9SJed Brown    import sys
638ae937f1dSJed Brown    self.cmakeboot_success = False
639994b4dadSSatish Balay    if sys.version_info >= (2,4) and hasattr(self.cmake,'cmake'):
6405a4feeedSSatish Balay      oldRead = self.argDB.readonly
6415a4feeedSSatish Balay      self.argDB.readonly = True
642356464bcSMatthew G Knepley      try:
6438b0282a9SJed Brown        import cmakeboot
644ae937f1dSJed Brown        self.cmakeboot_success = cmakeboot.main(petscdir=self.petscdir.dir,petscarch=self.arch.arch,argDB=self.argDB,framework=self.framework,log=self.framework.log)
6455b6bfdb9SJed Brown      except (OSError) as e:
6468b0282a9SJed Brown        self.framework.logPrint('Booting CMake in PETSC_ARCH failed:\n' + str(e))
6475b6bfdb9SJed Brown      except (ImportError, KeyError) as e:
648356464bcSMatthew G Knepley        self.framework.logPrint('Importing cmakeboot failed:\n' + str(e))
6495a4feeedSSatish Balay      self.argDB.readonly = oldRead
6509b12c9c7SJed Brown      if self.cmakeboot_success:
6512f730bc2SKarl Rupp        if hasattr(self.compilers, 'FC') and self.compilers.fortranIsF90 and not self.setCompilers.fortranModuleOutputFlag:
65291f9b906SSatish Balay          self.framework.logPrint('CMake configured successfully, but could not be used by default because of missing fortranModuleOutputFlag\n')
6539b12c9c7SJed Brown        else:
6549b12c9c7SJed Brown          self.framework.logPrint('CMake configured successfully, using as default build\n')
655f7b66a64SJed Brown          self.addMakeMacro('PETSC_BUILD_USING_CMAKE',1)
656aac20692SSatish Balay      else:
6579b12c9c7SJed Brown        self.framework.logPrint('CMake configuration was unsuccessful\n')
6589b12c9c7SJed Brown    else:
659aac20692SSatish Balay      self.framework.logPrint('Skipping cmakeboot due to old python version: ' +str(sys.version_info) )
660356464bcSMatthew G Knepley    return
6618b0282a9SJed Brown
662b2843cf1SBarry Smith  def configurePrefetch(self):
663b2843cf1SBarry Smith    '''Sees if there are any prefetch functions supported'''
6647fca349cSMatthew G. Knepley    if config.setCompilers.Configure.isSolaris(self.log) or self.framework.argDB['with-ios'] or not self.framework.argDB['with-prefetch']:
66593f78423SSatish Balay      self.addDefine('Prefetch(a,b,c)', ' ')
66693f78423SSatish Balay      return
667ec284106SBarry Smith    self.pushLanguage(self.languages.clanguage)
66810699583SJed Brown    if self.checkLink('#include <xmmintrin.h>', 'void *v = 0;_mm_prefetch((const char*)v,_MM_HINT_NTA);\n'):
66950d8bf02SJed Brown      # The Intel Intrinsics manual [1] specifies the prototype
67050d8bf02SJed Brown      #
67150d8bf02SJed Brown      #   void _mm_prefetch(char const *a, int sel);
67250d8bf02SJed Brown      #
67350d8bf02SJed Brown      # but other vendors seem to insist on using subtly different
67450d8bf02SJed Brown      # prototypes, including void* for the pointer, and an enum for
67550d8bf02SJed Brown      # sel.  These are both reasonable changes, but negatively impact
67650d8bf02SJed Brown      # portability.
67750d8bf02SJed Brown      #
678a8d69d7bSBarry Smith      # [1] https://software.intel.com/file/6373
67950d8bf02SJed Brown      self.addDefine('HAVE_XMMINTRIN_H', 1)
68050d8bf02SJed Brown      self.addDefine('Prefetch(a,b,c)', '_mm_prefetch((const char*)(a),(c))')
68150d8bf02SJed Brown      self.addDefine('PREFETCH_HINT_NTA', '_MM_HINT_NTA')
68250d8bf02SJed Brown      self.addDefine('PREFETCH_HINT_T0',  '_MM_HINT_T0')
68350d8bf02SJed Brown      self.addDefine('PREFETCH_HINT_T1',  '_MM_HINT_T1')
68450d8bf02SJed Brown      self.addDefine('PREFETCH_HINT_T2',  '_MM_HINT_T2')
68550d8bf02SJed Brown    elif self.checkLink('#include <xmmintrin.h>', 'void *v = 0;_mm_prefetch(v,_MM_HINT_NTA);\n'):
68650d8bf02SJed Brown      self.addDefine('HAVE_XMMINTRIN_H', 1)
68750d8bf02SJed Brown      self.addDefine('Prefetch(a,b,c)', '_mm_prefetch((const void*)(a),(c))')
68850d8bf02SJed Brown      self.addDefine('PREFETCH_HINT_NTA', '_MM_HINT_NTA')
68950d8bf02SJed Brown      self.addDefine('PREFETCH_HINT_T0',  '_MM_HINT_T0')
69050d8bf02SJed Brown      self.addDefine('PREFETCH_HINT_T1',  '_MM_HINT_T1')
69150d8bf02SJed Brown      self.addDefine('PREFETCH_HINT_T2',  '_MM_HINT_T2')
69210699583SJed Brown    elif self.checkLink('', 'void *v = 0;__builtin_prefetch(v,0,0);\n'):
69310699583SJed Brown      # From GCC docs: void __builtin_prefetch(const void *addr,int rw,int locality)
69410699583SJed Brown      #
69510699583SJed Brown      #   The value of rw is a compile-time constant one or zero; one
69610699583SJed Brown      #   means that the prefetch is preparing for a write to the memory
69710699583SJed Brown      #   address and zero, the default, means that the prefetch is
69810699583SJed Brown      #   preparing for a read. The value locality must be a compile-time
69910699583SJed Brown      #   constant integer between zero and three. A value of zero means
70010699583SJed Brown      #   that the data has no temporal locality, so it need not be left
70110699583SJed Brown      #   in the cache after the access. A value of three means that the
70210699583SJed Brown      #   data has a high degree of temporal locality and should be left
70310699583SJed Brown      #   in all levels of cache possible. Values of one and two mean,
70410699583SJed Brown      #   respectively, a low or moderate degree of temporal locality.
70510699583SJed Brown      #
70610699583SJed Brown      # Here we adopt Intel's x86/x86-64 naming scheme for the locality
70710699583SJed Brown      # hints.  Using macros for these values in necessary since some
70810699583SJed Brown      # compilers require an enum.
70910699583SJed Brown      self.addDefine('Prefetch(a,b,c)', '__builtin_prefetch((a),(b),(c))')
71010699583SJed Brown      self.addDefine('PREFETCH_HINT_NTA', '0')
71110699583SJed Brown      self.addDefine('PREFETCH_HINT_T0',  '3')
71210699583SJed Brown      self.addDefine('PREFETCH_HINT_T1',  '2')
71310699583SJed Brown      self.addDefine('PREFETCH_HINT_T2',  '1')
714b2843cf1SBarry Smith    else:
715b2843cf1SBarry Smith      self.addDefine('Prefetch(a,b,c)', ' ')
7167d490b44SBarry Smith    self.popLanguage()
717b2843cf1SBarry Smith
71809bc878fSSatish Balay  def configureAtoll(self):
71909bc878fSSatish Balay    '''Checks if atoll exists'''
720436b02dcSSatish 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")'):
72109bc878fSSatish Balay       self.addDefine('HAVE_ATOLL', '1')
72209bc878fSSatish Balay
7232400fdedSBarry Smith  def configureUnused(self):
7242400fdedSBarry Smith    '''Sees if __attribute((unused)) is supported'''
7251adaff47SSean Farley    if self.framework.argDB['with-ios']:
7262400fdedSBarry Smith      self.addDefine('UNUSED', ' ')
7272400fdedSBarry Smith      return
7282400fdedSBarry Smith    self.pushLanguage(self.languages.clanguage)
729edf21b64SSatish 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'):
7302400fdedSBarry Smith      self.addDefine('UNUSED', '__attribute((unused))')
7312400fdedSBarry Smith    else:
7322400fdedSBarry Smith      self.addDefine('UNUSED', ' ')
7332400fdedSBarry Smith    self.popLanguage()
7342400fdedSBarry Smith
73598ed35c3SBarry Smith  def configureIsatty(self):
73698ed35c3SBarry Smith    '''Check if the Unix C function isatty() works correctly
73798ed35c3SBarry Smith       Actually just assumes it does not work correctly on batch systems'''
73898ed35c3SBarry Smith    if not self.framework.argDB['with-batch']:
73998ed35c3SBarry Smith      self.addDefine('USE_ISATTY',1)
74098ed35c3SBarry Smith
7411ef8df7fSJed Brown  def configureDeprecated(self):
7421ef8df7fSJed Brown    '''Check if __attribute((deprecated)) is supported'''
7431ef8df7fSJed Brown    self.pushLanguage(self.languages.clanguage)
74459a26b54SJed Brown    ## Recent versions of gcc and clang support __attribute((deprecated("string argument"))), which is very useful, but
74559a26b54SJed Brown    ## Intel has conspired to make a supremely environment-sensitive compiler.  The Intel compiler looks at the gcc
74659a26b54SJed Brown    ## executable in the environment to determine the language compatibility that it should attempt to emulate.  Some
74759a26b54SJed Brown    ## important Cray installations have built PETSc using the Intel compiler, but with a newer gcc module loaded (e.g.,
748df3898eeSBarry Smith    ## 4.7).  Thus at PETSc configure time, the Intel compiler decides to support the string argument, but the gcc
74959a26b54SJed Brown    ## found in the default user environment is older and does not support the argument.  If GCC and Intel were cool
75059a26b54SJed Brown    ## like Clang and supported __has_attribute, we could avoid configure tests entirely, but they don't.  And that is
75159a26b54SJed Brown    ## why we can't have nice things.
75259a26b54SJed Brown    #
75359a26b54SJed Brown    # if self.checkCompile("""__attribute((deprecated("Why you shouldn't use myfunc"))) static int myfunc(void) { return 1;}""", ''):
75425ef9dfeSBarry Smith    #   self.addDefine('DEPRECATED_FUNCTION(why)', '__attribute((deprecated(why)))')
75525ef9dfeSBarry Smith    #   self.addDefine('DEPRECATED_TYPEDEF(why)', '__attribute((deprecated(why)))')
75659a26b54SJed Brown    if self.checkCompile("""__attribute((deprecated)) static int myfunc(void) { return 1;}""", ''):
75725ef9dfeSBarry Smith      self.addDefine('DEPRECATED_FUNCTION(why)', '__attribute((deprecated))')
75825ef9dfeSBarry Smith      self.addDefine('DEPRECATED_TYPEDEF(why)', '__attribute((deprecated))')
7591ef8df7fSJed Brown    else:
76025ef9dfeSBarry Smith      self.addDefine('DEPRECATED_FUNCTION(why)', ' ')
76125ef9dfeSBarry Smith      self.addDefine('DEPRECATED_TYPEDEF(why)', ' ')
762a8678870SSatish Balay    if self.checkCompile("""enum E {oldval __attribute((deprecated)), newval };""", ''):
763933a48e8SSatish Balay      self.addDefine('DEPRECATED_ENUM(why)', '__attribute((deprecated))')
764933a48e8SSatish Balay    else:
765933a48e8SSatish Balay      self.addDefine('DEPRECATED_ENUM(why)', ' ')
76605de396fSBarry Smith    # I was unable to make a CPP macro that takes the old and new values as seperate arguments and builds the message needed by _Pragma
76705de396fSBarry Smith    # hence the deprecation message is handled as it is
76805de396fSBarry Smith    if self.checkCompile('#define TEST _Pragma("GCC warning \"Testing _Pragma\"") value'):
76905de396fSBarry Smith      self.addDefine('DEPRECATED_MACRO(why)', '_Pragma(why)')
77005de396fSBarry Smith    else:
77105de396fSBarry Smith      self.addDefine('DEPRECATED_MACRO(why)', ' ')
7721ef8df7fSJed Brown    self.popLanguage()
7731ef8df7fSJed Brown
77418f41590SBarry Smith  def configureAlign(self):
775*7b7fc14bSLisandro Dalcin    '''Check if __attribute(aligned) is supported'''
776*7b7fc14bSLisandro Dalcin    code = '''\
777752d89a4SSatish Balaystruct mystruct {int myint;} __attribute((aligned(16)));
778*7b7fc14bSLisandro Dalcinchar assert_aligned[(sizeof(struct mystruct)==16)*2-1];
779752d89a4SSatish Balay'''
780752d89a4SSatish Balay    self.pushLanguage(self.languages.clanguage)
781*7b7fc14bSLisandro Dalcin    if self.checkCompile(code):
782752d89a4SSatish Balay      self.addDefine('ATTRIBUTEALIGNED(size)', '__attribute((aligned(size)))')
783752d89a4SSatish Balay      self.addDefine('HAVE_ATTRIBUTEALIGNED', 1)
784752d89a4SSatish Balay    else:
785*7b7fc14bSLisandro Dalcin      self.framework.logPrint('Incorrect attribute(aligned)')
786752d89a4SSatish Balay      self.addDefine('ATTRIBUTEALIGNED(size)', ' ')
787*7b7fc14bSLisandro Dalcin    self.popLanguage()
788752d89a4SSatish Balay    return
78918f41590SBarry Smith
7909800092aSJed Brown  def configureExpect(self):
7919800092aSJed Brown    '''Sees if the __builtin_expect directive is supported'''
7929800092aSJed Brown    self.pushLanguage(self.languages.clanguage)
7939800092aSJed Brown    if self.checkLink('', 'if (__builtin_expect(0,1)) return 1;'):
7949800092aSJed Brown      self.addDefine('HAVE_BUILTIN_EXPECT', 1)
7959800092aSJed Brown    self.popLanguage()
7969800092aSJed Brown
79753c77d0aSJed Brown  def configureFunctionName(self):
798fbfcfee5SBarry Smith    '''Sees if the compiler supports __func__ or a variant.'''
7991ec50b02SJed Brown    def getFunctionName(lang):
800fbfcfee5SBarry Smith      name = '"unknown"'
8011ec50b02SJed Brown      self.pushLanguage(lang)
802b6ff4c76SKarl Rupp      for fname in ['__func__','__FUNCTION__','__extension__ __func__']:
8030117e5a1SSatish Balay        code = "if ("+fname+"[0] != 'm') return 1;"
8040117e5a1SSatish Balay        if self.checkCompile('',code) and self.checkLink('',code):
8050117e5a1SSatish Balay          name = fname
8060117e5a1SSatish Balay          break
8071ec50b02SJed Brown      self.popLanguage()
8081ec50b02SJed Brown      return name
8091ec50b02SJed Brown    langs = []
810628773c9SSatish Balay
811628773c9SSatish Balay    self.addDefine('FUNCTION_NAME_C', getFunctionName('C'))
8125f6e5f85SSatish Balay    if hasattr(self.compilers, 'CXX'):
813628773c9SSatish Balay      self.addDefine('FUNCTION_NAME_CXX', getFunctionName('Cxx'))
81453c77d0aSJed Brown
815753ebd1dSJed Brown  def configureIntptrt(self):
816753ebd1dSJed Brown    '''Determine what to use for uintptr_t'''
817753ebd1dSJed Brown    def staticAssertSizeMatchesVoidStar(inc,typename):
818753ebd1dSJed Brown      # The declaration is an error if either array size is negative.
819753ebd1dSJed 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
820d26187a0SJed Brown      return self.checkCompile(inc, ('#define STATIC_ASSERT(cond) char negative_length_if_false[2*(!!(cond))-1]\n'
821979939cdSSatish Balay                                     + 'STATIC_ASSERT(sizeof(void*) == sizeof(%s));'%typename))
822753ebd1dSJed Brown    self.pushLanguage(self.languages.clanguage)
823753ebd1dSJed Brown    if self.checkCompile('#include <stdint.h>', 'int x; uintptr_t i = (uintptr_t)&x;'):
824753ebd1dSJed Brown      self.addDefine('UINTPTR_T', 'uintptr_t')
825753ebd1dSJed Brown    elif staticAssertSizeMatchesVoidStar('','unsigned long long'):
826753ebd1dSJed Brown      self.addDefine('UINTPTR_T', 'unsigned long long')
827753ebd1dSJed Brown    elif staticAssertSizeMatchesVoidStar('#include <stdlib.h>','size_t') or staticAssertSizeMatchesVoidStar('#include <string.h>', 'size_t'):
828753ebd1dSJed Brown      self.addDefine('UINTPTR_T', 'size_t')
829c82284b1SJed Brown    elif staticAssertSizeMatchesVoidStar('','unsigned long'):
830c82284b1SJed Brown      self.addDefine('UINTPTR_T', 'unsigned long')
8312d1b7972SSatish Balay    elif staticAssertSizeMatchesVoidStar('','unsigned'):
832753ebd1dSJed Brown      self.addDefine('UINTPTR_T', 'unsigned')
833d26187a0SJed Brown    else:
834d26187a0SJed Brown      raise RuntimeError('Could not find any unsigned integer type matching void*')
835753ebd1dSJed Brown    self.popLanguage()
836753ebd1dSJed Brown
837ed938b00SJed Brown  def configureRTLDDefault(self):
838bfef2c86SBarry Smith    if self.checkCompile('#include <dlfcn.h>\n void *ptr =  RTLD_DEFAULT;'):
839bfef2c86SBarry Smith      self.addDefine('RTLD_DEFAULT','1')
840f8833479SBarry Smith    return
841f8833479SBarry Smith
842f8833479SBarry Smith  def configureSolaris(self):
843f8833479SBarry Smith    '''Solaris specific stuff'''
844f8833479SBarry Smith    if os.path.isdir(os.path.join('/usr','ucblib')):
845f8833479SBarry Smith      try:
846f8833479SBarry Smith        flag = getattr(self.setCompilers, self.language[-1]+'SharedLinkerFlag')
847f8833479SBarry Smith      except AttributeError:
848f8833479SBarry Smith        flag = None
849f8833479SBarry Smith      if flag is None:
850f8833479SBarry Smith        self.compilers.LIBS += ' -L/usr/ucblib'
851f8833479SBarry Smith      else:
852f8833479SBarry Smith        self.compilers.LIBS += ' '+flag+'/usr/ucblib'
853f8833479SBarry Smith    return
854f8833479SBarry Smith
855f8833479SBarry Smith  def configureLinux(self):
856f8833479SBarry Smith    '''Linux specific stuff'''
8579f15855cSMatthew G Knepley    # TODO: Test for this by mallocing an odd number of floats and checking the address
858f8833479SBarry Smith    self.addDefine('HAVE_DOUBLE_ALIGN_MALLOC', 1)
859f8833479SBarry Smith    return
860f8833479SBarry Smith
861f8833479SBarry Smith  def configureWin32(self):
862f8833479SBarry Smith    '''Win32 non-cygwin specific stuff'''
863f8833479SBarry Smith    kernel32=0
864f8833479SBarry Smith    if self.libraries.add('Kernel32.lib','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    elif self.libraries.add('kernel32','GetComputerName',prototype='#include <Windows.h>', call='GetComputerName(NULL,NULL);'):
869f8833479SBarry Smith      self.addDefine('HAVE_WINDOWS_H',1)
870f8833479SBarry Smith      self.addDefine('HAVE_GETCOMPUTERNAME',1)
871f8833479SBarry Smith      kernel32=1
872f8833479SBarry Smith    if kernel32:
873eed94e11SSatish Balay      if self.framework.argDB['with-windows-graphics']:
874eed94e11SSatish Balay        self.addDefine('USE_WINDOWS_GRAPHICS',1)
875f8833479SBarry Smith      if self.checkLink('#include <Windows.h>','LoadLibrary(0)'):
876f8833479SBarry Smith        self.addDefine('HAVE_LOADLIBRARY',1)
877b50f6d9eSLisandro Dalcin      if self.checkLink('#include <Windows.h>','GetProcAddress(0,0)'):
878b50f6d9eSLisandro Dalcin        self.addDefine('HAVE_GETPROCADDRESS',1)
879b50f6d9eSLisandro Dalcin      if self.checkLink('#include <Windows.h>','FreeLibrary(0)'):
880b50f6d9eSLisandro Dalcin        self.addDefine('HAVE_FREELIBRARY',1)
881a21658a3SLisandro Dalcin      if self.checkLink('#include <Windows.h>','GetLastError()'):
882a21658a3SLisandro Dalcin        self.addDefine('HAVE_GETLASTERROR',1)
883a21658a3SLisandro Dalcin      if self.checkLink('#include <Windows.h>','SetLastError(0)'):
884a21658a3SLisandro Dalcin        self.addDefine('HAVE_SETLASTERROR',1)
885f8833479SBarry Smith      if self.checkLink('#include <Windows.h>\n','QueryPerformanceCounter(0);\n'):
886bea725cfSBarry Smith        self.addDefine('USE_MICROSOFT_TIME',1)
887f8833479SBarry Smith    if self.libraries.add('Advapi32.lib','GetUserName',prototype='#include <Windows.h>', call='GetUserName(NULL,NULL);'):
888f8833479SBarry Smith      self.addDefine('HAVE_GET_USER_NAME',1)
889f8833479SBarry Smith    elif self.libraries.add('advapi32','GetUserName',prototype='#include <Windows.h>', call='GetUserName(NULL,NULL);'):
890f8833479SBarry Smith      self.addDefine('HAVE_GET_USER_NAME',1)
891f8833479SBarry Smith
892f8833479SBarry Smith    if not self.libraries.add('User32.lib','GetDC',prototype='#include <Windows.h>',call='GetDC(0);'):
893f8833479SBarry Smith      self.libraries.add('user32','GetDC',prototype='#include <Windows.h>',call='GetDC(0);')
894f8833479SBarry Smith    if not self.libraries.add('Gdi32.lib','CreateCompatibleDC',prototype='#include <Windows.h>',call='CreateCompatibleDC(0);'):
895f8833479SBarry Smith      self.libraries.add('gdi32','CreateCompatibleDC',prototype='#include <Windows.h>',call='CreateCompatibleDC(0);')
896f8833479SBarry Smith
897f8833479SBarry Smith    self.types.check('int32_t', 'int')
898f8833479SBarry Smith    if not self.checkCompile('#include <sys/types.h>\n','uid_t u;\n'):
899f8833479SBarry Smith      self.addTypedef('int', 'uid_t')
900f8833479SBarry Smith      self.addTypedef('int', 'gid_t')
901f8833479SBarry Smith    if not self.checkLink('#if defined(PETSC_HAVE_UNISTD_H)\n#include <unistd.h>\n#endif\n','int a=R_OK;\n'):
902f8833479SBarry Smith      self.framework.addDefine('R_OK', '04')
903f8833479SBarry Smith      self.framework.addDefine('W_OK', '02')
904f8833479SBarry Smith      self.framework.addDefine('X_OK', '01')
905f8833479SBarry Smith    if not self.checkLink('#include <sys/stat.h>\n','int a=0;\nif (S_ISDIR(a)){}\n'):
906f8833479SBarry Smith      self.framework.addDefine('S_ISREG(a)', '(((a)&_S_IFMT) == _S_IFREG)')
907f8833479SBarry Smith      self.framework.addDefine('S_ISDIR(a)', '(((a)&_S_IFMT) == _S_IFDIR)')
908f8833479SBarry Smith    if self.checkCompile('#include <Windows.h>\n','LARGE_INTEGER a;\nDWORD b=a.u.HighPart;\n'):
909f8833479SBarry Smith      self.addDefine('HAVE_LARGE_INTEGER_U',1)
910f8833479SBarry Smith
911f8833479SBarry Smith    # Windows requires a Binary file creation flag when creating/opening binary files.  Is a better test in order?
912ef2cfba3SSatish Balay    if self.checkCompile('#include <Windows.h>\n#include <fcntl.h>\n', 'int flags = O_BINARY;'):
913f8833479SBarry Smith      self.addDefine('HAVE_O_BINARY',1)
914f8833479SBarry Smith
915f8833479SBarry Smith    if self.compilers.CC.find('win32fe') >= 0:
916ad4212abSSatish Balay      self.addDefine('HAVE_WINDOWS_COMPILERS',1)
917f8833479SBarry Smith      self.addDefine('PATH_SEPARATOR','\';\'')
918f8833479SBarry Smith      self.addDefine('DIR_SEPARATOR','\'\\\\\'')
919f8833479SBarry Smith      self.addDefine('REPLACE_DIR_SEPARATOR','\'/\'')
920f8833479SBarry Smith      self.addDefine('CANNOT_START_DEBUGGER',1)
9215188cb68SSatish Balay      (petscdir,error,status) = self.executeShellCommand('cygpath -w '+self.installdir.petscDir, log = self.log)
92234531a4dSSatish Balay      self.addDefine('DIR','"'+petscdir.replace('\\','\\\\')+'"')
9235188cb68SSatish Balay      (petscdir,error,status) = self.executeShellCommand('cygpath -m '+self.installdir.petscDir, log = self.log)
924e433681fSSatish Balay      self.addMakeMacro('wPETSC_DIR',petscdir)
925f8833479SBarry Smith    else:
926f8833479SBarry Smith      self.addDefine('PATH_SEPARATOR','\':\'')
927f8833479SBarry Smith      self.addDefine('REPLACE_DIR_SEPARATOR','\'\\\\\'')
928f8833479SBarry Smith      self.addDefine('DIR_SEPARATOR','\'/\'')
9295188cb68SSatish Balay      self.addDefine('DIR','"'+self.installdir.petscDir+'"')
9305188cb68SSatish Balay      self.addMakeMacro('wPETSC_DIR',self.installdir.petscDir)
9315188cb68SSatish Balay    self.addDefine('ARCH','"'+self.installdir.petscArch+'"')
932f8833479SBarry Smith    return
933f8833479SBarry Smith
934f8833479SBarry Smith#-----------------------------------------------------------------------------------------------------
935b10d012aSSatish Balay  def configureCygwinBrokenPipe(self):
936b10d012aSSatish Balay    '''Cygwin version <= 1.7.18 had issues with pipes and long commands invoked from gnu-make
937b10d012aSSatish Balay    http://cygwin.com/ml/cygwin/2013-05/msg00340.html '''
9387fca349cSMatthew G. Knepley    if config.setCompilers.Configure.isCygwin(self.log):
939b10d012aSSatish Balay      import platform
940b10d012aSSatish Balay      import re
941b10d012aSSatish Balay      r=re.compile("([0-9]+).([0-9]+).([0-9]+)")
942b10d012aSSatish Balay      m=r.match(platform.release())
943b10d012aSSatish Balay      major=int(m.group(1))
944b10d012aSSatish Balay      minor=int(m.group(2))
945b10d012aSSatish Balay      subminor=int(m.group(3))
946b10d012aSSatish Balay      if ((major < 1) or (major == 1 and minor < 7) or (major == 1 and minor == 7 and subminor <= 18)):
947b10d012aSSatish Balay        self.addMakeMacro('PETSC_CYGWIN_BROKEN_PIPE','1')
948b10d012aSSatish Balay    return
949b10d012aSSatish Balay
950b10d012aSSatish Balay#-----------------------------------------------------------------------------------------------------
951569865ddSSatish Balay  def configureDefaultArch(self):
952af0996ceSBarry Smith    conffile = os.path.join('lib','petsc','conf', 'petscvariables')
953569865ddSSatish Balay    if self.framework.argDB['with-default-arch']:
954c6ef1b5bSJed Brown      fd = open(conffile, 'w')
955569865ddSSatish Balay      fd.write('PETSC_ARCH='+self.arch.arch+'\n')
956da93591fSBarry Smith      fd.write('PETSC_DIR='+self.petscdir.dir+'\n')
957b9b902edSJed Brown      fd.write('include '+os.path.join('$(PETSC_DIR)','$(PETSC_ARCH)','lib','petsc','conf','petscvariables')+'\n')
958569865ddSSatish Balay      fd.close()
959569865ddSSatish Balay      self.framework.actions.addArgument('PETSc', 'Build', 'Set default architecture to '+self.arch.arch+' in '+conffile)
960569865ddSSatish Balay    elif os.path.isfile(conffile):
961569865ddSSatish Balay      try:
962569865ddSSatish Balay        os.unlink(conffile)
963569865ddSSatish Balay      except:
964569865ddSSatish Balay        raise RuntimeError('Unable to remove file '+conffile+'. Did a different user create it?')
965569865ddSSatish Balay    return
966569865ddSSatish Balay
967569865ddSSatish Balay#-----------------------------------------------------------------------------------------------------
968f8833479SBarry Smith  def configureScript(self):
969f8833479SBarry Smith    '''Output a script in the conf directory which will reproduce the configuration'''
970f8833479SBarry Smith    import nargs
971495bf1a9SSatish Balay    import sys
972af0996ceSBarry Smith    scriptName = os.path.join(self.arch.arch,'lib','petsc','conf', 'reconfigure-'+self.arch.arch+'.py')
973f8833479SBarry Smith    args = dict([(nargs.Arg.parseArgument(arg)[0], arg) for arg in self.framework.clArgs])
974e97fc2efSSatish Balay    if 'with-clean' in args:
975e97fc2efSSatish Balay      del args['with-clean']
976f8833479SBarry Smith    if 'configModules' in args:
9771063a081SSatish Balay      if nargs.Arg.parseArgument(args['configModules'])[1] == 'PETSc.Configure':
978f8833479SBarry Smith        del args['configModules']
979f8833479SBarry Smith    if 'optionsModule' in args:
98023a19ef1SSatish Balay      if nargs.Arg.parseArgument(args['optionsModule'])[1] == 'config.compilerOptions':
981f8833479SBarry Smith        del args['optionsModule']
982f8833479SBarry Smith    if not 'PETSC_ARCH' in args:
9831063a081SSatish Balay      args['PETSC_ARCH'] = 'PETSC_ARCH='+str(self.arch.arch)
984c6ef1b5bSJed Brown    f = open(scriptName, 'w')
985495bf1a9SSatish Balay    f.write('#!'+sys.executable+'\n')
986f8833479SBarry Smith    f.write('if __name__ == \'__main__\':\n')
987f8833479SBarry Smith    f.write('  import sys\n')
9887561c02cSSatish Balay    f.write('  import os\n')
9897561c02cSSatish Balay    f.write('  sys.path.insert(0, os.path.abspath(\'config\'))\n')
990f8833479SBarry Smith    f.write('  import configure\n')
9911063a081SSatish Balay    # pretty print repr(args.values())
9921063a081SSatish Balay    f.write('  configure_options = [\n')
9938bec23c5SJed Brown    for itm in sorted(args.values()):
9941063a081SSatish Balay      f.write('    \''+str(itm)+'\',\n')
9951063a081SSatish Balay    f.write('  ]\n')
996f8833479SBarry Smith    f.write('  configure.petsc_configure(configure_options)\n')
997f8833479SBarry Smith    f.close()
998f8833479SBarry Smith    try:
9995b6bfdb9SJed Brown      os.chmod(scriptName, 0o775)
10005b6bfdb9SJed Brown    except OSError as e:
1001f8833479SBarry Smith      self.framework.logPrint('Unable to make reconfigure script executable:\n'+str(e))
1002f8833479SBarry Smith    self.framework.actions.addArgument('PETSc', 'File creation', 'Created '+scriptName+' for automatic reconfiguration')
1003f8833479SBarry Smith    return
1004f8833479SBarry Smith
1005f8833479SBarry Smith  def configureInstall(self):
1006f8833479SBarry Smith    '''Setup the directories for installation'''
1007f8833479SBarry Smith    if self.framework.argDB['prefix']:
1008824e893fSSatish Balay      self.addMakeRule('shared_install','',['-@echo "Now to install the libraries do:"',\
1009d093bd8dSBarry Smith                                              '-@echo "'+self.installdir.installSudo+'make PETSC_DIR=${PETSC_DIR} PETSC_ARCH=${PETSC_ARCH} install"',\
1010315b77e6SSatish Balay                                              '-@echo "========================================="'])
1011f8833479SBarry Smith    else:
1012824e893fSSatish Balay      self.addMakeRule('shared_install','',['-@echo "Now to check if the libraries are working do:"',\
10130cd68fe6SJed Brown                                              '-@echo "make PETSC_DIR=${PETSC_DIR} PETSC_ARCH=${PETSC_ARCH} check"',\
1014315b77e6SSatish Balay                                              '-@echo "========================================="'])
1015f8833479SBarry Smith      return
1016f8833479SBarry Smith
1017f8833479SBarry Smith  def configureGCOV(self):
1018f8833479SBarry Smith    if self.framework.argDB['with-gcov']:
1019f8833479SBarry Smith      self.addDefine('USE_GCOV','1')
1020f8833479SBarry Smith    return
1021f8833479SBarry Smith
102228bb2e72SSatish Balay  def postProcessPackages(self):
102328bb2e72SSatish Balay    postPackages=[]
102428bb2e72SSatish Balay    for i in self.framework.packages:
102528bb2e72SSatish Balay      if hasattr(i,'postProcess'): postPackages.append(i)
102628bb2e72SSatish Balay    if postPackages:
1027e64d19dfSSatish Balay      # ctetgen needs petsc conf files. so attempt to create them early
1028a77eb93bSSatish Balay      self.framework.dumpConfFiles()
1029d9293e7bSBarry Smith      # tacky fix for dependency of Aluimia on Pflotran; requested via petsc-dev Matt provide a correct fix
1030d9293e7bSBarry Smith      for i in postPackages:
1031d9293e7bSBarry Smith        if i.name.upper() in ['PFLOTRAN']:
1032d9293e7bSBarry Smith          i.postProcess()
1033d9293e7bSBarry Smith          postPackages.remove(i)
103428bb2e72SSatish Balay      for i in postPackages: i.postProcess()
1035aa5c8b8eSBarry Smith      for i in postPackages:
1036aa5c8b8eSBarry Smith        if i.installedpetsc:
1037aa5c8b8eSBarry Smith          self.installed = 1
1038aa5c8b8eSBarry Smith          break
103928bb2e72SSatish Balay    return
1040f8833479SBarry Smith
1041f8833479SBarry Smith  def configure(self):
1042f8833479SBarry Smith    if not os.path.samefile(self.petscdir.dir, os.getcwd()):
1043f8833479SBarry Smith      raise RuntimeError('Wrong PETSC_DIR option specified: '+str(self.petscdir.dir) + '\n  Configure invoked in: '+os.path.realpath(os.getcwd()))
1044550489e3SMatthew 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):
10453552d8fbSSatish Balay      raise RuntimeError('Incorrect option --prefix='+self.framework.argDB['prefix']+' specified. It cannot be same as PETSC_DIR!')
10468fd0dbdbSBarry Smith    if self.framework.argDB['prefix'] and self.framework.argDB['prefix'].find(' ') > -1:
10478fd0dbdbSBarry Smith      raise RuntimeError('Your --prefix '+self.framework.argDB['prefix']+' has spaces in it; this is not allowed.\n Use a --prefix that does not have spaces in it')
1048c16c35a9SSatish 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)):
1049c16c35a9SSatish Balay      raise RuntimeError('Incorrect option --prefix='+self.framework.argDB['prefix']+' specified. It cannot be same as PETSC_DIR/PETSC_ARCH!')
1050f16c1317SJed Brown    self.framework.header          = os.path.join(self.arch.arch,'include','petscconf.h')
1051f16c1317SJed Brown    self.framework.cHeader         = os.path.join(self.arch.arch,'include','petscfix.h')
1052af0996ceSBarry Smith    self.framework.makeMacroHeader = os.path.join(self.arch.arch,'lib','petsc','conf','petscvariables')
1053af0996ceSBarry Smith    self.framework.makeRuleHeader  = os.path.join(self.arch.arch,'lib','petsc','conf','petscrules')
1054f8833479SBarry Smith    if self.libraries.math is None:
1055f8833479SBarry Smith      raise RuntimeError('PETSc requires a functional math library. Please send configure.log to petsc-maint@mcs.anl.gov.')
1056f8833479SBarry Smith    if self.languages.clanguage == 'Cxx' and not hasattr(self.compilers, 'CXX'):
1057f8833479SBarry Smith      raise RuntimeError('Cannot set C language to C++ without a functional C++ compiler.')
1058ed938b00SJed Brown    self.executeTest(self.configureRTLDDefault)
1059b2843cf1SBarry Smith    self.executeTest(self.configurePrefetch)
10602400fdedSBarry Smith    self.executeTest(self.configureUnused)
10611ef8df7fSJed Brown    self.executeTest(self.configureDeprecated)
106298ed35c3SBarry Smith    self.executeTest(self.configureIsatty)
10639800092aSJed Brown    self.executeTest(self.configureExpect);
106418f41590SBarry Smith    self.executeTest(self.configureAlign);
106553c77d0aSJed Brown    self.executeTest(self.configureFunctionName);
1066753ebd1dSJed Brown    self.executeTest(self.configureIntptrt);
1067f8833479SBarry Smith    self.executeTest(self.configureSolaris)
1068f8833479SBarry Smith    self.executeTest(self.configureLinux)
1069f8833479SBarry Smith    self.executeTest(self.configureWin32)
1070b10d012aSSatish Balay    self.executeTest(self.configureCygwinBrokenPipe)
1071569865ddSSatish Balay    self.executeTest(self.configureDefaultArch)
1072f8833479SBarry Smith    self.executeTest(self.configureScript)
1073f8833479SBarry Smith    self.executeTest(self.configureInstall)
1074f8833479SBarry Smith    self.executeTest(self.configureGCOV)
107509bc878fSSatish Balay    self.executeTest(self.configureAtoll)
1076f8833479SBarry Smith
1077f8833479SBarry Smith    self.Dump()
1078f8833479SBarry Smith    self.dumpConfigInfo()
10792a4161d9SMatthew G Knepley    self.dumpMachineInfo()
1080511a6afcSJed Brown    self.dumpCMakeConfig()
10818b0282a9SJed Brown    self.dumpCMakeLists()
108240277576SBarry Smith    # need to save the current state of BuildSystem so that postProcess() packages can read it in and perhaps run make install
108340277576SBarry Smith    self.framework.storeSubstitutions(self.framework.argDB)
1084492432c8SJed Brown    self.framework.argDB['configureCache'] = pickle.dumps(self.framework)
108540277576SBarry Smith    self.framework.argDB.save(force = True)
10868b0282a9SJed Brown    self.cmakeBoot()
1087262119f8SBarry Smith    self.DumpPkgconfig()
1088351d3a41SMatthew G Knepley    self.DumpModule()
1089f7ad81e1SBarry Smith    self.postProcessPackages()
1090f8833479SBarry Smith    self.framework.log.write('================================================================================\n')
1091f8833479SBarry Smith    self.logClear()
1092f8833479SBarry Smith    return
1093