xref: /petsc/config/PETSc/Configure.py (revision 7ce81a4ba60cb505db2538380c321649eb432b1c)
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
144161761dSBarry Smith    self.found        = 1
15f8833479SBarry Smith    return
16f8833479SBarry Smith
177c939e48SSatish Balay  def __str2__(self):
185b4fc442SVaclav Hapla    desc = ['  Using GNU make: ' + self.make.make]
19*7ce81a4bSJacob Faibussowitsch    if self.defines.get('USE_COVERAGE'):
20*7ce81a4bSJacob Faibussowitsch      desc.extend([
21*7ce81a4bSJacob Faibussowitsch        '  Code coverage: yes',
22*7ce81a4bSJacob Faibussowitsch        '  Using code coverage executable: {}'.format(self.getMakeMacro('PETSC_COVERAGE_EXEC'))
23*7ce81a4bSJacob Faibussowitsch      ])
24aa5c8b8eSBarry Smith    if not self.installed:
25a0022257SSatish Balay      desc.append('xxx=========================================================================xxx')
26dc0529c6SBarry Smith      desc.append(' Configure stage complete. Now build PETSc libraries with:')
275b4fc442SVaclav Hapla      desc.append('   %s PETSC_DIR=%s PETSC_ARCH=%s all' % (self.make.make_user, self.petscdir.dir, self.arch.arch))
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')
33*7ce81a4bSJacob Faibussowitsch    desc.append('')
34*7ce81a4bSJacob Faibussowitsch    return '\n'.join(desc)
35f8833479SBarry Smith
36f8833479SBarry Smith  def setupHelp(self, help):
37f8833479SBarry Smith    import nargs
38ce0b2093SBarry Smith    help.addArgument('PETSc',  '-prefix=<dir>',                              nargs.Arg(None, '', 'Specifiy location to install PETSc (eg. /usr/local)'))
397deb5ab3SBarry Smith    help.addArgument('PETSc',  '-with-prefetch=<bool>',                      nargs.ArgBool(None, 1,'Enable checking for prefetch instructions'))
40eed94e11SSatish Balay    help.addArgument('Windows','-with-windows-graphics=<bool>',              nargs.ArgBool(None, 1,'Enable check for Windows Graphics'))
41569865ddSSatish Balay    help.addArgument('PETSc', '-with-default-arch=<bool>',                   nargs.ArgBool(None, 1, 'Allow using the last configured arch without setting PETSC_ARCH'))
4257cb31baSSatish Balay    help.addArgument('PETSc','-with-single-library=<bool>',                  nargs.ArgBool(None, 1,'Put all PETSc code into the single -lpetsc library'))
43cb297985SSatish Balay    help.addArgument('PETSc','-with-fortran-bindings=<bool>',                nargs.ArgBool(None, 1,'Build PETSc fortran bindings in the library and corresponding module files'))
44525d6f2eSBarry Smith    help.addArgument('PETSc', '-with-ios=<bool>',                            nargs.ArgBool(None, 0, 'Build an iPhone/iPad version of PETSc library'))
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'))
47*7ce81a4bSJacob Faibussowitsch    help.addArgument('PETSc', '-with-coverage=<bool>',                       nargs.ArgFuzzyBool(None, value=0, help='Enable or disable code-coverage collection'))
48*7ce81a4bSJacob Faibussowitsch    help.addArgument('PETSc', '-with-coverage-exec=<executable>',            nargs.ArgExecutable(None, value='default-auto', mustExist=0, help='Name of executable to use for post-processing coverage data, e.g. \'gcov\' or \'llvm-cov\'. Pass \'auto\' to let configure infer from compiler'))
49f8833479SBarry Smith    return
50f8833479SBarry Smith
516dd73af6SBarry Smith  def registerPythonFile(self,filename,directory):
526dd73af6SBarry Smith    ''' Add a python file to the framework and registers its headerprefix, ... externalpackagedir
536dd73af6SBarry Smith        directory is the directory where the file relative to the BuildSystem or config path in python notation with . '''
546dd73af6SBarry Smith    (utilityName, ext) = os.path.splitext(filename)
556dd73af6SBarry Smith    if not utilityName.startswith('.') and not utilityName.startswith('#') and ext == '.py' and not utilityName == '__init__':
566dd73af6SBarry Smith      if directory: directory = directory+'.'
576dd73af6SBarry Smith      utilityObj                             = self.framework.require(directory+utilityName, self)
586dd73af6SBarry Smith      utilityObj.headerPrefix                = self.headerPrefix
596dd73af6SBarry Smith      utilityObj.archProvider                = self.arch
606dd73af6SBarry Smith      utilityObj.languageProvider            = self.languages
616dd73af6SBarry Smith      utilityObj.installDirProvider          = self.installdir
626dd73af6SBarry Smith      utilityObj.externalPackagesDirProvider = self.externalpackagesdir
636dd73af6SBarry Smith      utilityObj.precisionProvider           = self.scalartypes
646dd73af6SBarry Smith      utilityObj.indexProvider               = self.indexTypes
656dd73af6SBarry Smith      setattr(self, utilityName.lower(), utilityObj)
6651294b80SMatthew G. Knepley      return utilityObj
6751294b80SMatthew G. Knepley    return None
686dd73af6SBarry Smith
69f8833479SBarry Smith  def setupDependencies(self, framework):
70f8833479SBarry Smith    config.base.Configure.setupDependencies(self, framework)
71dca78d2bSSatish Balay    self.programs      = framework.require('config.programs',           self)
72f8833479SBarry Smith    self.setCompilers  = framework.require('config.setCompilers',       self)
73*7ce81a4bSJacob Faibussowitsch    self.compilerFlags = framework.require('config.compilerFlags',      self)
7430b8aa07SMatthew G. Knepley    self.compilers     = framework.require('config.compilers',          self)
759d310bb7SBarry Smith    self.arch          = framework.require('PETSc.options.arch',        self.setCompilers)
769d310bb7SBarry Smith    self.petscdir      = framework.require('PETSc.options.petscdir',    self.arch)
779d310bb7SBarry Smith    self.installdir    = framework.require('PETSc.options.installDir',  self)
784e00a515SSatish Balay    self.dataFilesPath = framework.require('PETSc.options.dataFilesPath',self)
796dd73af6SBarry Smith    self.scalartypes   = framework.require('PETSc.options.scalarTypes', self)
806dd73af6SBarry Smith    self.indexTypes    = framework.require('PETSc.options.indexTypes',  self)
819d310bb7SBarry Smith    self.languages     = framework.require('PETSc.options.languages',   self.setCompilers)
8230b8aa07SMatthew G. Knepley    self.indexTypes    = framework.require('PETSc.options.indexTypes',  self.compilers)
83f8833479SBarry Smith    self.types         = framework.require('config.types',              self)
84f8833479SBarry Smith    self.headers       = framework.require('config.headers',            self)
85f8833479SBarry Smith    self.functions     = framework.require('config.functions',          self)
86f8833479SBarry Smith    self.libraries     = framework.require('config.libraries',          self)
87cd37d877SShri Abhyankar    self.atomics       = framework.require('config.atomics',            self)
889481793eSSatish Balay    self.make          = framework.require('config.packages.make',      self)
899552296fSBarry Smith    self.blasLapack    = framework.require('config.packages.BlasLapack',self)
90e6b0c433SBarry Smith    self.mpi           = framework.require('config.packages.MPI',       self)
910542e31aSBarry Smith    self.fortran       = framework.require('config.compilersFortran',   self)
920542e31aSBarry Smith    self.externalpackagesdir = framework.require('PETSc.options.externalpackagesdir',self)
9349d43ecaSSatish Balay
9409a6cbfcSBernhard M. Wiedemann    for utility in sorted(os.listdir(os.path.join('config','PETSc','options'))):
956dd73af6SBarry Smith      self.registerPythonFile(utility,'PETSc.options')
969d310bb7SBarry Smith
9709a6cbfcSBernhard M. Wiedemann    for utility in sorted(os.listdir(os.path.join('config','BuildSystem','config','utilities'))):
986dd73af6SBarry Smith      self.registerPythonFile(utility,'config.utilities')
9906e08bc7SBarry Smith
10009a6cbfcSBernhard M. Wiedemann    for package in sorted(os.listdir(os.path.join('config', 'BuildSystem', 'config', 'packages'))):
10151294b80SMatthew G. Knepley      obj = self.registerPythonFile(package,'config.packages')
10251294b80SMatthew G. Knepley      if obj:
10351294b80SMatthew G. Knepley        obj.archProvider                = self.framework.requireModule(obj.archProvider, obj)
10451294b80SMatthew G. Knepley        obj.languageProvider            = self.framework.requireModule(obj.languageProvider, obj)
10551294b80SMatthew G. Knepley        obj.installDirProvider          = self.framework.requireModule(obj.installDirProvider, obj)
10651294b80SMatthew G. Knepley        obj.externalPackagesDirProvider = self.framework.requireModule(obj.externalPackagesDirProvider, obj)
10751294b80SMatthew G. Knepley        obj.precisionProvider           = self.framework.requireModule(obj.precisionProvider, obj)
10851294b80SMatthew G. Knepley        obj.indexProvider               = self.framework.requireModule(obj.indexProvider, obj)
1096dd73af6SBarry Smith
1105faf1eacSMatthew G. Knepley    # Force blaslapack and opencl to depend on scalarType so precision is set before BlasLapack is built
1119d310bb7SBarry Smith    framework.require('PETSc.options.scalarTypes', self.f2cblaslapack)
1129d310bb7SBarry Smith    framework.require('PETSc.options.scalarTypes', self.fblaslapack)
1139d310bb7SBarry Smith    framework.require('PETSc.options.scalarTypes', self.blaslapack)
1145faf1eacSMatthew G. Knepley    framework.require('PETSc.options.scalarTypes', self.opencl)
115f8833479SBarry Smith
116dca78d2bSSatish Balay    self.programs.headerPrefix     = self.headerPrefix
1175b3958d3SJacob Faibussowitsch    self.setCompilers.headerPrefix = self.headerPrefix
118f8833479SBarry Smith    self.compilers.headerPrefix    = self.headerPrefix
1190542e31aSBarry Smith    self.fortran.headerPrefix      = self.headerPrefix
120f8833479SBarry Smith    self.types.headerPrefix        = self.headerPrefix
121f8833479SBarry Smith    self.headers.headerPrefix      = self.headerPrefix
122f8833479SBarry Smith    self.functions.headerPrefix    = self.headerPrefix
123f8833479SBarry Smith    self.libraries.headerPrefix    = self.headerPrefix
1246dd73af6SBarry Smith
1252c30b4dfSSatish Balay    # Register user provided package scripts
1262c30b4dfSSatish Balay    if 'with-package-scripts' in self.framework.argDB:
1272c30b4dfSSatish Balay      for script in self.framework.argDB['with-package-scripts']:
1282c30b4dfSSatish Balay        if os.path.splitext(script)[1] != '.py':
1292c30b4dfSSatish Balay          raise RuntimeError('Only python scripts compatible with configure package script format should be specified! Invalid option -with-package-scripts='+script)
1302c30b4dfSSatish Balay        self.framework.logPrint('User is registering a new package script: '+script)
1312c30b4dfSSatish Balay        dname,fname = os.path.split(script)
1322c30b4dfSSatish Balay        if dname: sys.path.append(dname)
1332c30b4dfSSatish Balay        self.registerPythonFile(fname,'')
1346dd73af6SBarry Smith
1356dd73af6SBarry Smith    # test for a variety of basic headers and functions
1364211eb48SBarry Smith    headersC = map(lambda name: name+'.h',['setjmp','dos','fcntl','float','io','malloc','pwd','strings',
137ba61063dSBarry Smith                                            'unistd','sys/sysinfo','machine/endian','sys/param','sys/procfs','sys/resource',
1382475b7caSBarry Smith                                            'sys/systeminfo','sys/times','sys/utsname',
1397e4f0192SMosè Giordano                                            'sys/socket','sys/wait','netinet/in','netdb','direct','time','Ws2tcpip','sys/types',
1402475b7caSBarry Smith                                            'WindowsX','float','ieeefp','stdint','pthread','inttypes','immintrin','zmmintrin'])
14145082d64SJed Brown    functions = ['access','_access','clock','drand48','getcwd','_getcwd','getdomainname','gethostname',
142d7976ebaSJed Brown                 'getwd','posix_memalign','popen','PXFGETARG','rand','getpagesize',
1434211eb48SBarry Smith                 'readlink','realpath','usleep','sleep','_sleep',
1442475b7caSBarry Smith                 'uname','snprintf','_snprintf','lseek','_lseek','time','fork','stricmp',
1452475b7caSBarry Smith                 'strcasecmp','bzero','dlopen','dlsym','dlclose','dlerror',
146aaf3846cSSatish Balay                 '_set_output_format','_mkdir','socket','gethostbyname','_pipe','fpresetsticky',
1470866c665SSatish Balay                 'fpsetsticky','__gcov_dump','fstatat']
148b0651e32SBarry Smith    libraries = [(['fpe'],'handle_sigfpes')]
149b0651e32SBarry Smith    librariessock = [(['socket','nsl'],'socket')]
150f8833479SBarry Smith    self.headers.headers.extend(headersC)
151f8833479SBarry Smith    self.functions.functions.extend(functions)
152b0651e32SBarry Smith    self.libraries.libraries.extend(libraries)
153b0651e32SBarry Smith    if not hasattr(self,'socket'):
154b0651e32SBarry Smith      self.libraries.libraries.extend(librariessock)
155f8833479SBarry Smith    return
156f8833479SBarry Smith
1578244ab14SJed Brown  def DumpPkgconfig(self, petsc_pc):
158262119f8SBarry Smith    ''' Create a pkg-config file '''
159262119f8SBarry Smith    if not os.path.exists(os.path.join(self.petscdir.dir,self.arch.arch,'lib','pkgconfig')):
160262119f8SBarry Smith      os.makedirs(os.path.join(self.petscdir.dir,self.arch.arch,'lib','pkgconfig'))
1612eefe1c6SJed Brown    with open(os.path.join(self.petscdir.dir,self.arch.arch,'lib','pkgconfig',petsc_pc),'w') as fd:
1625e3311eeSJed Brown      cflags_inc = ['-I${includedir}']
163262119f8SBarry Smith      if self.framework.argDB['prefix']:
1645bb5b263SMatthew G. Knepley        fd.write('prefix='+self.installdir.dir+'\n')
165262119f8SBarry Smith      else:
166e1e675deSJed Brown        fd.write('prefix='+os.path.join(self.petscdir.dir, self.arch.arch)+'\n')
167e1e675deSJed Brown        cflags_inc.append('-I' + os.path.join(self.petscdir.dir, 'include'))
168262119f8SBarry Smith      fd.write('exec_prefix=${prefix}\n')
169262119f8SBarry Smith      fd.write('includedir=${prefix}/include\n')
1705e3311eeSJed Brown      fd.write('libdir=${prefix}/lib\n')
171262119f8SBarry Smith
1722eefe1c6SJed Brown      with self.setCompilers.Language('C'):
173262119f8SBarry Smith        fd.write('ccompiler='+self.setCompilers.getCompiler()+'\n')
174756c7f9fSJed Brown        fd.write('cflags_extra='+self.setCompilers.getCompilerFlags().strip()+'\n')
175756c7f9fSJed Brown        fd.write('cflags_dep='+self.compilers.dependenciesGenerationFlag.get('C','')+'\n')
176756c7f9fSJed Brown        fd.write('ldflag_rpath='+self.setCompilers.CSharedLinkerFlag+'\n')
17703e383c8SJed Brown      if hasattr(self.compilers, 'CXX'):
1782eefe1c6SJed Brown        with self.setCompilers.Language('C++'):
179262119f8SBarry Smith          fd.write('cxxcompiler='+self.setCompilers.getCompiler()+'\n')
180756c7f9fSJed Brown          fd.write('cxxflags_extra='+self.setCompilers.getCompilerFlags().strip()+'\n')
181262119f8SBarry Smith      if hasattr(self.compilers, 'FC'):
1822eefe1c6SJed Brown        with self.setCompilers.Language('FC'):
183262119f8SBarry Smith          fd.write('fcompiler='+self.setCompilers.getCompiler()+'\n')
184756c7f9fSJed Brown          fd.write('fflags_extra='+self.setCompilers.getCompilerFlags().strip()+'\n')
18550520af6SPatrick Sanan      if hasattr(self.compilers, 'CUDAC'):
18650520af6SPatrick Sanan        with self.setCompilers.Language('CUDA'):
18750520af6SPatrick Sanan          fd.write('cudacompiler='+self.setCompilers.getCompiler()+'\n')
18850520af6SPatrick Sanan          fd.write('cudaflags_extra='+self.setCompilers.getCompilerFlags().strip()+'\n')
1898f561fa3SPatrick Sanan          p = self.framework.require('config.packages.cuda')
19050520af6SPatrick Sanan          fd.write('cudalib='+self.libraries.toStringNoDupes(p.lib)+'\n')
19150520af6SPatrick Sanan          fd.write('cudainclude='+self.headers.toStringNoDupes(p.include)+'\n')
1927ba7a817SBarry Smith          if hasattr(self.setCompilers,'CUDA_CXX'):
1937ba7a817SBarry Smith            fd.write('cuda_cxx='+self.setCompilers.CUDA_CXX+'\n')
1947ba7a817SBarry Smith            fd.write('cuda_cxxflags='+self.setCompilers.CUDA_CXXFLAGS+'\n')
195262119f8SBarry Smith
196262119f8SBarry Smith      fd.write('\n')
197262119f8SBarry Smith      fd.write('Name: PETSc\n')
198262119f8SBarry Smith      fd.write('Description: Library to solve ODEs and algebraic equations\n')
199351d3a41SMatthew G Knepley      fd.write('Version: %s\n' % self.petscdir.version)
2005e3311eeSJed Brown      fd.write('Cflags: ' + ' '.join([self.setCompilers.CPPFLAGS] + cflags_inc) + '\n')
20137371b91SJed Brown      fd.write('Libs: '+self.libraries.toStringNoDupes(['-L${libdir}', self.petsclib], with_rpath=False)+'\n')
2028ebf8858SJed Brown      # Remove RPATH flags from library list.  User can add them using
2038ebf8858SJed Brown      # pkg-config --variable=ldflag_rpath and pkg-config --libs-only-L
204de8f682fSSatish 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')
205262119f8SBarry Smith    return
206262119f8SBarry Smith
207351d3a41SMatthew G Knepley  def DumpModule(self):
208351d3a41SMatthew G Knepley    ''' Create a module file '''
209af0996ceSBarry Smith    if not os.path.exists(os.path.join(self.petscdir.dir,self.arch.arch,'lib','petsc','conf','modules')):
210af0996ceSBarry Smith      os.makedirs(os.path.join(self.petscdir.dir,self.arch.arch,'lib','petsc','conf','modules'))
211af0996ceSBarry Smith    if not os.path.exists(os.path.join(self.petscdir.dir,self.arch.arch,'lib','petsc','conf','modules','petsc')):
212af0996ceSBarry Smith      os.makedirs(os.path.join(self.petscdir.dir,self.arch.arch,'lib','petsc','conf','modules','petsc'))
213351d3a41SMatthew G Knepley    if self.framework.argDB['prefix']:
2145bb5b263SMatthew G. Knepley      installdir  = self.installdir.dir
21555d606a3SSatish Balay      installarch = ''
21655d606a3SSatish Balay      installpath = os.path.join(installdir,'bin')
217351d3a41SMatthew G Knepley    else:
218351d3a41SMatthew G Knepley      installdir  = self.petscdir.dir
21955d606a3SSatish Balay      installarch = self.arch.arch
22055d606a3SSatish Balay      installpath = os.path.join(installdir,installarch,'bin')+':'+os.path.join(installdir,'bin')
221af0996ceSBarry Smith    fd = open(os.path.join(self.petscdir.dir,self.arch.arch,'lib','petsc','conf','modules','petsc',self.petscdir.version),'w')
222351d3a41SMatthew G Knepley    fd.write('''\
223351d3a41SMatthew G Knepley#%%Module
224351d3a41SMatthew G Knepley
225351d3a41SMatthew G Knepleyproc ModulesHelp { } {
226351d3a41SMatthew G Knepley    puts stderr "This module sets the path and environment variables for petsc-%s"
227a17b96a8SKyle Gerard Felker    puts stderr "     see https://petsc.org/ for more information      "
228351d3a41SMatthew G Knepley    puts stderr ""
229351d3a41SMatthew G Knepley}
230351d3a41SMatthew G Knepleymodule-whatis "PETSc - Portable, Extensible Toolkit for Scientific Computation"
231351d3a41SMatthew G Knepley
232dd486775SJed Brownset petsc_dir   "%s"
233dd486775SJed Brownset petsc_arch  "%s"
234351d3a41SMatthew G Knepley
235dd486775SJed Brownsetenv PETSC_ARCH "$petsc_arch"
236dd486775SJed Brownsetenv PETSC_DIR "$petsc_dir"
237dd486775SJed Brownprepend-path PATH "%s"
23855d606a3SSatish Balay''' % (self.petscdir.version, installdir, installarch, installpath))
239351d3a41SMatthew G Knepley    fd.close()
240351d3a41SMatthew G Knepley    return
241351d3a41SMatthew G Knepley
242f8833479SBarry Smith  def Dump(self):
243f8833479SBarry Smith    ''' Actually put the values into the configuration files '''
244f8833479SBarry Smith    # eventually everything between -- should be gone
24517f368bcSBarry Smith    if self.mpi.usingMPIUni:
24617f368bcSBarry Smith      #
24717f368bcSBarry Smith      # Remove any MPI/MPICH include files that may have been put here by previous runs of ./configure
2487908f030SMatthew 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)
24917f368bcSBarry Smith
250b5f71184SBarry Smith    self.logPrintDivider()
251b5f71184SBarry Smith    # Test for compiler-specific macros that need to be defined.
252b5f71184SBarry Smith    if self.setCompilers.isCrayVector('CC', self.log):
253b5f71184SBarry Smith      self.addDefine('HAVE_CRAY_VECTOR','1')
254b5f71184SBarry Smith
255b5f71184SBarry Smith    if self.functions.haveFunction('gethostbyname') and self.functions.haveFunction('socket') and self.headers.haveHeader('netinet/in.h'):
256b5f71184SBarry Smith      self.addDefine('USE_SOCKET_VIEWER','1')
257b5f71184SBarry Smith      if self.checkCompile('#include <sys/socket.h>','setsockopt(0,SOL_SOCKET,SO_REUSEADDR,0,0)'):
258b5f71184SBarry Smith        self.addDefine('HAVE_SO_REUSEADDR','1')
259b5f71184SBarry Smith
260b5f71184SBarry Smith    self.logPrintDivider()
2615f27b2e0SBarry Smith    self.setCompilers.pushLanguage('C')
2625f27b2e0SBarry Smith    compiler = self.setCompilers.getCompiler()
263217fe27eSSatish Balay    if [s for s in ['mpicc','mpiicc'] if os.path.basename(compiler).find(s)>=0]:
2645f27b2e0SBarry Smith      try:
2655f27b2e0SBarry Smith        output   = self.executeShellCommand(compiler + ' -show', log = self.log)[0]
2665f27b2e0SBarry Smith        compiler = output.split(' ')[0]
267f424265bSStefano Zampini        self.addDefine('MPICC_SHOW','"'+output.strip().replace('\n','\\\\n').replace('"','')+'"')
2685f27b2e0SBarry Smith      except:
269c9872b61SBarry Smith        self.addDefine('MPICC_SHOW','"Unavailable"')
270c9872b61SBarry Smith    else:
271c9872b61SBarry Smith      self.addDefine('MPICC_SHOW','"Unavailable"')
2725f27b2e0SBarry Smith    self.setCompilers.popLanguage()
273f8833479SBarry Smith#-----------------------------------------------------------------------------------------------------
274f8833479SBarry Smith
275f8833479SBarry Smith    # Sometimes we need C compiler, even if built with C++
276f8833479SBarry Smith    self.setCompilers.pushLanguage('C')
277e4c30378SBarry Smith    # do not use getCompilerFlags() because that automatically includes the CPPFLAGS so one ends up with duplication flags in makefile usage
278e4c30378SBarry Smith    self.addMakeMacro('CC_FLAGS',self.setCompilers.CFLAGS)
279f8833479SBarry Smith    self.setCompilers.popLanguage()
280f8833479SBarry Smith
28134f774f6SJed Brown    # And sometimes we need a C++ compiler even when PETSc is built with C
28234f774f6SJed Brown    if hasattr(self.compilers, 'CXX'):
28334f774f6SJed Brown      self.setCompilers.pushLanguage('Cxx')
28429921a8fSScott Kruger      self.addDefine('HAVE_CXX','1')
2850b119762SSatish Balay      self.addMakeMacro('CXXPP_FLAGS',self.setCompilers.CXXPPFLAGS)
286e4c30378SBarry Smith      # do not use getCompilerFlags() because that automatically includes the CXXPPFLAGS so one ends up with duplication flags in makefile usage
287e4c30378SBarry Smith      self.addMakeMacro('CXX_FLAGS',self.setCompilers.CXXFLAGS+' '+self.setCompilers.CXX_CXXFLAGS)
2882f4326f3SSatish Balay      cxx_linker = self.setCompilers.getLinker()
2892f4326f3SSatish Balay      self.addMakeMacro('CXX_LINKER',cxx_linker)
2902f4326f3SSatish Balay      self.addMakeMacro('CXX_LINKER_FLAGS',self.setCompilers.getLinkerFlags())
29134f774f6SJed Brown      self.setCompilers.popLanguage()
29234f774f6SJed Brown
293f8833479SBarry Smith    # C preprocessor values
2941315f054SBarry Smith    self.addMakeMacro('CPP_FLAGS',self.setCompilers.CPPFLAGS)
295f8833479SBarry Smith
296f8833479SBarry Smith    # compiler values
297f8833479SBarry Smith    self.setCompilers.pushLanguage(self.languages.clanguage)
298f8833479SBarry Smith    self.addMakeMacro('PCC',self.setCompilers.getCompiler())
299e4c30378SBarry Smith    # do not use getCompilerFlags() because that automatically includes the preprocessor flags so one ends up with duplication flags in makefile usage
300e4c30378SBarry Smith    if self.languages.clanguage == 'C':
301e4c30378SBarry Smith      self.addMakeMacro('PCC_FLAGS','$(CC_FLAGS)')
302e4c30378SBarry Smith    else:
303e4c30378SBarry Smith      self.addMakeMacro('PCC_FLAGS','$(CXX_FLAGS)')
304f8833479SBarry Smith    self.setCompilers.popLanguage()
305f8833479SBarry Smith    # .o or .obj
306f8833479SBarry Smith    self.addMakeMacro('CC_SUFFIX','o')
307f8833479SBarry Smith
308f8833479SBarry Smith    # executable linker values
309f8833479SBarry Smith    self.setCompilers.pushLanguage(self.languages.clanguage)
310f8833479SBarry Smith    pcc_linker = self.setCompilers.getLinker()
311f8833479SBarry Smith    self.addMakeMacro('PCC_LINKER',pcc_linker)
312c84a332bSSatish Balay    self.addMakeMacro('PCC_LINKER_FLAGS',self.setCompilers.getLinkerFlags())
313f8833479SBarry Smith    self.setCompilers.popLanguage()
314f8833479SBarry Smith    # '' for Unix, .exe for Windows
315f8833479SBarry Smith    self.addMakeMacro('CC_LINKER_SUFFIX','')
316f8833479SBarry Smith
317f8833479SBarry Smith    if hasattr(self.compilers, 'FC'):
318cb297985SSatish Balay      if self.framework.argDB['with-fortran-bindings']:
319257f4e5aSSatish Balay        if not self.fortran.fortranIsF90:
320257f4e5aSSatish Balay          raise RuntimeError('Error! Fortran compiler "'+self.compilers.FC+'" does not support F90! PETSc fortran bindings require a F90 compiler')
321cb297985SSatish Balay        self.addDefine('HAVE_FORTRAN','1')
322f8833479SBarry Smith      self.setCompilers.pushLanguage('FC')
323f8833479SBarry Smith      # need FPPFLAGS in config/setCompilers
3240b119762SSatish Balay      self.addMakeMacro('FPP_FLAGS',self.setCompilers.FPPFLAGS)
325f8833479SBarry Smith
326f8833479SBarry Smith      # compiler values
327f8833479SBarry Smith      self.addMakeMacro('FC_FLAGS',self.setCompilers.getCompilerFlags())
328f8833479SBarry Smith      self.setCompilers.popLanguage()
329f8833479SBarry Smith      # .o or .obj
330f8833479SBarry Smith      self.addMakeMacro('FC_SUFFIX','o')
331f8833479SBarry Smith
332f8833479SBarry Smith      # executable linker values
333f8833479SBarry Smith      self.setCompilers.pushLanguage('FC')
334a9acdec7SBarry Smith      self.addMakeMacro('FC_LINKER',self.setCompilers.getLinker())
3356d53d35eSSatish Balay      self.addMakeMacro('FC_LINKER_FLAGS',self.setCompilers.getLinkerFlags())
3363feacd00SBarry Smith      # apple requires this shared library linker flag on SOME versions of the os
3373feacd00SBarry Smith      if self.setCompilers.getLinkerFlags().find('-Wl,-commons,use_dylibs') > -1:
3383feacd00SBarry Smith        self.addMakeMacro('DARWIN_COMMONS_USE_DYLIBS',' -Wl,-commons,use_dylibs ')
339bb82cf9cSSatish Balay      self.setCompilers.popLanguage()
3405d631499SMatthew Knepley
3415d631499SMatthew Knepley      # F90 Modules
3425d631499SMatthew Knepley      if self.setCompilers.fortranModuleIncludeFlag:
3435d631499SMatthew Knepley        self.addMakeMacro('FC_MODULE_FLAG', self.setCompilers.fortranModuleIncludeFlag)
3446ddd6694SSatish Balay      else: # for non-f90 compilers like g77
3456ddd6694SSatish Balay        self.addMakeMacro('FC_MODULE_FLAG', '-I')
346a324c51cSMatthew G Knepley      if self.setCompilers.fortranModuleIncludeFlag:
347a324c51cSMatthew G Knepley        self.addMakeMacro('FC_MODULE_OUTPUT_FLAG', self.setCompilers.fortranModuleOutputFlag)
348f8833479SBarry Smith    else:
349f8833479SBarry Smith      self.addMakeMacro('FC','')
350f8833479SBarry Smith
35146a3958fSBarry Smith    if hasattr(self.compilers, 'CUDAC'):
3527ff2890cSSatish Balay      self.setCompilers.pushLanguage('CUDA')
353d93a25ecSSatish Balay      self.addMakeMacro('CUDAC_FLAGS',self.setCompilers.getCompilerFlags())
35450dcbc5aSJunchao Zhang      self.addMakeMacro('CUDAPP_FLAGS',self.setCompilers.CUDAPPFLAGS)
3557ff2890cSSatish Balay      self.setCompilers.popLanguage()
3567ff2890cSSatish Balay
357694a2f0eSJunchao Zhang    if hasattr(self.compilers, 'HIPC'):
35828f796eaSScott Kruger      self.setCompilers.pushLanguage('HIP')
359694a2f0eSJunchao Zhang      self.addMakeMacro('HIPC_FLAGS',self.setCompilers.getCompilerFlags())
3607fb1458fSStefano Zampini      self.addMakeMacro('HIPPP_FLAGS',self.setCompilers.HIPPPFLAGS)
36128f796eaSScott Kruger      self.setCompilers.popLanguage()
36228f796eaSScott Kruger
36350dcbc5aSJunchao Zhang    if hasattr(self.compilers, 'SYCLC'):
36428f796eaSScott Kruger      self.setCompilers.pushLanguage('SYCL')
36550dcbc5aSJunchao Zhang      self.addMakeMacro('SYCLC_FLAGS',self.setCompilers.getCompilerFlags())
36650dcbc5aSJunchao Zhang      self.addMakeMacro('SYCLPP_FLAGS',self.setCompilers.SYCLPPFLAGS)
36728f796eaSScott Kruger      self.setCompilers.popLanguage()
36828f796eaSScott Kruger
369f8833479SBarry Smith    # shared library linker values
370f8833479SBarry Smith    self.setCompilers.pushLanguage(self.languages.clanguage)
371f8833479SBarry Smith    # need to fix BuildSystem to collect these separately
372f8833479SBarry Smith    self.addMakeMacro('SL_LINKER',self.setCompilers.getLinker())
37370db8aa6SSatish Balay    self.addMakeMacro('SL_LINKER_FLAGS','${PCC_LINKER_FLAGS}')
374f8833479SBarry Smith    self.setCompilers.popLanguage()
375f8833479SBarry Smith    # One of 'a', 'so', 'lib', 'dll', 'dylib' (perhaps others also?) depending on the library generator and architecture
376f8833479SBarry Smith    # Note: . is not included in this macro, consistent with AR_LIB_SUFFIX
377f8833479SBarry Smith    if self.setCompilers.sharedLibraryExt == self.setCompilers.AR_LIB_SUFFIX:
378f8833479SBarry Smith      self.addMakeMacro('SL_LINKER_SUFFIX', '')
37946bc77b6SBarry Smith      self.addDefine('SLSUFFIX','""')
380f8833479SBarry Smith    else:
381f8833479SBarry Smith      self.addMakeMacro('SL_LINKER_SUFFIX', self.setCompilers.sharedLibraryExt)
38246bc77b6SBarry Smith      self.addDefine('SLSUFFIX','"'+self.setCompilers.sharedLibraryExt+'"')
383bb82cf9cSSatish Balay
38423e93537SBarry Smith    self.addMakeMacro('SL_LINKER_LIBS','${PETSC_EXTERNAL_LIB_BASIC}')
385bb82cf9cSSatish Balay
386f8833479SBarry Smith#-----------------------------------------------------------------------------------------------------
387f8833479SBarry Smith
388f8833479SBarry Smith    # CONLY or CPP. We should change the PETSc makefiles to do this better
389f8833479SBarry Smith    if self.languages.clanguage == 'C': lang = 'CONLY'
390f8833479SBarry Smith    else: lang = 'CXXONLY'
391f8833479SBarry Smith    self.addMakeMacro('PETSC_LANGUAGE',lang)
392f8833479SBarry Smith
393f8833479SBarry Smith    # real or complex
394f8833479SBarry Smith    self.addMakeMacro('PETSC_SCALAR',self.scalartypes.scalartype)
395f8833479SBarry Smith    # double or float
396f8833479SBarry Smith    self.addMakeMacro('PETSC_PRECISION',self.scalartypes.precision)
397f8833479SBarry Smith
398f8833479SBarry Smith    if self.framework.argDB['with-batch']:
399f8833479SBarry Smith      self.addMakeMacro('PETSC_WITH_BATCH','1')
400f8833479SBarry Smith
401f8833479SBarry Smith#-----------------------------------------------------------------------------------------------------
402a6cc6bb1SBarry Smith    # print include and lib for makefiles
403b5f71184SBarry Smith    self.logPrintDivider()
404f8833479SBarry Smith    self.framework.packages.reverse()
4055a21677cSJed Brown    petscincludes = [os.path.join(self.petscdir.dir,'include'),os.path.join(self.petscdir.dir,self.arch.arch,'include')]
4068749e224SJunchao Zhang    petscincludes_install = [os.path.join(self.installdir.dir, 'include')] if self.framework.argDB['prefix'] else petscincludes
4075a21677cSJed Brown    includes = []
408de8f682fSSatish Balay    self.packagelibs = []
409f8833479SBarry Smith    for i in self.framework.packages:
4107f0ff1afSBarry Smith      if not i.required:
4113972cb20SJacob Faibussowitsch        if i.devicePackage:
4123972cb20SJacob Faibussowitsch          self.addDefine('HAVE_DEVICE',1)
413eeb16384SBarry Smith        self.addDefine('HAVE_'+i.PACKAGE.replace('-','_'), 1)  # ONLY list package if it is used directly by PETSc (and not only by another package)
414f8833479SBarry Smith      if not isinstance(i.lib, list):
415f8833479SBarry Smith        i.lib = [i.lib]
416de8f682fSSatish Balay      if i.linkedbypetsc: self.packagelibs.extend(i.lib)
417eeb16384SBarry Smith      self.addMakeMacro(i.PACKAGE.replace('-','_')+'_LIB', self.libraries.toStringNoDupes(i.lib))
418f8833479SBarry Smith      if hasattr(i,'include'):
419f8833479SBarry Smith        if not isinstance(i.include,list):
420f8833479SBarry Smith          i.include = [i.include]
421ac9e4c42SSatish Balay        includes.extend(i.include)
422eeb16384SBarry Smith        self.addMakeMacro(i.PACKAGE.replace('-','_')+'_INCLUDE',self.headers.toStringNoDupes(i.include))
4232df986feSBarry Smith    if self.framework.argDB['with-single-library']:
424e282ce78SJed Brown      self.petsclib = '-lpetsc'
42591bb3077SSatish Balay    else:
426e282ce78SJed Brown      self.petsclib = '-lpetscts -lpetscsnes -lpetscksp -lpetscdm -lpetscmat -lpetscvec -lpetscsys'
427de8f682fSSatish Balay    self.complibs = self.compilers.flibs+self.compilers.cxxlibs+self.compilers.LIBS.split()
4285a21677cSJed Brown    self.PETSC_WITH_EXTERNAL_LIB = self.libraries.toStringNoDupes(['-L${PETSC_DIR}/${PETSC_ARCH}/lib', self.petsclib]+self.packagelibs+self.complibs)
429de8f682fSSatish Balay    self.PETSC_EXTERNAL_LIB_BASIC = self.libraries.toStringNoDupes(self.packagelibs+self.complibs)
430de8f682fSSatish Balay
431de8f682fSSatish Balay    self.addMakeMacro('PETSC_EXTERNAL_LIB_BASIC',self.PETSC_EXTERNAL_LIB_BASIC)
4325a21677cSJed Brown    allincludes = petscincludes + includes
4335a21677cSJed Brown    allincludes_install = petscincludes_install + includes
4345a21677cSJed Brown    self.PETSC_CC_INCLUDES = self.headers.toStringNoDupes(allincludes)
4355a21677cSJed Brown    self.PETSC_CC_INCLUDES_INSTALL = self.headers.toStringNoDupes(allincludes_install)
4365a21677cSJed Brown    self.addMakeMacro('PETSC_CC_INCLUDES',self.PETSC_CC_INCLUDES)
4375a21677cSJed Brown    self.addMakeMacro('PETSC_CC_INCLUDES_INSTALL', self.PETSC_CC_INCLUDES_INSTALL)
438cbd5cc15SBarry Smith    if hasattr(self.compilers, 'FC'):
4395a21677cSJed Brown      def modinc(includes):
4400542e31aSBarry Smith        return includes if self.fortran.fortranIsF90 else []
4415a21677cSJed Brown      self.addMakeMacro('PETSC_FC_INCLUDES',self.headers.toStringNoDupes(allincludes,modinc(allincludes)))
4425a21677cSJed Brown      self.addMakeMacro('PETSC_FC_INCLUDES_INSTALL',self.headers.toStringNoDupes(allincludes_install,modinc(allincludes_install)))
443f8833479SBarry Smith
4445bb5b263SMatthew G. Knepley    self.addDefine('LIB_DIR','"'+os.path.join(self.installdir.dir,'lib')+'"')
445f8833479SBarry Smith
4460f3b21c2SBarry Smith    if self.framework.argDB['with-single-library']:
4470f3b21c2SBarry Smith      # overrides the values set in conf/variables
4480f3b21c2SBarry Smith      self.addMakeMacro('LIBNAME','${INSTALL_LIB_DIR}/libpetsc.${AR_LIB_SUFFIX}')
44957cb31baSSatish Balay      self.addMakeMacro('SHLIBS','libpetsc')
450bccf1c12SBarry Smith      self.addMakeMacro('PETSC_LIB_BASIC','-lpetsc')
451797063a9SSatish Balay      self.addMakeMacro('PETSC_KSP_LIB_BASIC','-lpetsc')
452797063a9SSatish Balay      self.addMakeMacro('PETSC_TS_LIB_BASIC','-lpetsc')
453b0a7d7e7SSatish Balay      self.addMakeMacro('PETSC_TAO_LIB_BASIC','-lpetsc')
454de8f682fSSatish Balay      self.addMakeMacro('PETSC_WITH_EXTERNAL_LIB',self.PETSC_WITH_EXTERNAL_LIB)
455bb84e0fdSBarry Smith      self.addDefine('USE_SINGLE_LIBRARY', '1')
4562df986feSBarry Smith      if self.sharedlibraries.useShared:
457ea820d49SSatish Balay        self.addMakeMacro('PETSC_SYS_LIB','${C_SH_LIB_PATH} ${PETSC_WITH_EXTERNAL_LIB}')
458ea820d49SSatish Balay        self.addMakeMacro('PETSC_VEC_LIB','${C_SH_LIB_PATH} ${PETSC_WITH_EXTERNAL_LIB}')
459ea820d49SSatish Balay        self.addMakeMacro('PETSC_MAT_LIB','${C_SH_LIB_PATH} ${PETSC_WITH_EXTERNAL_LIB}')
460ea820d49SSatish Balay        self.addMakeMacro('PETSC_DM_LIB','${C_SH_LIB_PATH} ${PETSC_WITH_EXTERNAL_LIB}')
461ea820d49SSatish Balay        self.addMakeMacro('PETSC_KSP_LIB','${C_SH_LIB_PATH} ${PETSC_WITH_EXTERNAL_LIB}')
462ea820d49SSatish Balay        self.addMakeMacro('PETSC_SNES_LIB','${C_SH_LIB_PATH} ${PETSC_WITH_EXTERNAL_LIB}')
463ea820d49SSatish Balay        self.addMakeMacro('PETSC_TS_LIB','${C_SH_LIB_PATH} ${PETSC_WITH_EXTERNAL_LIB}')
464b0a7d7e7SSatish Balay        self.addMakeMacro('PETSC_TAO_LIB','${C_SH_LIB_PATH} ${PETSC_WITH_EXTERNAL_LIB}')
465fdb87e33SJed Brown        self.addMakeMacro('PETSC_CHARACTERISTIC_LIB','${C_SH_LIB_PATH} ${PETSC_WITH_EXTERNAL_LIB}')
466ea820d49SSatish Balay        self.addMakeMacro('PETSC_LIB','${C_SH_LIB_PATH} ${PETSC_WITH_EXTERNAL_LIB}')
467ea820d49SSatish Balay        self.addMakeMacro('PETSC_CONTRIB','${C_SH_LIB_PATH} ${PETSC_WITH_EXTERNAL_LIB}')
4682df986feSBarry Smith      else:
469ea820d49SSatish Balay        self.addMakeMacro('PETSC_SYS_LIB','${PETSC_WITH_EXTERNAL_LIB}')
470ea820d49SSatish Balay        self.addMakeMacro('PETSC_VEC_LIB','${PETSC_WITH_EXTERNAL_LIB}')
471ea820d49SSatish Balay        self.addMakeMacro('PETSC_MAT_LIB','${PETSC_WITH_EXTERNAL_LIB}')
472ea820d49SSatish Balay        self.addMakeMacro('PETSC_DM_LIB','${PETSC_WITH_EXTERNAL_LIB}')
473ea820d49SSatish Balay        self.addMakeMacro('PETSC_KSP_LIB','${PETSC_WITH_EXTERNAL_LIB}')
474ea820d49SSatish Balay        self.addMakeMacro('PETSC_SNES_LIB','${PETSC_WITH_EXTERNAL_LIB}')
475ea820d49SSatish Balay        self.addMakeMacro('PETSC_TS_LIB','${PETSC_WITH_EXTERNAL_LIB}')
476b0a7d7e7SSatish Balay        self.addMakeMacro('PETSC_TAO_LIB','${PETSC_WITH_EXTERNAL_LIB}')
477fdb87e33SJed Brown        self.addMakeMacro('PETSC_CHARACTERISTIC_LIB','${PETSC_WITH_EXTERNAL_LIB}')
478ea820d49SSatish Balay        self.addMakeMacro('PETSC_LIB','${PETSC_WITH_EXTERNAL_LIB}')
479ea820d49SSatish Balay        self.addMakeMacro('PETSC_CONTRIB','${PETSC_WITH_EXTERNAL_LIB}')
4800f3b21c2SBarry Smith
481f8833479SBarry Smith    if not os.path.exists(os.path.join(self.petscdir.dir,self.arch.arch,'lib')):
482f8833479SBarry Smith      os.makedirs(os.path.join(self.petscdir.dir,self.arch.arch,'lib'))
483f8833479SBarry Smith
48413f9d092SSatish Balay# add a makefile endtry for display
48513f9d092SSatish Balay    if self.framework.argDB['with-display']:
48613f9d092SSatish Balay      self.addMakeMacro('DISPLAY',self.framework.argDB['with-display'])
48713f9d092SSatish Balay
488f8833479SBarry Smith    # add a makefile entry for configure options
489f8833479SBarry Smith    self.addMakeMacro('CONFIGURE_OPTIONS', self.framework.getOptionsString(['configModules', 'optionsModule']).replace('\"','\\"'))
490f8833479SBarry Smith    return
491f8833479SBarry Smith
492f8833479SBarry Smith  def dumpConfigInfo(self):
493f8833479SBarry Smith    import time
494c6ef1b5bSJed Brown    fd = open(os.path.join(self.arch.arch,'include','petscconfiginfo.h'),'w')
495dc25a686SPierre Jolivet    fd.write('static const char *petscconfigureoptions = "'+self.framework.getOptionsString(['configModules', 'optionsModule']).replace('\"','\\"').replace('\\ ','\\\\ ')+'";\n')
496f8833479SBarry Smith    fd.close()
497f8833479SBarry Smith    return
498f8833479SBarry Smith
4992a4161d9SMatthew G Knepley  def dumpMachineInfo(self):
5002a4161d9SMatthew G Knepley    import platform
501a970bd74SBernhard M. Wiedemann    import datetime
5022a4161d9SMatthew G Knepley    import time
50340373944SSatish Balay    import script
504ca77dbeeSGeoffrey Irving    def escape(s):
505e08ecd42SSatish Balay      return s.replace('"',r'\"').replace(r'\ ',r'\\ ') # novermin
506c6ef1b5bSJed Brown    fd = open(os.path.join(self.arch.arch,'include','petscmachineinfo.h'),'w')
5072a4161d9SMatthew G Knepley    fd.write('static const char *petscmachineinfo = \"\\n\"\n')
5082a4161d9SMatthew G Knepley    fd.write('\"-----------------------------------------\\n\"\n')
509a970bd74SBernhard M. Wiedemann    buildhost = platform.node()
510a970bd74SBernhard M. Wiedemann    if os.environ.get('SOURCE_DATE_EPOCH'):
511a970bd74SBernhard M. Wiedemann      buildhost = "reproducible"
512a970bd74SBernhard M. Wiedemann    buildtime = datetime.datetime.utcfromtimestamp(int(os.environ.get('SOURCE_DATE_EPOCH', time.time())))
513a970bd74SBernhard M. Wiedemann    fd.write('\"Libraries compiled on %s on %s \\n\"\n' % (buildtime, buildhost))
51460acdfe7SSatish Balay    fd.write('\"Machine characteristics: %s\\n\"\n' % (platform.platform()))
5155188cb68SSatish Balay    fd.write('\"Using PETSc directory: %s\\n\"\n' % (escape(self.installdir.petscDir)))
5165188cb68SSatish Balay    fd.write('\"Using PETSc arch: %s\\n\"\n' % (escape(self.installdir.petscArch)))
517cdec380aSBarry Smith    fd.write('\"-----------------------------------------\\n\";\n')
5182a4161d9SMatthew G Knepley    fd.write('static const char *petsccompilerinfo = \"\\n\"\n')
5192a4161d9SMatthew G Knepley    self.setCompilers.pushLanguage(self.languages.clanguage)
5205f27b2e0SBarry Smith    fd.write('\"Using C compiler: %s %s \\n\"\n' % (escape(self.setCompilers.getCompiler()), escape(self.setCompilers.getCompilerFlags())))
5212a4161d9SMatthew G Knepley    self.setCompilers.popLanguage()
5228782282cSMatthew G Knepley    if hasattr(self.compilers, 'FC'):
5232a4161d9SMatthew G Knepley      self.setCompilers.pushLanguage('FC')
5245f27b2e0SBarry Smith      fd.write('\"Using Fortran compiler: %s %s  %s\\n\"\n' % (escape(self.setCompilers.getCompiler()), escape(self.setCompilers.getCompilerFlags()), escape(self.setCompilers.CPPFLAGS)))
5252a4161d9SMatthew G Knepley      self.setCompilers.popLanguage()
526cdec380aSBarry Smith    fd.write('\"-----------------------------------------\\n\";\n')
5272a4161d9SMatthew G Knepley    fd.write('static const char *petsccompilerflagsinfo = \"\\n\"\n')
5285a21677cSJed Brown    fd.write('\"Using include paths: %s\\n\"\n' % (escape(self.PETSC_CC_INCLUDES_INSTALL.replace('${PETSC_DIR}', self.installdir.petscDir))))
529cdec380aSBarry Smith    fd.write('\"-----------------------------------------\\n\";\n')
5302a4161d9SMatthew G Knepley    fd.write('static const char *petsclinkerinfo = \"\\n\"\n')
5312a4161d9SMatthew G Knepley    self.setCompilers.pushLanguage(self.languages.clanguage)
532ca77dbeeSGeoffrey Irving    fd.write('\"Using C linker: %s\\n\"\n' % (escape(self.setCompilers.getLinker())))
5332a4161d9SMatthew G Knepley    self.setCompilers.popLanguage()
5348782282cSMatthew G Knepley    if hasattr(self.compilers, 'FC'):
5352a4161d9SMatthew G Knepley      self.setCompilers.pushLanguage('FC')
536ca77dbeeSGeoffrey Irving      fd.write('\"Using Fortran linker: %s\\n\"\n' % (escape(self.setCompilers.getLinker())))
5372a4161d9SMatthew G Knepley      self.setCompilers.popLanguage()
5385188cb68SSatish 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)))
539cdec380aSBarry Smith    fd.write('\"-----------------------------------------\\n\";\n')
5402a4161d9SMatthew G Knepley    fd.close()
5412a4161d9SMatthew G Knepley    return
542b2843cf1SBarry Smith
543b2843cf1SBarry Smith  def configurePrefetch(self):
544b2843cf1SBarry Smith    '''Sees if there are any prefetch functions supported'''
5457fca349cSMatthew G. Knepley    if config.setCompilers.Configure.isSolaris(self.log) or self.framework.argDB['with-ios'] or not self.framework.argDB['with-prefetch']:
54693f78423SSatish Balay      self.addDefine('Prefetch(a,b,c)', ' ')
54793f78423SSatish Balay      return
548ec284106SBarry Smith    self.pushLanguage(self.languages.clanguage)
54910699583SJed Brown    if self.checkLink('#include <xmmintrin.h>', 'void *v = 0;_mm_prefetch((const char*)v,_MM_HINT_NTA);\n'):
55050d8bf02SJed Brown      # The Intel Intrinsics manual [1] specifies the prototype
55150d8bf02SJed Brown      #
55250d8bf02SJed Brown      #   void _mm_prefetch(char const *a, int sel);
55350d8bf02SJed Brown      #
55450d8bf02SJed Brown      # but other vendors seem to insist on using subtly different
55550d8bf02SJed Brown      # prototypes, including void* for the pointer, and an enum for
55650d8bf02SJed Brown      # sel.  These are both reasonable changes, but negatively impact
55750d8bf02SJed Brown      # portability.
55850d8bf02SJed Brown      #
559a8d69d7bSBarry Smith      # [1] https://software.intel.com/file/6373
56050d8bf02SJed Brown      self.addDefine('HAVE_XMMINTRIN_H', 1)
56150d8bf02SJed Brown      self.addDefine('Prefetch(a,b,c)', '_mm_prefetch((const char*)(a),(c))')
56250d8bf02SJed Brown      self.addDefine('PREFETCH_HINT_NTA', '_MM_HINT_NTA')
56350d8bf02SJed Brown      self.addDefine('PREFETCH_HINT_T0',  '_MM_HINT_T0')
56450d8bf02SJed Brown      self.addDefine('PREFETCH_HINT_T1',  '_MM_HINT_T1')
56550d8bf02SJed Brown      self.addDefine('PREFETCH_HINT_T2',  '_MM_HINT_T2')
56650d8bf02SJed Brown    elif self.checkLink('#include <xmmintrin.h>', 'void *v = 0;_mm_prefetch(v,_MM_HINT_NTA);\n'):
56750d8bf02SJed Brown      self.addDefine('HAVE_XMMINTRIN_H', 1)
56850d8bf02SJed Brown      self.addDefine('Prefetch(a,b,c)', '_mm_prefetch((const void*)(a),(c))')
56950d8bf02SJed Brown      self.addDefine('PREFETCH_HINT_NTA', '_MM_HINT_NTA')
57050d8bf02SJed Brown      self.addDefine('PREFETCH_HINT_T0',  '_MM_HINT_T0')
57150d8bf02SJed Brown      self.addDefine('PREFETCH_HINT_T1',  '_MM_HINT_T1')
57250d8bf02SJed Brown      self.addDefine('PREFETCH_HINT_T2',  '_MM_HINT_T2')
57310699583SJed Brown    elif self.checkLink('', 'void *v = 0;__builtin_prefetch(v,0,0);\n'):
57410699583SJed Brown      # From GCC docs: void __builtin_prefetch(const void *addr,int rw,int locality)
57510699583SJed Brown      #
57610699583SJed Brown      #   The value of rw is a compile-time constant one or zero; one
57710699583SJed Brown      #   means that the prefetch is preparing for a write to the memory
57810699583SJed Brown      #   address and zero, the default, means that the prefetch is
57910699583SJed Brown      #   preparing for a read. The value locality must be a compile-time
58010699583SJed Brown      #   constant integer between zero and three. A value of zero means
58110699583SJed Brown      #   that the data has no temporal locality, so it need not be left
58210699583SJed Brown      #   in the cache after the access. A value of three means that the
58310699583SJed Brown      #   data has a high degree of temporal locality and should be left
58410699583SJed Brown      #   in all levels of cache possible. Values of one and two mean,
58510699583SJed Brown      #   respectively, a low or moderate degree of temporal locality.
58610699583SJed Brown      #
58710699583SJed Brown      # Here we adopt Intel's x86/x86-64 naming scheme for the locality
58810699583SJed Brown      # hints.  Using macros for these values in necessary since some
58910699583SJed Brown      # compilers require an enum.
59010699583SJed Brown      self.addDefine('Prefetch(a,b,c)', '__builtin_prefetch((a),(b),(c))')
59110699583SJed Brown      self.addDefine('PREFETCH_HINT_NTA', '0')
59210699583SJed Brown      self.addDefine('PREFETCH_HINT_T0',  '3')
59310699583SJed Brown      self.addDefine('PREFETCH_HINT_T1',  '2')
59410699583SJed Brown      self.addDefine('PREFETCH_HINT_T2',  '1')
595b2843cf1SBarry Smith    else:
596b2843cf1SBarry Smith      self.addDefine('Prefetch(a,b,c)', ' ')
5977d490b44SBarry Smith    self.popLanguage()
598b2843cf1SBarry Smith
59949fe22e6SSatish Balay  def delGenFiles(self):
60049fe22e6SSatish Balay    '''Delete generated files'''
60149fe22e6SSatish Balay    delfile = os.path.join(self.arch.arch,'lib','petsc','conf','files')
60249fe22e6SSatish Balay    try:
60349fe22e6SSatish Balay      os.unlink(delfile)
60449fe22e6SSatish Balay    except: pass
60549fe22e6SSatish Balay
60609bc878fSSatish Balay  def configureAtoll(self):
60709bc878fSSatish Balay    '''Checks if atoll exists'''
608436b02dcSSatish 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")'):
60909bc878fSSatish Balay       self.addDefine('HAVE_ATOLL', '1')
61009bc878fSSatish Balay
6112400fdedSBarry Smith  def configureUnused(self):
6122400fdedSBarry Smith    '''Sees if __attribute((unused)) is supported'''
6131adaff47SSean Farley    if self.framework.argDB['with-ios']:
6142400fdedSBarry Smith      self.addDefine('UNUSED', ' ')
6152400fdedSBarry Smith      return
6162400fdedSBarry Smith    self.pushLanguage(self.languages.clanguage)
617edf21b64SSatish 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'):
6182400fdedSBarry Smith      self.addDefine('UNUSED', '__attribute((unused))')
6192400fdedSBarry Smith    else:
6202400fdedSBarry Smith      self.addDefine('UNUSED', ' ')
6212400fdedSBarry Smith    self.popLanguage()
6222400fdedSBarry Smith
62398ed35c3SBarry Smith  def configureIsatty(self):
62498ed35c3SBarry Smith    '''Check if the Unix C function isatty() works correctly
62598ed35c3SBarry Smith       Actually just assumes it does not work correctly on batch systems'''
62698ed35c3SBarry Smith    if not self.framework.argDB['with-batch']:
62798ed35c3SBarry Smith      self.addDefine('USE_ISATTY',1)
62898ed35c3SBarry Smith
6291ef8df7fSJed Brown  def configureDeprecated(self):
6301ef8df7fSJed Brown    '''Check if __attribute((deprecated)) is supported'''
6317efe37a1SJacob Faibussowitsch    def checkDeprecated(macro_base, src, is_intel):
6327efe37a1SJacob Faibussowitsch      '''
6337efe37a1SJacob Faibussowitsch      run through the various attribute deprecated combinations and define MACRO_BAS(why) to the result
6347efe37a1SJacob Faibussowitsch      it if it compiles.
6357efe37a1SJacob Faibussowitsch
6367efe37a1SJacob Faibussowitsch      If none of the combos work, defines MACRO_BASE(why) as empty
6377efe37a1SJacob Faibussowitsch      '''
6387efe37a1SJacob Faibussowitsch      full_macro_name = macro_base + '(why)'
6397efe37a1SJacob Faibussowitsch      for prefix in ('__attribute__', '__attribute','__declspec'):
6407efe37a1SJacob Faibussowitsch        if prefix == '__declspec':
6417efe37a1SJacob Faibussowitsch          # declspec does not have an extra set of brackets around the arguments
6427efe37a1SJacob Faibussowitsch          attr_bodies = ('deprecated(why)', 'deprecated')
6431ef8df7fSJed Brown        else:
6447efe37a1SJacob Faibussowitsch          attr_bodies = ('(deprecated(why))', '(deprecated)')
6457efe37a1SJacob Faibussowitsch
6467efe37a1SJacob Faibussowitsch        for attr_body in attr_bodies:
6477efe37a1SJacob Faibussowitsch          attr_def = '{}({})'.format(prefix, attr_body)
6487efe37a1SJacob Faibussowitsch          test_src = '\n'.join((
6497efe37a1SJacob Faibussowitsch            '#define {} {}'.format(full_macro_name, attr_def),
6507efe37a1SJacob Faibussowitsch            src.format(macro_base + '("asdasdadsasd")')
6517efe37a1SJacob Faibussowitsch          ))
6527efe37a1SJacob Faibussowitsch          if self.checkCompile(test_src, ''):
6537efe37a1SJacob Faibussowitsch            self.logPrint('configureDeprecated: \'{}\' appears to work'.format(attr_def))
6547efe37a1SJacob Faibussowitsch            if is_intel and '(why)' in attr_body:
6557efe37a1SJacob Faibussowitsch              self.logPrint('configureDeprecated: Intel has conspired to make a supremely environment-sensitive compiler. The Intel compiler looks at the gcc executable in the environment to determine the language compatibility that it should attempt to emulate. Some important Cray installations have built PETSc using the Intel compiler, but with a newer gcc module loaded (e.g. 4.7). Thus at PETSc configure time, the Intel compiler decides to support the string argument, but the gcc found in the default user environment is older and does not support the argument.\n'.format(attr_def))
6567efe37a1SJacob Faibussowitsch              self.logPrint('*** WE WILL THEREFORE REJECT \'{}\' AND CONTINUE TESTING ***'.format(attr_def))
6577efe37a1SJacob Faibussowitsch              continue
6587efe37a1SJacob Faibussowitsch            self.addDefine(full_macro_name, attr_def)
6597efe37a1SJacob Faibussowitsch            return
6607efe37a1SJacob Faibussowitsch
6617efe37a1SJacob Faibussowitsch      self.addDefine(full_macro_name, ' ')
6627efe37a1SJacob Faibussowitsch      return
6637efe37a1SJacob Faibussowitsch
6647efe37a1SJacob Faibussowitsch    lang = self.languages.clanguage
6657efe37a1SJacob Faibussowitsch    with self.Language(lang):
6667efe37a1SJacob Faibussowitsch      is_intel = self.setCompilers.isIntel(self.getCompiler(lang=lang), self.log)
6677efe37a1SJacob Faibussowitsch      checkDeprecated('DEPRECATED_FUNCTION', '{} int myfunc(void) {{ return 1; }}', is_intel)
6687efe37a1SJacob Faibussowitsch      checkDeprecated('DEPRECATED_TYPEDEF', 'typedef int my_int {};', is_intel)
6697efe37a1SJacob Faibussowitsch      checkDeprecated('DEPRECATED_ENUM', 'enum E {{ oldval {}, newval }};', is_intel)
6707efe37a1SJacob Faibussowitsch      # I was unable to make a CPP macro that takes the old and new values as separate
6717efe37a1SJacob Faibussowitsch      # arguments and builds the message needed by _Pragma hence the deprecation message is
6727efe37a1SJacob Faibussowitsch      # handled as it is
67305de396fSBarry Smith      if self.checkCompile('#define TEST _Pragma("GCC warning \"Testing _Pragma\"") value'):
67405de396fSBarry Smith        self.addDefine('DEPRECATED_MACRO(why)', '_Pragma(why)')
67505de396fSBarry Smith      else:
67605de396fSBarry Smith        self.addDefine('DEPRECATED_MACRO(why)', ' ')
6771ef8df7fSJed Brown
67818f41590SBarry Smith  def configureAlign(self):
6797b7fc14bSLisandro Dalcin    '''Check if __attribute(aligned) is supported'''
6807b7fc14bSLisandro Dalcin    code = '''\
681752d89a4SSatish Balaystruct mystruct {int myint;} __attribute((aligned(16)));
6827b7fc14bSLisandro Dalcinchar assert_aligned[(sizeof(struct mystruct)==16)*2-1];
683752d89a4SSatish Balay'''
684752d89a4SSatish Balay    self.pushLanguage(self.languages.clanguage)
6857b7fc14bSLisandro Dalcin    if self.checkCompile(code):
686752d89a4SSatish Balay      self.addDefine('ATTRIBUTEALIGNED(size)', '__attribute((aligned(size)))')
687752d89a4SSatish Balay      self.addDefine('HAVE_ATTRIBUTEALIGNED', 1)
688752d89a4SSatish Balay    else:
6897b7fc14bSLisandro Dalcin      self.framework.logPrint('Incorrect attribute(aligned)')
690752d89a4SSatish Balay      self.addDefine('ATTRIBUTEALIGNED(size)', ' ')
6917b7fc14bSLisandro Dalcin    self.popLanguage()
692752d89a4SSatish Balay    return
69318f41590SBarry Smith
6949800092aSJed Brown  def configureExpect(self):
6959800092aSJed Brown    '''Sees if the __builtin_expect directive is supported'''
6969800092aSJed Brown    self.pushLanguage(self.languages.clanguage)
6979800092aSJed Brown    if self.checkLink('', 'if (__builtin_expect(0,1)) return 1;'):
6989800092aSJed Brown      self.addDefine('HAVE_BUILTIN_EXPECT', 1)
6999800092aSJed Brown    self.popLanguage()
7009800092aSJed Brown
70153c77d0aSJed Brown  def configureFunctionName(self):
702fbfcfee5SBarry Smith    '''Sees if the compiler supports __func__ or a variant.'''
7031ec50b02SJed Brown    def getFunctionName(lang):
704fbfcfee5SBarry Smith      name = '"unknown"'
7051ec50b02SJed Brown      self.pushLanguage(lang)
706b6ff4c76SKarl Rupp      for fname in ['__func__','__FUNCTION__','__extension__ __func__']:
7070117e5a1SSatish Balay        code = "if ("+fname+"[0] != 'm') return 1;"
7080117e5a1SSatish Balay        if self.checkCompile('',code) and self.checkLink('',code):
7090117e5a1SSatish Balay          name = fname
7100117e5a1SSatish Balay          break
7111ec50b02SJed Brown      self.popLanguage()
7121ec50b02SJed Brown      return name
7131ec50b02SJed Brown    langs = []
714628773c9SSatish Balay
715628773c9SSatish Balay    self.addDefine('FUNCTION_NAME_C', getFunctionName('C'))
7165f6e5f85SSatish Balay    if hasattr(self.compilers, 'CXX'):
717628773c9SSatish Balay      self.addDefine('FUNCTION_NAME_CXX', getFunctionName('Cxx'))
71853c77d0aSJed Brown
719753ebd1dSJed Brown  def configureIntptrt(self):
720753ebd1dSJed Brown    '''Determine what to use for uintptr_t'''
721753ebd1dSJed Brown    def staticAssertSizeMatchesVoidStar(inc,typename):
722753ebd1dSJed Brown      # The declaration is an error if either array size is negative.
723753ebd1dSJed 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
724d26187a0SJed Brown      return self.checkCompile(inc, ('#define STATIC_ASSERT(cond) char negative_length_if_false[2*(!!(cond))-1]\n'
725979939cdSSatish Balay                                     + 'STATIC_ASSERT(sizeof(void*) == sizeof(%s));'%typename))
726753ebd1dSJed Brown    self.pushLanguage(self.languages.clanguage)
727753ebd1dSJed Brown    if self.checkCompile('#include <stdint.h>', 'int x; uintptr_t i = (uintptr_t)&x;'):
728753ebd1dSJed Brown      self.addDefine('UINTPTR_T', 'uintptr_t')
729753ebd1dSJed Brown    elif staticAssertSizeMatchesVoidStar('','unsigned long long'):
730753ebd1dSJed Brown      self.addDefine('UINTPTR_T', 'unsigned long long')
731753ebd1dSJed Brown    elif staticAssertSizeMatchesVoidStar('#include <stdlib.h>','size_t') or staticAssertSizeMatchesVoidStar('#include <string.h>', 'size_t'):
732753ebd1dSJed Brown      self.addDefine('UINTPTR_T', 'size_t')
733c82284b1SJed Brown    elif staticAssertSizeMatchesVoidStar('','unsigned long'):
734c82284b1SJed Brown      self.addDefine('UINTPTR_T', 'unsigned long')
7352d1b7972SSatish Balay    elif staticAssertSizeMatchesVoidStar('','unsigned'):
736753ebd1dSJed Brown      self.addDefine('UINTPTR_T', 'unsigned')
737d26187a0SJed Brown    else:
738d26187a0SJed Brown      raise RuntimeError('Could not find any unsigned integer type matching void*')
739753ebd1dSJed Brown    self.popLanguage()
740753ebd1dSJed Brown
741ed938b00SJed Brown  def configureRTLDDefault(self):
7427b65ca21SBarry Smith    '''Check for dynamic library feature'''
743bfef2c86SBarry Smith    if self.checkCompile('#include <dlfcn.h>\n void *ptr =  RTLD_DEFAULT;'):
744bfef2c86SBarry Smith      self.addDefine('RTLD_DEFAULT','1')
745f8833479SBarry Smith    return
746f8833479SBarry Smith
747f8833479SBarry Smith  def configureSolaris(self):
748f8833479SBarry Smith    '''Solaris specific stuff'''
749f8833479SBarry Smith    if os.path.isdir(os.path.join('/usr','ucblib')):
750f8833479SBarry Smith      try:
751f8833479SBarry Smith        flag = getattr(self.setCompilers, self.language[-1]+'SharedLinkerFlag')
752f8833479SBarry Smith      except AttributeError:
753f8833479SBarry Smith        flag = None
754f8833479SBarry Smith      if flag is None:
755f8833479SBarry Smith        self.compilers.LIBS += ' -L/usr/ucblib'
756f8833479SBarry Smith      else:
757f8833479SBarry Smith        self.compilers.LIBS += ' '+flag+'/usr/ucblib'
758f8833479SBarry Smith    return
759f8833479SBarry Smith
7600f64ec89SBarry Smith  def configureDarwin(self):
7610f64ec89SBarry Smith    '''Log brew configuration for Apple systems'''
7620f64ec89SBarry Smith    try:
7630f64ec89SBarry Smith      self.executeShellCommand(['brew', 'config'], log = self.log)
7640f64ec89SBarry Smith      self.executeShellCommand(['brew', 'info', 'gcc'], log = self.log)
7650f64ec89SBarry Smith    except:
7660f64ec89SBarry Smith      pass
7670f64ec89SBarry Smith    return
7680f64ec89SBarry Smith
769f8833479SBarry Smith  def configureLinux(self):
770f8833479SBarry Smith    '''Linux specific stuff'''
7719f15855cSMatthew G Knepley    # TODO: Test for this by mallocing an odd number of floats and checking the address
772f8833479SBarry Smith    self.addDefine('HAVE_DOUBLE_ALIGN_MALLOC', 1)
773f8833479SBarry Smith    return
774f8833479SBarry Smith
775f8833479SBarry Smith  def configureWin32(self):
776f8833479SBarry Smith    '''Win32 non-cygwin specific stuff'''
777f8833479SBarry Smith    kernel32=0
7784e8afd12SMosè Giordano    if self.libraries.add('Kernel32.lib','GetComputerName',prototype='#include <windows.h>', call='GetComputerName(NULL,NULL);'):
779f8833479SBarry Smith      self.addDefine('HAVE_WINDOWS_H',1)
780f8833479SBarry Smith      self.addDefine('HAVE_GETCOMPUTERNAME',1)
781f8833479SBarry Smith      kernel32=1
7824e8afd12SMosè Giordano    elif self.libraries.add('kernel32','GetComputerName',prototype='#include <windows.h>', call='GetComputerName(NULL,NULL);'):
783f8833479SBarry Smith      self.addDefine('HAVE_WINDOWS_H',1)
784f8833479SBarry Smith      self.addDefine('HAVE_GETCOMPUTERNAME',1)
785f8833479SBarry Smith      kernel32=1
786f8833479SBarry Smith    if kernel32:
787eed94e11SSatish Balay      if self.framework.argDB['with-windows-graphics']:
788eed94e11SSatish Balay        self.addDefine('USE_WINDOWS_GRAPHICS',1)
7894e8afd12SMosè Giordano      if self.checkLink('#include <windows.h>','LoadLibrary(0)'):
790f8833479SBarry Smith        self.addDefine('HAVE_LOADLIBRARY',1)
7914e8afd12SMosè Giordano      if self.checkLink('#include <windows.h>','GetProcAddress(0,0)'):
792b50f6d9eSLisandro Dalcin        self.addDefine('HAVE_GETPROCADDRESS',1)
7934e8afd12SMosè Giordano      if self.checkLink('#include <windows.h>','FreeLibrary(0)'):
794b50f6d9eSLisandro Dalcin        self.addDefine('HAVE_FREELIBRARY',1)
7954e8afd12SMosè Giordano      if self.checkLink('#include <windows.h>','GetLastError()'):
796a21658a3SLisandro Dalcin        self.addDefine('HAVE_GETLASTERROR',1)
7974e8afd12SMosè Giordano      if self.checkLink('#include <windows.h>','SetLastError(0)'):
798a21658a3SLisandro Dalcin        self.addDefine('HAVE_SETLASTERROR',1)
7994e8afd12SMosè Giordano      if self.checkLink('#include <windows.h>\n','QueryPerformanceCounter(0);\n'):
800bea725cfSBarry Smith        self.addDefine('USE_MICROSOFT_TIME',1)
8014e8afd12SMosè Giordano    if self.libraries.add('Advapi32.lib','GetUserName',prototype='#include <windows.h>', call='GetUserName(NULL,NULL);'):
802f8833479SBarry Smith      self.addDefine('HAVE_GET_USER_NAME',1)
8034e8afd12SMosè Giordano    elif self.libraries.add('advapi32','GetUserName',prototype='#include <windows.h>', call='GetUserName(NULL,NULL);'):
804f8833479SBarry Smith      self.addDefine('HAVE_GET_USER_NAME',1)
805f8833479SBarry Smith
8064e8afd12SMosè Giordano    if not self.libraries.add('User32.lib','GetDC',prototype='#include <windows.h>',call='GetDC(0);'):
8074e8afd12SMosè Giordano      self.libraries.add('user32','GetDC',prototype='#include <windows.h>',call='GetDC(0);')
8084e8afd12SMosè Giordano    if not self.libraries.add('Gdi32.lib','CreateCompatibleDC',prototype='#include <windows.h>',call='CreateCompatibleDC(0);'):
8094e8afd12SMosè Giordano      self.libraries.add('gdi32','CreateCompatibleDC',prototype='#include <windows.h>',call='CreateCompatibleDC(0);')
810f8833479SBarry Smith
811f8833479SBarry Smith    self.types.check('int32_t', 'int')
812f8833479SBarry Smith    if not self.checkCompile('#include <sys/types.h>\n','uid_t u;\n'):
813f8833479SBarry Smith      self.addTypedef('int', 'uid_t')
814f8833479SBarry Smith      self.addTypedef('int', 'gid_t')
815f8833479SBarry Smith    if not self.checkLink('#if defined(PETSC_HAVE_UNISTD_H)\n#include <unistd.h>\n#endif\n','int a=R_OK;\n'):
816f8833479SBarry Smith      self.framework.addDefine('R_OK', '04')
817f8833479SBarry Smith      self.framework.addDefine('W_OK', '02')
818f8833479SBarry Smith      self.framework.addDefine('X_OK', '01')
819f8833479SBarry Smith    if not self.checkLink('#include <sys/stat.h>\n','int a=0;\nif (S_ISDIR(a)){}\n'):
820f8833479SBarry Smith      self.framework.addDefine('S_ISREG(a)', '(((a)&_S_IFMT) == _S_IFREG)')
821f8833479SBarry Smith      self.framework.addDefine('S_ISDIR(a)', '(((a)&_S_IFMT) == _S_IFDIR)')
8224e8afd12SMosè Giordano    if self.checkCompile('#include <windows.h>\n','LARGE_INTEGER a;\nDWORD b=a.u.HighPart;\n'):
823f8833479SBarry Smith      self.addDefine('HAVE_LARGE_INTEGER_U',1)
824f8833479SBarry Smith
825f8833479SBarry Smith    # Windows requires a Binary file creation flag when creating/opening binary files.  Is a better test in order?
8264e8afd12SMosè Giordano    if self.checkCompile('#include <windows.h>\n#include <fcntl.h>\n', 'int flags = O_BINARY;'):
827f8833479SBarry Smith      self.addDefine('HAVE_O_BINARY',1)
828f8833479SBarry Smith
829f8833479SBarry Smith    if self.compilers.CC.find('win32fe') >= 0:
830ad4212abSSatish Balay      self.addDefine('HAVE_WINDOWS_COMPILERS',1)
831f8833479SBarry Smith      self.addDefine('DIR_SEPARATOR','\'\\\\\'')
832f8833479SBarry Smith      self.addDefine('REPLACE_DIR_SEPARATOR','\'/\'')
833f8833479SBarry Smith      self.addDefine('CANNOT_START_DEBUGGER',1)
8345188cb68SSatish Balay      (petscdir,error,status) = self.executeShellCommand('cygpath -w '+self.installdir.petscDir, log = self.log)
83534531a4dSSatish Balay      self.addDefine('DIR','"'+petscdir.replace('\\','\\\\')+'"')
8365188cb68SSatish Balay      (petscdir,error,status) = self.executeShellCommand('cygpath -m '+self.installdir.petscDir, log = self.log)
837e433681fSSatish Balay      self.addMakeMacro('wPETSC_DIR',petscdir)
8384e00a515SSatish Balay      if self.dataFilesPath.datafilespath:
8394e00a515SSatish Balay        (datafilespath,error,status) = self.executeShellCommand('cygpath -m '+self.dataFilesPath.datafilespath, log = self.log)
8404e00a515SSatish Balay        self.addMakeMacro('DATAFILESPATH',datafilespath)
8414e00a515SSatish Balay
842f8833479SBarry Smith    else:
843f8833479SBarry Smith      self.addDefine('REPLACE_DIR_SEPARATOR','\'\\\\\'')
844f8833479SBarry Smith      self.addDefine('DIR_SEPARATOR','\'/\'')
8455188cb68SSatish Balay      self.addDefine('DIR','"'+self.installdir.petscDir+'"')
8465188cb68SSatish Balay      self.addMakeMacro('wPETSC_DIR',self.installdir.petscDir)
8474e00a515SSatish Balay      if self.dataFilesPath.datafilespath:
8484e00a515SSatish Balay        self.addMakeMacro('DATAFILESPATH',self.dataFilesPath.datafilespath)
8495188cb68SSatish Balay    self.addDefine('ARCH','"'+self.installdir.petscArch+'"')
850f8833479SBarry Smith    return
851f8833479SBarry Smith
852*7ce81a4bSJacob Faibussowitsch  def configureCoverage(self, lang, extra_coverage_flags=None, extra_debug_flags=None):
853*7ce81a4bSJacob Faibussowitsch    """
854*7ce81a4bSJacob Faibussowitsch    Check that a compiler accepts code-coverage flags. If the compiler does accept code-coverage flags
855*7ce81a4bSJacob Faibussowitsch    try to set debugging flags equivalent to -Og.
856*7ce81a4bSJacob Faibussowitsch
857*7ce81a4bSJacob Faibussowitsch    Arguments:
858*7ce81a4bSJacob Faibussowitsch    - lang: the language to check the coverage flag for
859*7ce81a4bSJacob Faibussowitsch    - extra_coverage_flags: a list of extra flags to use when checking the coverage flags
860*7ce81a4bSJacob Faibussowitsch    - extra_debug_flags: a list of extra flags to try when setting debug flags
861*7ce81a4bSJacob Faibussowitsch
862*7ce81a4bSJacob Faibussowitsch    On success:
863*7ce81a4bSJacob Faibussowitsch    - defines PETSC_USE_COVERAGE to 1
864*7ce81a4bSJacob Faibussowitsch    """
865*7ce81a4bSJacob Faibussowitsch    def log_print(msg, *args, **kwargs):
866*7ce81a4bSJacob Faibussowitsch      self.logPrint('checkCoverage: '+str(msg), *args, **kwargs)
867*7ce81a4bSJacob Faibussowitsch      return
868*7ce81a4bSJacob Faibussowitsch
869*7ce81a4bSJacob Faibussowitsch    def quoted(string):
870*7ce81a4bSJacob Faibussowitsch      return string.join(("'", "'"))
871*7ce81a4bSJacob Faibussowitsch
872*7ce81a4bSJacob Faibussowitsch    def make_flag_list(default, extra):
873*7ce81a4bSJacob Faibussowitsch      ret = [default]
874*7ce81a4bSJacob Faibussowitsch      if extra is not None:
875*7ce81a4bSJacob Faibussowitsch        assert isinstance(extra, list)
876*7ce81a4bSJacob Faibussowitsch        ret.extend(extra)
877*7ce81a4bSJacob Faibussowitsch      return ret
878*7ce81a4bSJacob Faibussowitsch
879*7ce81a4bSJacob Faibussowitsch    log_print('Checking coverage flag for language {}'.format(lang))
880*7ce81a4bSJacob Faibussowitsch    if not self.argDB['with-coverage']:
881*7ce81a4bSJacob Faibussowitsch      log_print('coverage was disabled from command line or default')
882*7ce81a4bSJacob Faibussowitsch      return
883*7ce81a4bSJacob Faibussowitsch
884*7ce81a4bSJacob Faibussowitsch    compiler  = self.getCompiler(lang=lang)
885*7ce81a4bSJacob Faibussowitsch    is_gnuish = self.setCompilers.isGNU(compiler, self.log) or self.setCompilers.isClang(compiler, self.log)
886*7ce81a4bSJacob Faibussowitsch
887*7ce81a4bSJacob Faibussowitsch    # if not gnuish and we don't have a set of extra flags, bail
888*7ce81a4bSJacob Faibussowitsch    if not is_gnuish and extra_coverage_flags is None:
889*7ce81a4bSJacob Faibussowitsch      log_print('Don\'t know how to add coverage for compiler {}. Only know how to add coverage for gnu-like compilers (either gcc or clang). Skipping it!'.format(quoted(compiler)))
890*7ce81a4bSJacob Faibussowitsch      return
891*7ce81a4bSJacob Faibussowitsch
892*7ce81a4bSJacob Faibussowitsch    coverage_flags = make_flag_list('--coverage', extra_coverage_flags)
893*7ce81a4bSJacob Faibussowitsch    log_print('Checking set of coverage flags: {}'.format(coverage_flags))
894*7ce81a4bSJacob Faibussowitsch
895*7ce81a4bSJacob Faibussowitsch    found = 0
896*7ce81a4bSJacob Faibussowitsch    with self.Language(lang):
897*7ce81a4bSJacob Faibussowitsch      with self.setCompilers.Language(lang):
898*7ce81a4bSJacob Faibussowitsch        for flag in coverage_flags:
899*7ce81a4bSJacob Faibussowitsch          if self.setCompilers.checkCompilerFlag(flag) and self.checkLink():
900*7ce81a4bSJacob Faibussowitsch            # compilerOnly = False, the linker also needs to see the coverage flag
901*7ce81a4bSJacob Faibussowitsch            self.setCompilers.insertCompilerFlag(flag, False)
902*7ce81a4bSJacob Faibussowitsch            found = 1
903*7ce81a4bSJacob Faibussowitsch            break
904*7ce81a4bSJacob Faibussowitsch          log_print(
905*7ce81a4bSJacob Faibussowitsch            'Compiler {} did not accept coverage flag {}'.format(quoted(compiler), quoted(flag))
906*7ce81a4bSJacob Faibussowitsch          )
907*7ce81a4bSJacob Faibussowitsch
908*7ce81a4bSJacob Faibussowitsch    if not found:
909*7ce81a4bSJacob Faibussowitsch      log_print('Compiler {} did not accept ANY coverage flags: {}, bailing!'.format(quoted(compiler), coverage_flags))
910*7ce81a4bSJacob Faibussowitsch      return
911*7ce81a4bSJacob Faibussowitsch
912*7ce81a4bSJacob Faibussowitsch    if not self.functions.haveFunction('__gcov_dump'):
913*7ce81a4bSJacob Faibussowitsch      self.functions.checkClassify(['__gcov_dump'])
914*7ce81a4bSJacob Faibussowitsch
915*7ce81a4bSJacob Faibussowitsch    # now check if we can override the optimization level. It is only kosher to do so if
916*7ce81a4bSJacob Faibussowitsch    # the user did not explicitly set the optimization flags (via CFLAGS, CXXFLAGS,
917*7ce81a4bSJacob Faibussowitsch    # CXXOPTFLAGS, etc). If they have done so, we sternly warn them about their lapse in
918*7ce81a4bSJacob Faibussowitsch    # judgement
919*7ce81a4bSJacob Faibussowitsch    with self.Language(lang):
920*7ce81a4bSJacob Faibussowitsch      compiler_flags = self.getCompilerFlags()
921*7ce81a4bSJacob Faibussowitsch
922*7ce81a4bSJacob Faibussowitsch    user_set          = 0
923*7ce81a4bSJacob Faibussowitsch    allowed_opt_flags = re.compile(r'|'.join((r'-O[01g]', r'-g[1-9]*')))
924*7ce81a4bSJacob Faibussowitsch    for flagsname in [self.getCompilerFlagsName(lang), self.compilerFlags.getOptionalFlagsName(lang)]:
925*7ce81a4bSJacob Faibussowitsch      if flagsname in self.argDB:
926*7ce81a4bSJacob Faibussowitsch        opt_flags = [
927*7ce81a4bSJacob Faibussowitsch          f for f in self.compilerFlags.findOptFlags(compiler_flags) if not allowed_opt_flags.match(f)
928*7ce81a4bSJacob Faibussowitsch        ]
929*7ce81a4bSJacob Faibussowitsch        if opt_flags:
930*7ce81a4bSJacob Faibussowitsch          self.logPrintWarning('Coverage requested, but optimization flag(s) {} found in {}. Coverage collection will work, but may be less accurate. Suggest removing the flag and/or using -Og (or equivalent) instead'.format(', '.join(map(quoted, opt_flags)), quoted(flagsname)))
931*7ce81a4bSJacob Faibussowitsch          user_set = 1
932*7ce81a4bSJacob Faibussowitsch          break
933*7ce81a4bSJacob Faibussowitsch
934*7ce81a4bSJacob Faibussowitsch    if not user_set:
935*7ce81a4bSJacob Faibussowitsch      debug_flags = make_flag_list('-Og', extra_debug_flags)
936*7ce81a4bSJacob Faibussowitsch      with self.setCompilers.Language(lang):
937*7ce81a4bSJacob Faibussowitsch        for flag in debug_flags:
938*7ce81a4bSJacob Faibussowitsch          try:
939*7ce81a4bSJacob Faibussowitsch            self.setCompilers.addCompilerFlag(flag)
940*7ce81a4bSJacob Faibussowitsch          except RuntimeError:
941*7ce81a4bSJacob Faibussowitsch            continue
942*7ce81a4bSJacob Faibussowitsch          break
943*7ce81a4bSJacob Faibussowitsch
944*7ce81a4bSJacob Faibussowitsch    self.addDefine('USE_COVERAGE', 1)
945*7ce81a4bSJacob Faibussowitsch    return
946*7ce81a4bSJacob Faibussowitsch
947*7ce81a4bSJacob Faibussowitsch  def configureCoverageExecutable(self):
948*7ce81a4bSJacob Faibussowitsch    """
949*7ce81a4bSJacob Faibussowitsch    Check that a code-coverage collecting tool exists and is on PATH.
950*7ce81a4bSJacob Faibussowitsch
951*7ce81a4bSJacob Faibussowitsch    On success:
952*7ce81a4bSJacob Faibussowitsch    - Adds PETSC_COVERAGE_EXEC make macro containing the full path to the coverage tool executable.
953*7ce81a4bSJacob Faibussowitsch
954*7ce81a4bSJacob Faibussowitsch    Raises RuntimeError if:
955*7ce81a4bSJacob Faibussowitsch    - User explicitly requests auto-detection of the coverage tool from command line, and this
956*7ce81a4bSJacob Faibussowitsch      routine fails to guess the suitable tool name.
957*7ce81a4bSJacob Faibussowitsch    - The routine fails to find the tool, and --with-coverage is true
958*7ce81a4bSJacob Faibussowitsch    """
959*7ce81a4bSJacob Faibussowitsch    def log_print(msg, *args, **kwargs):
960*7ce81a4bSJacob Faibussowitsch      self.logPrint('checkCoverage: '+str(msg), *args, **kwargs)
961*7ce81a4bSJacob Faibussowitsch      return
962*7ce81a4bSJacob Faibussowitsch
963*7ce81a4bSJacob Faibussowitsch    def quoted(string):
964*7ce81a4bSJacob Faibussowitsch      return string.join(("'", "'"))
965*7ce81a4bSJacob Faibussowitsch
966*7ce81a4bSJacob Faibussowitsch    required         = bool(self.argDB['with-coverage'])
967*7ce81a4bSJacob Faibussowitsch    arg_opt          = self.argDB['with-coverage-exec']
968*7ce81a4bSJacob Faibussowitsch    use_default_path = True
969*7ce81a4bSJacob Faibussowitsch    search_path      = ''
970*7ce81a4bSJacob Faibussowitsch
971*7ce81a4bSJacob Faibussowitsch    log_print('{} to find an executable'.format('REQUIRED' if required else 'NOT required'))
972*7ce81a4bSJacob Faibussowitsch    if arg_opt in {'auto', 'default-auto', '1'}:
973*7ce81a4bSJacob Faibussowitsch      # detect it based on the C language compiler, hopefully this does not clash!
974*7ce81a4bSJacob Faibussowitsch      compiler = self.getCompiler(lang=self.setCompilers.languages.clanguage)
975*7ce81a4bSJacob Faibussowitsch      log_print('User did not explicitly set coverage exec (got {}), trying to auto-detect based on compiler {}'.format(quoted(arg_opt), quoted(compiler)))
976*7ce81a4bSJacob Faibussowitsch      if self.setCompilers.isGNU(compiler, self.log):
977*7ce81a4bSJacob Faibussowitsch        exec_names = ['gcov']
978*7ce81a4bSJacob Faibussowitsch      elif self.setCompilers.isClang(compiler, self.log):
979*7ce81a4bSJacob Faibussowitsch        exec_names = ['llvm-cov']
980*7ce81a4bSJacob Faibussowitsch        if self.setCompilers.isDarwin(self.log):
981*7ce81a4bSJacob Faibussowitsch          # macOS masquerades llvm-cov as just 'gcov', so we add this to the list in case
982*7ce81a4bSJacob Faibussowitsch          # bare llvm-cov does not work
983*7ce81a4bSJacob Faibussowitsch          exec_names.append('gcov')
984*7ce81a4bSJacob Faibussowitsch      elif arg_opt == 'default-auto' and not required:
985*7ce81a4bSJacob Faibussowitsch        # default-auto implies the user did not set it via command line!
986*7ce81a4bSJacob Faibussowitsch        log_print('Could not auto-detect coverage tool for {}, not a gnuish compiler. Bailing since user did not explicitly set exec on the commandline'.format(quoted(compiler)))
987*7ce81a4bSJacob Faibussowitsch        return
988*7ce81a4bSJacob Faibussowitsch      else:
989*7ce81a4bSJacob Faibussowitsch        # implies 'auto' explicitly set by user, or we were required to find
990*7ce81a4bSJacob Faibussowitsch        # something. either way we should error
991*7ce81a4bSJacob Faibussowitsch        raise RuntimeError('Could not auto-detect coverage tool for {}, please set coverage tool name explicitly'.format(quoted(compiler)))
992*7ce81a4bSJacob Faibussowitsch    else:
993*7ce81a4bSJacob Faibussowitsch      log_print('User explicitly set coverage exec as {}'.format(quoted(arg_opt)))
994*7ce81a4bSJacob Faibussowitsch      par_dir = os.path.dirname(arg_opt)
995*7ce81a4bSJacob Faibussowitsch      if os.path.exists(par_dir):
996*7ce81a4bSJacob Faibussowitsch        # arg_opt is path-like, we should only search the provided directory when we go
997*7ce81a4bSJacob Faibussowitsch        # looking for the tool
998*7ce81a4bSJacob Faibussowitsch        use_default_path = False
999*7ce81a4bSJacob Faibussowitsch        search_path      = par_dir
1000*7ce81a4bSJacob Faibussowitsch      exec_names = [arg_opt]
1001*7ce81a4bSJacob Faibussowitsch
1002*7ce81a4bSJacob Faibussowitsch    make_macro_name = 'PETSC_COVERAGE_EXEC'
1003*7ce81a4bSJacob Faibussowitsch    log_print('Checking for coverage tool(s):\n{}'.format('\n'.join('- '+t for t in exec_names)))
1004*7ce81a4bSJacob Faibussowitsch    found_exec = self.getExecutables(
1005*7ce81a4bSJacob Faibussowitsch      exec_names,
1006*7ce81a4bSJacob Faibussowitsch      path=search_path, getFullPath=True, useDefaultPath=use_default_path, resultName=make_macro_name
1007*7ce81a4bSJacob Faibussowitsch    )
1008*7ce81a4bSJacob Faibussowitsch
1009*7ce81a4bSJacob Faibussowitsch    if found_exec is None:
1010*7ce81a4bSJacob Faibussowitsch      # didn't find the coverage tool
1011*7ce81a4bSJacob Faibussowitsch      if required:
1012*7ce81a4bSJacob Faibussowitsch        raise RuntimeError('Coverage tool(s) {} could not be found. Please provide explicit path to coverage tool'.format(exec_names))
1013*7ce81a4bSJacob Faibussowitsch      return
1014*7ce81a4bSJacob Faibussowitsch
1015*7ce81a4bSJacob Faibussowitsch    found_exec_name = os.path.basename(found_exec)
1016*7ce81a4bSJacob Faibussowitsch    if 'llvm-cov' in found_exec_name and 'gcov' not in found_exec_name:
1017*7ce81a4bSJacob Faibussowitsch      # llvm-cov needs to be called as 'llvm-cov gcov' to work
1018*7ce81a4bSJacob Faibussowitsch      self.addMakeMacro(make_macro_name, found_exec + ' gcov')
1019*7ce81a4bSJacob Faibussowitsch    return
1020*7ce81a4bSJacob Faibussowitsch
1021f8833479SBarry Smith#-----------------------------------------------------------------------------------------------------
1022b10d012aSSatish Balay  def configureCygwinBrokenPipe(self):
1023b10d012aSSatish Balay    '''Cygwin version <= 1.7.18 had issues with pipes and long commands invoked from gnu-make
1024b10d012aSSatish Balay    http://cygwin.com/ml/cygwin/2013-05/msg00340.html '''
10257fca349cSMatthew G. Knepley    if config.setCompilers.Configure.isCygwin(self.log):
1026b10d012aSSatish Balay      import platform
1027b10d012aSSatish Balay      import re
1028b10d012aSSatish Balay      r=re.compile("([0-9]+).([0-9]+).([0-9]+)")
1029b10d012aSSatish Balay      m=r.match(platform.release())
1030b10d012aSSatish Balay      major=int(m.group(1))
1031b10d012aSSatish Balay      minor=int(m.group(2))
1032b10d012aSSatish Balay      subminor=int(m.group(3))
1033b10d012aSSatish Balay      if ((major < 1) or (major == 1 and minor < 7) or (major == 1 and minor == 7 and subminor <= 18)):
1034b10d012aSSatish Balay        self.addMakeMacro('PETSC_CYGWIN_BROKEN_PIPE','1')
1035b10d012aSSatish Balay    return
1036b10d012aSSatish Balay
1037b10d012aSSatish Balay#-----------------------------------------------------------------------------------------------------
1038569865ddSSatish Balay  def configureDefaultArch(self):
1039af0996ceSBarry Smith    conffile = os.path.join('lib','petsc','conf', 'petscvariables')
1040569865ddSSatish Balay    if self.framework.argDB['with-default-arch']:
1041c6ef1b5bSJed Brown      fd = open(conffile, 'w')
1042569865ddSSatish Balay      fd.write('PETSC_ARCH='+self.arch.arch+'\n')
1043da93591fSBarry Smith      fd.write('PETSC_DIR='+self.petscdir.dir+'\n')
1044b9b902edSJed Brown      fd.write('include '+os.path.join('$(PETSC_DIR)','$(PETSC_ARCH)','lib','petsc','conf','petscvariables')+'\n')
1045569865ddSSatish Balay      fd.close()
1046569865ddSSatish Balay      self.framework.actions.addArgument('PETSc', 'Build', 'Set default architecture to '+self.arch.arch+' in '+conffile)
1047569865ddSSatish Balay    elif os.path.isfile(conffile):
1048569865ddSSatish Balay      try:
1049569865ddSSatish Balay        os.unlink(conffile)
1050569865ddSSatish Balay      except:
1051569865ddSSatish Balay        raise RuntimeError('Unable to remove file '+conffile+'. Did a different user create it?')
1052569865ddSSatish Balay    return
1053569865ddSSatish Balay
1054569865ddSSatish Balay#-----------------------------------------------------------------------------------------------------
1055f8833479SBarry Smith  def configureScript(self):
1056f8833479SBarry Smith    '''Output a script in the conf directory which will reproduce the configuration'''
1057f8833479SBarry Smith    import nargs
1058495bf1a9SSatish Balay    import sys
1059af0996ceSBarry Smith    scriptName = os.path.join(self.arch.arch,'lib','petsc','conf', 'reconfigure-'+self.arch.arch+'.py')
1060f8833479SBarry Smith    args = dict([(nargs.Arg.parseArgument(arg)[0], arg) for arg in self.framework.clArgs])
1061e97fc2efSSatish Balay    if 'with-clean' in args:
1062e97fc2efSSatish Balay      del args['with-clean']
1063d418e2d7SSatish Balay    if 'force' in args:
1064d418e2d7SSatish Balay      del args['force']
1065f8833479SBarry Smith    if 'configModules' in args:
10661063a081SSatish Balay      if nargs.Arg.parseArgument(args['configModules'])[1] == 'PETSc.Configure':
1067f8833479SBarry Smith        del args['configModules']
1068f8833479SBarry Smith    if 'optionsModule' in args:
106923a19ef1SSatish Balay      if nargs.Arg.parseArgument(args['optionsModule'])[1] == 'config.compilerOptions':
1070f8833479SBarry Smith        del args['optionsModule']
1071f8833479SBarry Smith    if not 'PETSC_ARCH' in args:
10721063a081SSatish Balay      args['PETSC_ARCH'] = 'PETSC_ARCH='+str(self.arch.arch)
1073c6ef1b5bSJed Brown    f = open(scriptName, 'w')
1074495bf1a9SSatish Balay    f.write('#!'+sys.executable+'\n')
1075f8833479SBarry Smith    f.write('if __name__ == \'__main__\':\n')
1076f8833479SBarry Smith    f.write('  import sys\n')
10777561c02cSSatish Balay    f.write('  import os\n')
10787561c02cSSatish Balay    f.write('  sys.path.insert(0, os.path.abspath(\'config\'))\n')
1079f8833479SBarry Smith    f.write('  import configure\n')
10801063a081SSatish Balay    # pretty print repr(args.values())
10811063a081SSatish Balay    f.write('  configure_options = [\n')
10828bec23c5SJed Brown    for itm in sorted(args.values()):
10831063a081SSatish Balay      f.write('    \''+str(itm)+'\',\n')
10841063a081SSatish Balay    f.write('  ]\n')
1085f8833479SBarry Smith    f.write('  configure.petsc_configure(configure_options)\n')
1086f8833479SBarry Smith    f.close()
1087f8833479SBarry Smith    try:
10885b6bfdb9SJed Brown      os.chmod(scriptName, 0o775)
10895b6bfdb9SJed Brown    except OSError as e:
1090f8833479SBarry Smith      self.framework.logPrint('Unable to make reconfigure script executable:\n'+str(e))
1091f8833479SBarry Smith    self.framework.actions.addArgument('PETSc', 'File creation', 'Created '+scriptName+' for automatic reconfiguration')
1092f8833479SBarry Smith    return
1093f8833479SBarry Smith
1094f8833479SBarry Smith  def configureInstall(self):
1095f8833479SBarry Smith    '''Setup the directories for installation'''
1096f8833479SBarry Smith    if self.framework.argDB['prefix']:
10975b4fc442SVaclav Hapla      self.addMakeRule('print_mesg_after_build','',
10985b4fc442SVaclav Hapla       ['-@echo "========================================="',
10995b4fc442SVaclav Hapla        '-@echo "Now to install the libraries do:"',
11005b4fc442SVaclav Hapla        '-@echo "%s${MAKE_USER} PETSC_DIR=${PETSC_DIR} PETSC_ARCH=${PETSC_ARCH} install"' % self.installdir.installSudo,
1101315b77e6SSatish Balay        '-@echo "========================================="'])
1102f8833479SBarry Smith    else:
11035b4fc442SVaclav Hapla      self.addMakeRule('print_mesg_after_build','',
11045b4fc442SVaclav Hapla       ['-@echo "========================================="',
11055b4fc442SVaclav Hapla        '-@echo "Now to check if the libraries are working do:"',
11065b4fc442SVaclav Hapla        '-@echo "${MAKE_USER} PETSC_DIR=${PETSC_DIR} PETSC_ARCH=${PETSC_ARCH} check"',
1107315b77e6SSatish Balay        '-@echo "========================================="'])
1108f8833479SBarry Smith      return
1109f8833479SBarry Smith
111028bb2e72SSatish Balay  def postProcessPackages(self):
111128bb2e72SSatish Balay    postPackages=[]
111228bb2e72SSatish Balay    for i in self.framework.packages:
111328bb2e72SSatish Balay      if hasattr(i,'postProcess'): postPackages.append(i)
111428bb2e72SSatish Balay    if postPackages:
1115e64d19dfSSatish Balay      # ctetgen needs petsc conf files. so attempt to create them early
1116a77eb93bSSatish Balay      self.framework.dumpConfFiles()
1117d9293e7bSBarry Smith      # tacky fix for dependency of Aluimia on Pflotran; requested via petsc-dev Matt provide a correct fix
1118d9293e7bSBarry Smith      for i in postPackages:
1119d9293e7bSBarry Smith        if i.name.upper() in ['PFLOTRAN']:
1120d9293e7bSBarry Smith          i.postProcess()
1121d9293e7bSBarry Smith          postPackages.remove(i)
112228bb2e72SSatish Balay      for i in postPackages: i.postProcess()
1123aa5c8b8eSBarry Smith      for i in postPackages:
1124aa5c8b8eSBarry Smith        if i.installedpetsc:
1125aa5c8b8eSBarry Smith          self.installed = 1
1126aa5c8b8eSBarry Smith          break
112728bb2e72SSatish Balay    return
1128f8833479SBarry Smith
1129f8833479SBarry Smith  def configure(self):
1130bf3e94a3SBarry Smith    if 'package-prefix-hash' in self.argDB:
1131bf3e94a3SBarry Smith      # turn off prefix if it was only used to for installing external packages.
1132bf3e94a3SBarry Smith      self.framework.argDB['prefix'] = ''
1133bf3e94a3SBarry Smith      self.dir = os.path.abspath(os.path.join(self.petscdir.dir, self.arch.arch))
1134bf3e94a3SBarry Smith      self.installdir.dir = self.dir
1135bf3e94a3SBarry Smith      self.installdir.petscDir = self.petscdir.dir
1136bf3e94a3SBarry Smith      self.petscDir = self.petscdir.dir
1137bf3e94a3SBarry Smith      self.petscArch = self.arch.arch
1138bf3e94a3SBarry Smith      self.addMakeMacro('PREFIXDIR',self.dir)
1139bf3e94a3SBarry Smith      self.confDir = os.path.abspath(os.path.join(self.petscdir.dir, self.arch.arch))
1140bf3e94a3SBarry Smith
1141f8833479SBarry Smith    if not os.path.samefile(self.petscdir.dir, os.getcwd()):
1142f8833479SBarry Smith      raise RuntimeError('Wrong PETSC_DIR option specified: '+str(self.petscdir.dir) + '\n  Configure invoked in: '+os.path.realpath(os.getcwd()))
1143550489e3SMatthew 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):
11443552d8fbSSatish Balay      raise RuntimeError('Incorrect option --prefix='+self.framework.argDB['prefix']+' specified. It cannot be same as PETSC_DIR!')
11458fd0dbdbSBarry Smith    if self.framework.argDB['prefix'] and self.framework.argDB['prefix'].find(' ') > -1:
11468fd0dbdbSBarry 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')
1147c16c35a9SSatish 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)):
1148c16c35a9SSatish Balay      raise RuntimeError('Incorrect option --prefix='+self.framework.argDB['prefix']+' specified. It cannot be same as PETSC_DIR/PETSC_ARCH!')
1149f16c1317SJed Brown    self.framework.header          = os.path.join(self.arch.arch,'include','petscconf.h')
1150f16c1317SJed Brown    self.framework.cHeader         = os.path.join(self.arch.arch,'include','petscfix.h')
1151bf113f49SJacob Faibussowitsch    self.framework.poisonheader    = os.path.join(self.arch.arch,'include','petscconf_poison.h')
11529c735a01SStefano Zampini    self.framework.pkgheader       = os.path.join(self.arch.arch,'include','petscpkg_version.h')
1153af0996ceSBarry Smith    self.framework.makeMacroHeader = os.path.join(self.arch.arch,'lib','petsc','conf','petscvariables')
1154af0996ceSBarry Smith    self.framework.makeRuleHeader  = os.path.join(self.arch.arch,'lib','petsc','conf','petscrules')
1155f8833479SBarry Smith    if self.libraries.math is None:
1156f8833479SBarry Smith      raise RuntimeError('PETSc requires a functional math library. Please send configure.log to petsc-maint@mcs.anl.gov.')
1157f8833479SBarry Smith    if self.languages.clanguage == 'Cxx' and not hasattr(self.compilers, 'CXX'):
1158f8833479SBarry Smith      raise RuntimeError('Cannot set C language to C++ without a functional C++ compiler.')
1159ed938b00SJed Brown    self.executeTest(self.configureRTLDDefault)
1160b2843cf1SBarry Smith    self.executeTest(self.configurePrefetch)
11612400fdedSBarry Smith    self.executeTest(self.configureUnused)
11621ef8df7fSJed Brown    self.executeTest(self.configureDeprecated)
116398ed35c3SBarry Smith    self.executeTest(self.configureIsatty)
1164e8e972b2SVaclav Hapla    self.executeTest(self.configureExpect)
1165e8e972b2SVaclav Hapla    self.executeTest(self.configureAlign)
1166e8e972b2SVaclav Hapla    self.executeTest(self.configureFunctionName)
1167e8e972b2SVaclav Hapla    self.executeTest(self.configureIntptrt)
1168f8833479SBarry Smith    self.executeTest(self.configureSolaris)
1169f8833479SBarry Smith    self.executeTest(self.configureLinux)
11700f64ec89SBarry Smith    self.executeTest(self.configureDarwin)
1171f8833479SBarry Smith    self.executeTest(self.configureWin32)
1172b10d012aSSatish Balay    self.executeTest(self.configureCygwinBrokenPipe)
1173569865ddSSatish Balay    self.executeTest(self.configureDefaultArch)
1174f8833479SBarry Smith    self.executeTest(self.configureScript)
1175f8833479SBarry Smith    self.executeTest(self.configureInstall)
117609bc878fSSatish Balay    self.executeTest(self.configureAtoll)
1177*7ce81a4bSJacob Faibussowitsch    for LANG in ['C', 'Cxx', 'CUDA', 'HIP', 'SYCL', 'FC']:
1178*7ce81a4bSJacob Faibussowitsch      compilerName = LANG.upper() if LANG in {'Cxx', 'FC'} else LANG+'C'
1179*7ce81a4bSJacob Faibussowitsch      if hasattr(self.setCompilers, compilerName):
1180*7ce81a4bSJacob Faibussowitsch        kwargs = {}
1181*7ce81a4bSJacob Faibussowitsch        if LANG in {'CUDA'}:
1182*7ce81a4bSJacob Faibussowitsch          # nvcc preprocesses the base file into a bunch of intermediate files, which are
1183*7ce81a4bSJacob Faibussowitsch          # then compiled by the host compiler. Why is this a problem?  Because the
1184*7ce81a4bSJacob Faibussowitsch          # generated coverage data is based on these preprocessed source files! So gcov
1185*7ce81a4bSJacob Faibussowitsch          # tries to read it later, but since its in the tmp directory it cannot. Thus we
1186*7ce81a4bSJacob Faibussowitsch          # need to keep them around (in a place we know about).
1187*7ce81a4bSJacob Faibussowitsch          nvcc_tmp_dir = os.path.join(self.petscdir.dir, self.arch.arch, 'nvcc_tmp')
1188*7ce81a4bSJacob Faibussowitsch          try:
1189*7ce81a4bSJacob Faibussowitsch            os.mkdir(nvcc_tmp_dir)
1190*7ce81a4bSJacob Faibussowitsch          except FileExistsError:
1191*7ce81a4bSJacob Faibussowitsch            pass
1192*7ce81a4bSJacob Faibussowitsch          kwargs['extra_coverage_flags'] = [
1193*7ce81a4bSJacob Faibussowitsch            '-Xcompiler --coverage -Xcompiler -fPIC --keep --keep-dir={}'.format(nvcc_tmp_dir)
1194*7ce81a4bSJacob Faibussowitsch          ]
1195*7ce81a4bSJacob Faibussowitsch          if self.kokkos.found:
1196*7ce81a4bSJacob Faibussowitsch            # yet again the kokkos nvcc_wrapper goes out of its way to be as useless as
1197*7ce81a4bSJacob Faibussowitsch            # possible. Its default arch (sm_35) is actually too low to compile kokkos,
1198*7ce81a4bSJacob Faibussowitsch            # for whatever reason this works if you dont use the --keep and --keep-dir
1199*7ce81a4bSJacob Faibussowitsch            # flags above.
1200*7ce81a4bSJacob Faibussowitsch            kwargs['extra_coverage_flags'].append('-arch=native')
1201*7ce81a4bSJacob Faibussowitsch          kwargs['extra_debug_flags'] = ['-Xcompiler -Og']
1202*7ce81a4bSJacob Faibussowitsch        self.executeTest(self.configureCoverage, args=[LANG], kargs=kwargs)
1203*7ce81a4bSJacob Faibussowitsch    self.executeTest(self.configureCoverageExecutable)
1204f8833479SBarry Smith
1205f8833479SBarry Smith    self.Dump()
1206f8833479SBarry Smith    self.dumpConfigInfo()
12072a4161d9SMatthew G Knepley    self.dumpMachineInfo()
120849fe22e6SSatish Balay    self.delGenFiles()
120940277576SBarry Smith    # need to save the current state of BuildSystem so that postProcess() packages can read it in and perhaps run make install
121040277576SBarry Smith    self.framework.storeSubstitutions(self.framework.argDB)
1211492432c8SJed Brown    self.framework.argDB['configureCache'] = pickle.dumps(self.framework)
121240277576SBarry Smith    self.framework.argDB.save(force = True)
12138244ab14SJed Brown    self.DumpPkgconfig('PETSc.pc')
12148244ab14SJed Brown    self.DumpPkgconfig('petsc.pc')
1215351d3a41SMatthew G Knepley    self.DumpModule()
1216f7ad81e1SBarry Smith    self.postProcessPackages()
1217f8833479SBarry Smith    self.framework.log.write('================================================================================\n')
1218f8833479SBarry Smith    self.logClear()
1219f8833479SBarry Smith    return
1220