xref: /petsc/config/PETSc/Configure.py (revision 9b81490a800f1846a667dcbc6448a0dbbfe9b1c7)
1f8833479SBarry Smithimport config.base
2f8833479SBarry Smith
3f8833479SBarry Smithimport os
46dd73af6SBarry Smithimport sys
5f8833479SBarry Smithimport re
6492432c8SJed Brownimport pickle
7*9b81490aSJacob Faibussowitschimport textwrap
8f8833479SBarry Smith
9f8833479SBarry Smithclass Configure(config.base.Configure):
10f8833479SBarry Smith  def __init__(self, framework):
11f8833479SBarry Smith    config.base.Configure.__init__(self, framework)
12f8833479SBarry Smith    self.headerPrefix = 'PETSC'
13f8833479SBarry Smith    self.substPrefix  = 'PETSC'
14aa5c8b8eSBarry Smith    self.installed    = 0 # 1 indicates that Configure itself has already compiled and installed PETSc
154161761dSBarry Smith    self.found        = 1
16f8833479SBarry Smith    return
17f8833479SBarry Smith
187c939e48SSatish Balay  def __str2__(self):
195b4fc442SVaclav Hapla    desc = ['  Using GNU make: ' + self.make.make]
207ce81a4bSJacob Faibussowitsch    if self.defines.get('USE_COVERAGE'):
217ce81a4bSJacob Faibussowitsch      desc.extend([
227ce81a4bSJacob Faibussowitsch        '  Code coverage: yes',
237ce81a4bSJacob Faibussowitsch        '  Using code coverage executable: {}'.format(self.getMakeMacro('PETSC_COVERAGE_EXEC'))
247ce81a4bSJacob Faibussowitsch      ])
25aa5c8b8eSBarry Smith    if not self.installed:
26a0022257SSatish Balay      desc.append('xxx=========================================================================xxx')
27dc0529c6SBarry Smith      desc.append(' Configure stage complete. Now build PETSc libraries with:')
285b4fc442SVaclav Hapla      desc.append('   %s PETSC_DIR=%s PETSC_ARCH=%s all' % (self.make.make_user, self.petscdir.dir, self.arch.arch))
29a0022257SSatish Balay      desc.append('xxx=========================================================================xxx')
30aa5c8b8eSBarry Smith    else:
31aa5c8b8eSBarry Smith      desc.append('xxx=========================================================================xxx')
32aa5c8b8eSBarry Smith      desc.append(' Installation complete. You do not need to run make to compile or install the software')
33aa5c8b8eSBarry Smith      desc.append('xxx=========================================================================xxx')
347ce81a4bSJacob Faibussowitsch    desc.append('')
357ce81a4bSJacob Faibussowitsch    return '\n'.join(desc)
36f8833479SBarry Smith
37f8833479SBarry Smith  def setupHelp(self, help):
38f8833479SBarry Smith    import nargs
39ce0b2093SBarry Smith    help.addArgument('PETSc',  '-prefix=<dir>',                              nargs.Arg(None, '', 'Specifiy location to install PETSc (eg. /usr/local)'))
407deb5ab3SBarry Smith    help.addArgument('PETSc',  '-with-prefetch=<bool>',                      nargs.ArgBool(None, 1,'Enable checking for prefetch instructions'))
41eed94e11SSatish Balay    help.addArgument('Windows','-with-windows-graphics=<bool>',              nargs.ArgBool(None, 1,'Enable check for Windows Graphics'))
42569865ddSSatish Balay    help.addArgument('PETSc', '-with-default-arch=<bool>',                   nargs.ArgBool(None, 1, 'Allow using the last configured arch without setting PETSC_ARCH'))
4357cb31baSSatish Balay    help.addArgument('PETSc','-with-single-library=<bool>',                  nargs.ArgBool(None, 1,'Put all PETSc code into the single -lpetsc library'))
44cb297985SSatish Balay    help.addArgument('PETSc','-with-fortran-bindings=<bool>',                nargs.ArgBool(None, 1,'Build PETSc fortran bindings in the library and corresponding module files'))
45525d6f2eSBarry Smith    help.addArgument('PETSc', '-with-ios=<bool>',                            nargs.ArgBool(None, 0, 'Build an iPhone/iPad version of PETSc library'))
4613f9d092SSatish Balay    help.addArgument('PETSc', '-with-display=<x11display>',                  nargs.Arg(None, '', 'Specifiy DISPLAY env variable for use with matlab test)'))
472c30b4dfSSatish Balay    help.addArgument('PETSc', '-with-package-scripts=<pyscripts>',           nargs.ArgFileList(None,None,'Specify configure package scripts for user provided packages'))
487ce81a4bSJacob Faibussowitsch    help.addArgument('PETSc', '-with-coverage=<bool>',                       nargs.ArgFuzzyBool(None, value=0, help='Enable or disable code-coverage collection'))
497ce81a4bSJacob 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'))
50a0c7f9aaSSamuel Khuvis    help.addArgument('PETSc', '-with-tau-perfstubs=<bool>',                  nargs.ArgBool(None, 1,'Enable TAU profiler stubs'))
51689a5dfaSJacob Faibussowitsch    help.addArgument('PETSc', '-with-strict-petscerrorcode=<bool>',          nargs.ArgFuzzyBool(None, value=0, help='Enable strict PetscErrorCode mode, which enables additional compile-time checking for misuse of PetscErrorCode and error handling'))
52f8833479SBarry Smith    return
53f8833479SBarry Smith
546dd73af6SBarry Smith  def registerPythonFile(self,filename,directory):
556dd73af6SBarry Smith    ''' Add a python file to the framework and registers its headerprefix, ... externalpackagedir
566dd73af6SBarry Smith        directory is the directory where the file relative to the BuildSystem or config path in python notation with . '''
576dd73af6SBarry Smith    (utilityName, ext) = os.path.splitext(filename)
586dd73af6SBarry Smith    if not utilityName.startswith('.') and not utilityName.startswith('#') and ext == '.py' and not utilityName == '__init__':
596dd73af6SBarry Smith      if directory: directory = directory+'.'
606dd73af6SBarry Smith      utilityObj                             = self.framework.require(directory+utilityName, self)
616dd73af6SBarry Smith      utilityObj.headerPrefix                = self.headerPrefix
626dd73af6SBarry Smith      utilityObj.archProvider                = self.arch
636dd73af6SBarry Smith      utilityObj.languageProvider            = self.languages
646dd73af6SBarry Smith      utilityObj.installDirProvider          = self.installdir
656dd73af6SBarry Smith      utilityObj.externalPackagesDirProvider = self.externalpackagesdir
666dd73af6SBarry Smith      utilityObj.precisionProvider           = self.scalartypes
676dd73af6SBarry Smith      utilityObj.indexProvider               = self.indexTypes
686dd73af6SBarry Smith      setattr(self, utilityName.lower(), utilityObj)
6951294b80SMatthew G. Knepley      return utilityObj
7051294b80SMatthew G. Knepley    return None
716dd73af6SBarry Smith
72f8833479SBarry Smith  def setupDependencies(self, framework):
73f8833479SBarry Smith    config.base.Configure.setupDependencies(self, framework)
74dca78d2bSSatish Balay    self.programs      = framework.require('config.programs',           self)
75f8833479SBarry Smith    self.setCompilers  = framework.require('config.setCompilers',       self)
767ce81a4bSJacob Faibussowitsch    self.compilerFlags = framework.require('config.compilerFlags',      self)
7730b8aa07SMatthew G. Knepley    self.compilers     = framework.require('config.compilers',          self)
789d310bb7SBarry Smith    self.arch          = framework.require('PETSc.options.arch',        self.setCompilers)
799d310bb7SBarry Smith    self.petscdir      = framework.require('PETSc.options.petscdir',    self.arch)
809d310bb7SBarry Smith    self.installdir    = framework.require('PETSc.options.installDir',  self)
814e00a515SSatish Balay    self.dataFilesPath = framework.require('PETSc.options.dataFilesPath',self)
826dd73af6SBarry Smith    self.scalartypes   = framework.require('PETSc.options.scalarTypes', self)
836dd73af6SBarry Smith    self.indexTypes    = framework.require('PETSc.options.indexTypes',  self)
849d310bb7SBarry Smith    self.languages     = framework.require('PETSc.options.languages',   self.setCompilers)
8530b8aa07SMatthew G. Knepley    self.indexTypes    = framework.require('PETSc.options.indexTypes',  self.compilers)
86f8833479SBarry Smith    self.types         = framework.require('config.types',              self)
87f8833479SBarry Smith    self.headers       = framework.require('config.headers',            self)
88f8833479SBarry Smith    self.functions     = framework.require('config.functions',          self)
89f8833479SBarry Smith    self.libraries     = framework.require('config.libraries',          self)
90cd37d877SShri Abhyankar    self.atomics       = framework.require('config.atomics',            self)
919481793eSSatish Balay    self.make          = framework.require('config.packages.make',      self)
929552296fSBarry Smith    self.blasLapack    = framework.require('config.packages.BlasLapack',self)
93e6b0c433SBarry Smith    self.mpi           = framework.require('config.packages.MPI',       self)
940542e31aSBarry Smith    self.fortran       = framework.require('config.compilersFortran',   self)
950542e31aSBarry Smith    self.externalpackagesdir = framework.require('PETSc.options.externalpackagesdir',self)
9649d43ecaSSatish Balay
9709a6cbfcSBernhard M. Wiedemann    for utility in sorted(os.listdir(os.path.join('config','PETSc','options'))):
986dd73af6SBarry Smith      self.registerPythonFile(utility,'PETSc.options')
999d310bb7SBarry Smith
10009a6cbfcSBernhard M. Wiedemann    for utility in sorted(os.listdir(os.path.join('config','BuildSystem','config','utilities'))):
1016dd73af6SBarry Smith      self.registerPythonFile(utility,'config.utilities')
10206e08bc7SBarry Smith
10309a6cbfcSBernhard M. Wiedemann    for package in sorted(os.listdir(os.path.join('config', 'BuildSystem', 'config', 'packages'))):
10451294b80SMatthew G. Knepley      obj = self.registerPythonFile(package,'config.packages')
10551294b80SMatthew G. Knepley      if obj:
10651294b80SMatthew G. Knepley        obj.archProvider                = self.framework.requireModule(obj.archProvider, obj)
10751294b80SMatthew G. Knepley        obj.languageProvider            = self.framework.requireModule(obj.languageProvider, obj)
10851294b80SMatthew G. Knepley        obj.installDirProvider          = self.framework.requireModule(obj.installDirProvider, obj)
10951294b80SMatthew G. Knepley        obj.externalPackagesDirProvider = self.framework.requireModule(obj.externalPackagesDirProvider, obj)
11051294b80SMatthew G. Knepley        obj.precisionProvider           = self.framework.requireModule(obj.precisionProvider, obj)
11151294b80SMatthew G. Knepley        obj.indexProvider               = self.framework.requireModule(obj.indexProvider, obj)
1126dd73af6SBarry Smith
1135faf1eacSMatthew G. Knepley    # Force blaslapack and opencl to depend on scalarType so precision is set before BlasLapack is built
1149d310bb7SBarry Smith    framework.require('PETSc.options.scalarTypes', self.f2cblaslapack)
1159d310bb7SBarry Smith    framework.require('PETSc.options.scalarTypes', self.fblaslapack)
1169d310bb7SBarry Smith    framework.require('PETSc.options.scalarTypes', self.blaslapack)
1175faf1eacSMatthew G. Knepley    framework.require('PETSc.options.scalarTypes', self.opencl)
118f8833479SBarry Smith
119dca78d2bSSatish Balay    self.programs.headerPrefix     = self.headerPrefix
1205b3958d3SJacob Faibussowitsch    self.setCompilers.headerPrefix = self.headerPrefix
121f8833479SBarry Smith    self.compilers.headerPrefix    = self.headerPrefix
1220542e31aSBarry Smith    self.fortran.headerPrefix      = self.headerPrefix
123f8833479SBarry Smith    self.types.headerPrefix        = self.headerPrefix
124f8833479SBarry Smith    self.headers.headerPrefix      = self.headerPrefix
125f8833479SBarry Smith    self.functions.headerPrefix    = self.headerPrefix
126f8833479SBarry Smith    self.libraries.headerPrefix    = self.headerPrefix
1276dd73af6SBarry Smith
1282c30b4dfSSatish Balay    # Register user provided package scripts
1292c30b4dfSSatish Balay    if 'with-package-scripts' in self.framework.argDB:
1302c30b4dfSSatish Balay      for script in self.framework.argDB['with-package-scripts']:
1312c30b4dfSSatish Balay        if os.path.splitext(script)[1] != '.py':
1322c30b4dfSSatish Balay          raise RuntimeError('Only python scripts compatible with configure package script format should be specified! Invalid option -with-package-scripts='+script)
1332c30b4dfSSatish Balay        self.framework.logPrint('User is registering a new package script: '+script)
1342c30b4dfSSatish Balay        dname,fname = os.path.split(script)
1352c30b4dfSSatish Balay        if dname: sys.path.append(dname)
1362c30b4dfSSatish Balay        self.registerPythonFile(fname,'')
1376dd73af6SBarry Smith
1386dd73af6SBarry Smith    # test for a variety of basic headers and functions
1394211eb48SBarry Smith    headersC = map(lambda name: name+'.h',['setjmp','dos','fcntl','float','io','malloc','pwd','strings',
140ace159c0SJed Brown                                            'unistd','machine/endian','sys/param','sys/procfs','sys/resource',
1412475b7caSBarry Smith                                            'sys/systeminfo','sys/times','sys/utsname',
1427e4f0192SMosè Giordano                                            'sys/socket','sys/wait','netinet/in','netdb','direct','time','Ws2tcpip','sys/types',
14318da0197SPierre Jolivet                                            'WindowsX','float','ieeefp','stdint','inttypes','immintrin'])
14445082d64SJed Brown    functions = ['access','_access','clock','drand48','getcwd','_getcwd','getdomainname','gethostname',
145d7976ebaSJed Brown                 'getwd','posix_memalign','popen','PXFGETARG','rand','getpagesize',
1464211eb48SBarry Smith                 'readlink','realpath','usleep','sleep','_sleep',
1472475b7caSBarry Smith                 'uname','snprintf','_snprintf','lseek','_lseek','time','fork','stricmp',
1482475b7caSBarry Smith                 'strcasecmp','bzero','dlopen','dlsym','dlclose','dlerror',
14967f4e542SPierre Jolivet                 '_set_output_format','_mkdir','socket','gethostbyname','fpresetsticky',
15018da0197SPierre Jolivet                 'fpsetsticky','__gcov_dump']
151b0651e32SBarry Smith    libraries = [(['fpe'],'handle_sigfpes')]
152b0651e32SBarry Smith    librariessock = [(['socket','nsl'],'socket')]
153f8833479SBarry Smith    self.headers.headers.extend(headersC)
154f8833479SBarry Smith    self.functions.functions.extend(functions)
155b0651e32SBarry Smith    self.libraries.libraries.extend(libraries)
156b0651e32SBarry Smith    if not hasattr(self,'socket'):
157b0651e32SBarry Smith      self.libraries.libraries.extend(librariessock)
158f8833479SBarry Smith    return
159f8833479SBarry Smith
1608244ab14SJed Brown  def DumpPkgconfig(self, petsc_pc):
161262119f8SBarry Smith    ''' Create a pkg-config file '''
162262119f8SBarry Smith    if not os.path.exists(os.path.join(self.petscdir.dir,self.arch.arch,'lib','pkgconfig')):
163262119f8SBarry Smith      os.makedirs(os.path.join(self.petscdir.dir,self.arch.arch,'lib','pkgconfig'))
1642eefe1c6SJed Brown    with open(os.path.join(self.petscdir.dir,self.arch.arch,'lib','pkgconfig',petsc_pc),'w') as fd:
1655e3311eeSJed Brown      cflags_inc = ['-I${includedir}']
166262119f8SBarry Smith      if self.framework.argDB['prefix']:
1675bb5b263SMatthew G. Knepley        fd.write('prefix='+self.installdir.dir+'\n')
168262119f8SBarry Smith      else:
169e1e675deSJed Brown        fd.write('prefix='+os.path.join(self.petscdir.dir, self.arch.arch)+'\n')
170e1e675deSJed Brown        cflags_inc.append('-I' + os.path.join(self.petscdir.dir, 'include'))
171262119f8SBarry Smith      fd.write('exec_prefix=${prefix}\n')
172262119f8SBarry Smith      fd.write('includedir=${prefix}/include\n')
1735e3311eeSJed Brown      fd.write('libdir=${prefix}/lib\n')
174262119f8SBarry Smith
1752eefe1c6SJed Brown      with self.setCompilers.Language('C'):
176262119f8SBarry Smith        fd.write('ccompiler='+self.setCompilers.getCompiler()+'\n')
177756c7f9fSJed Brown        fd.write('cflags_extra='+self.setCompilers.getCompilerFlags().strip()+'\n')
178756c7f9fSJed Brown        fd.write('cflags_dep='+self.compilers.dependenciesGenerationFlag.get('C','')+'\n')
179756c7f9fSJed Brown        fd.write('ldflag_rpath='+self.setCompilers.CSharedLinkerFlag+'\n')
18003e383c8SJed Brown      if hasattr(self.compilers, 'CXX'):
1812eefe1c6SJed Brown        with self.setCompilers.Language('C++'):
182262119f8SBarry Smith          fd.write('cxxcompiler='+self.setCompilers.getCompiler()+'\n')
183756c7f9fSJed Brown          fd.write('cxxflags_extra='+self.setCompilers.getCompilerFlags().strip()+'\n')
184262119f8SBarry Smith      if hasattr(self.compilers, 'FC'):
1852eefe1c6SJed Brown        with self.setCompilers.Language('FC'):
186262119f8SBarry Smith          fd.write('fcompiler='+self.setCompilers.getCompiler()+'\n')
187756c7f9fSJed Brown          fd.write('fflags_extra='+self.setCompilers.getCompilerFlags().strip()+'\n')
18850520af6SPatrick Sanan      if hasattr(self.compilers, 'CUDAC'):
18950520af6SPatrick Sanan        with self.setCompilers.Language('CUDA'):
19050520af6SPatrick Sanan          fd.write('cudacompiler='+self.setCompilers.getCompiler()+'\n')
19150520af6SPatrick Sanan          fd.write('cudaflags_extra='+self.setCompilers.getCompilerFlags().strip()+'\n')
1928f561fa3SPatrick Sanan          p = self.framework.require('config.packages.cuda')
19350520af6SPatrick Sanan          fd.write('cudalib='+self.libraries.toStringNoDupes(p.lib)+'\n')
19450520af6SPatrick Sanan          fd.write('cudainclude='+self.headers.toStringNoDupes(p.include)+'\n')
1957ba7a817SBarry Smith          if hasattr(self.setCompilers,'CUDA_CXX'):
1967ba7a817SBarry Smith            fd.write('cuda_cxx='+self.setCompilers.CUDA_CXX+'\n')
1977ba7a817SBarry Smith            fd.write('cuda_cxxflags='+self.setCompilers.CUDA_CXXFLAGS+'\n')
198262119f8SBarry Smith
199262119f8SBarry Smith      fd.write('\n')
200262119f8SBarry Smith      fd.write('Name: PETSc\n')
201262119f8SBarry Smith      fd.write('Description: Library to solve ODEs and algebraic equations\n')
202351d3a41SMatthew G Knepley      fd.write('Version: %s\n' % self.petscdir.version)
2035e3311eeSJed Brown      fd.write('Cflags: ' + ' '.join([self.setCompilers.CPPFLAGS] + cflags_inc) + '\n')
20437371b91SJed Brown      fd.write('Libs: '+self.libraries.toStringNoDupes(['-L${libdir}', self.petsclib], with_rpath=False)+'\n')
2058ebf8858SJed Brown      # Remove RPATH flags from library list.  User can add them using
2068ebf8858SJed Brown      # pkg-config --variable=ldflag_rpath and pkg-config --libs-only-L
207de8f682fSSatish 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')
208262119f8SBarry Smith    return
209262119f8SBarry Smith
210351d3a41SMatthew G Knepley  def DumpModule(self):
211351d3a41SMatthew G Knepley    ''' Create a module file '''
212af0996ceSBarry Smith    if not os.path.exists(os.path.join(self.petscdir.dir,self.arch.arch,'lib','petsc','conf','modules')):
213af0996ceSBarry Smith      os.makedirs(os.path.join(self.petscdir.dir,self.arch.arch,'lib','petsc','conf','modules'))
214af0996ceSBarry Smith    if not os.path.exists(os.path.join(self.petscdir.dir,self.arch.arch,'lib','petsc','conf','modules','petsc')):
215af0996ceSBarry Smith      os.makedirs(os.path.join(self.petscdir.dir,self.arch.arch,'lib','petsc','conf','modules','petsc'))
216351d3a41SMatthew G Knepley    if self.framework.argDB['prefix']:
2175bb5b263SMatthew G. Knepley      installdir  = self.installdir.dir
21855d606a3SSatish Balay      installarch = ''
21955d606a3SSatish Balay      installpath = os.path.join(installdir,'bin')
220351d3a41SMatthew G Knepley    else:
221351d3a41SMatthew G Knepley      installdir  = self.petscdir.dir
22255d606a3SSatish Balay      installarch = self.arch.arch
22355d606a3SSatish Balay      installpath = os.path.join(installdir,installarch,'bin')+':'+os.path.join(installdir,'bin')
224af0996ceSBarry Smith    fd = open(os.path.join(self.petscdir.dir,self.arch.arch,'lib','petsc','conf','modules','petsc',self.petscdir.version),'w')
225351d3a41SMatthew G Knepley    fd.write('''\
226351d3a41SMatthew G Knepley#%%Module
227351d3a41SMatthew G Knepley
228351d3a41SMatthew G Knepleyproc ModulesHelp { } {
229351d3a41SMatthew G Knepley    puts stderr "This module sets the path and environment variables for petsc-%s"
230a17b96a8SKyle Gerard Felker    puts stderr "     see https://petsc.org/ for more information      "
231351d3a41SMatthew G Knepley    puts stderr ""
232351d3a41SMatthew G Knepley}
233351d3a41SMatthew G Knepleymodule-whatis "PETSc - Portable, Extensible Toolkit for Scientific Computation"
234351d3a41SMatthew G Knepley
235dd486775SJed Brownset petsc_dir   "%s"
236dd486775SJed Brownset petsc_arch  "%s"
237351d3a41SMatthew G Knepley
238dd486775SJed Brownsetenv PETSC_ARCH "$petsc_arch"
239dd486775SJed Brownsetenv PETSC_DIR "$petsc_dir"
240dd486775SJed Brownprepend-path PATH "%s"
24155d606a3SSatish Balay''' % (self.petscdir.version, installdir, installarch, installpath))
242351d3a41SMatthew G Knepley    fd.close()
243351d3a41SMatthew G Knepley    return
244351d3a41SMatthew G Knepley
245f8833479SBarry Smith  def Dump(self):
246f8833479SBarry Smith    ''' Actually put the values into the configuration files '''
247f8833479SBarry Smith    # eventually everything between -- should be gone
24817f368bcSBarry Smith    if self.mpi.usingMPIUni:
24917f368bcSBarry Smith      #
25017f368bcSBarry Smith      # Remove any MPI/MPICH include files that may have been put here by previous runs of ./configure
2517908f030SMatthew 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)
25217f368bcSBarry Smith
253b5f71184SBarry Smith    self.logPrintDivider()
254b5f71184SBarry Smith    # Test for compiler-specific macros that need to be defined.
255b5f71184SBarry Smith    if self.setCompilers.isCrayVector('CC', self.log):
256b5f71184SBarry Smith      self.addDefine('HAVE_CRAY_VECTOR','1')
257b5f71184SBarry Smith
258b5f71184SBarry Smith    if self.functions.haveFunction('gethostbyname') and self.functions.haveFunction('socket') and self.headers.haveHeader('netinet/in.h'):
259b5f71184SBarry Smith      self.addDefine('USE_SOCKET_VIEWER','1')
260b5f71184SBarry Smith      if self.checkCompile('#include <sys/socket.h>','setsockopt(0,SOL_SOCKET,SO_REUSEADDR,0,0)'):
261b5f71184SBarry Smith        self.addDefine('HAVE_SO_REUSEADDR','1')
262b5f71184SBarry Smith
263b5f71184SBarry Smith    self.logPrintDivider()
2645f27b2e0SBarry Smith    self.setCompilers.pushLanguage('C')
2655f27b2e0SBarry Smith    compiler = self.setCompilers.getCompiler()
266217fe27eSSatish Balay    if [s for s in ['mpicc','mpiicc'] if os.path.basename(compiler).find(s)>=0]:
2675f27b2e0SBarry Smith      try:
2685f27b2e0SBarry Smith        output   = self.executeShellCommand(compiler + ' -show', log = self.log)[0]
2695f27b2e0SBarry Smith        compiler = output.split(' ')[0]
270f424265bSStefano Zampini        self.addDefine('MPICC_SHOW','"'+output.strip().replace('\n','\\\\n').replace('"','')+'"')
2715f27b2e0SBarry Smith      except:
272c9872b61SBarry Smith        self.addDefine('MPICC_SHOW','"Unavailable"')
273c9872b61SBarry Smith    else:
274c9872b61SBarry Smith      self.addDefine('MPICC_SHOW','"Unavailable"')
2755f27b2e0SBarry Smith    self.setCompilers.popLanguage()
276f8833479SBarry Smith#-----------------------------------------------------------------------------------------------------
277f8833479SBarry Smith
278f8833479SBarry Smith    # Sometimes we need C compiler, even if built with C++
279f8833479SBarry Smith    self.setCompilers.pushLanguage('C')
280e4c30378SBarry Smith    # do not use getCompilerFlags() because that automatically includes the CPPFLAGS so one ends up with duplication flags in makefile usage
281e4c30378SBarry Smith    self.addMakeMacro('CC_FLAGS',self.setCompilers.CFLAGS)
282f8833479SBarry Smith    self.setCompilers.popLanguage()
283f8833479SBarry Smith
28434f774f6SJed Brown    # And sometimes we need a C++ compiler even when PETSc is built with C
28534f774f6SJed Brown    if hasattr(self.compilers, 'CXX'):
28634f774f6SJed Brown      self.setCompilers.pushLanguage('Cxx')
28729921a8fSScott Kruger      self.addDefine('HAVE_CXX','1')
2880b119762SSatish Balay      self.addMakeMacro('CXXPP_FLAGS',self.setCompilers.CXXPPFLAGS)
289e4c30378SBarry Smith      # do not use getCompilerFlags() because that automatically includes the CXXPPFLAGS so one ends up with duplication flags in makefile usage
290e4c30378SBarry Smith      self.addMakeMacro('CXX_FLAGS',self.setCompilers.CXXFLAGS+' '+self.setCompilers.CXX_CXXFLAGS)
2912f4326f3SSatish Balay      cxx_linker = self.setCompilers.getLinker()
2922f4326f3SSatish Balay      self.addMakeMacro('CXX_LINKER',cxx_linker)
2932f4326f3SSatish Balay      self.addMakeMacro('CXX_LINKER_FLAGS',self.setCompilers.getLinkerFlags())
29434f774f6SJed Brown      self.setCompilers.popLanguage()
29534f774f6SJed Brown
296f8833479SBarry Smith    # C preprocessor values
2971315f054SBarry Smith    self.addMakeMacro('CPP_FLAGS',self.setCompilers.CPPFLAGS)
298f8833479SBarry Smith
299f8833479SBarry Smith    # compiler values
300f8833479SBarry Smith    self.setCompilers.pushLanguage(self.languages.clanguage)
301f8833479SBarry Smith    self.addMakeMacro('PCC',self.setCompilers.getCompiler())
302e4c30378SBarry Smith    # do not use getCompilerFlags() because that automatically includes the preprocessor flags so one ends up with duplication flags in makefile usage
303e4c30378SBarry Smith    if self.languages.clanguage == 'C':
304e4c30378SBarry Smith      self.addMakeMacro('PCC_FLAGS','$(CC_FLAGS)')
305e4c30378SBarry Smith    else:
306e4c30378SBarry Smith      self.addMakeMacro('PCC_FLAGS','$(CXX_FLAGS)')
307f8833479SBarry Smith    self.setCompilers.popLanguage()
308f8833479SBarry Smith    # .o or .obj
309f8833479SBarry Smith    self.addMakeMacro('CC_SUFFIX','o')
310f8833479SBarry Smith
311f8833479SBarry Smith    # executable linker values
312f8833479SBarry Smith    self.setCompilers.pushLanguage(self.languages.clanguage)
313f8833479SBarry Smith    pcc_linker = self.setCompilers.getLinker()
314f8833479SBarry Smith    self.addMakeMacro('PCC_LINKER',pcc_linker)
315c84a332bSSatish Balay    self.addMakeMacro('PCC_LINKER_FLAGS',self.setCompilers.getLinkerFlags())
316f8833479SBarry Smith    self.setCompilers.popLanguage()
317f8833479SBarry Smith    # '' for Unix, .exe for Windows
318f8833479SBarry Smith    self.addMakeMacro('CC_LINKER_SUFFIX','')
319f8833479SBarry Smith
320f8833479SBarry Smith    if hasattr(self.compilers, 'FC'):
321cb297985SSatish Balay      if self.framework.argDB['with-fortran-bindings']:
322257f4e5aSSatish Balay        if not self.fortran.fortranIsF90:
323257f4e5aSSatish Balay          raise RuntimeError('Error! Fortran compiler "'+self.compilers.FC+'" does not support F90! PETSc fortran bindings require a F90 compiler')
324cb297985SSatish Balay        self.addDefine('HAVE_FORTRAN','1')
325f8833479SBarry Smith      self.setCompilers.pushLanguage('FC')
326f8833479SBarry Smith      # need FPPFLAGS in config/setCompilers
3270b119762SSatish Balay      self.addMakeMacro('FPP_FLAGS',self.setCompilers.FPPFLAGS)
328f8833479SBarry Smith
329f8833479SBarry Smith      # compiler values
330f8833479SBarry Smith      self.addMakeMacro('FC_FLAGS',self.setCompilers.getCompilerFlags())
331f8833479SBarry Smith      self.setCompilers.popLanguage()
332f8833479SBarry Smith      # .o or .obj
333f8833479SBarry Smith      self.addMakeMacro('FC_SUFFIX','o')
334f8833479SBarry Smith
335f8833479SBarry Smith      # executable linker values
336f8833479SBarry Smith      self.setCompilers.pushLanguage('FC')
337a9acdec7SBarry Smith      self.addMakeMacro('FC_LINKER',self.setCompilers.getLinker())
3386d53d35eSSatish Balay      self.addMakeMacro('FC_LINKER_FLAGS',self.setCompilers.getLinkerFlags())
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('\"','\\"'))
490a0c7f9aaSSamuel Khuvis
491a0c7f9aaSSamuel Khuvis    if self.framework.argDB['with-tau-perfstubs']:
492a0c7f9aaSSamuel Khuvis      self.addDefine('HAVE_TAU_PERFSTUBS',1)
493f8833479SBarry Smith    return
494f8833479SBarry Smith
495f8833479SBarry Smith  def dumpConfigInfo(self):
496f8833479SBarry Smith    import time
497c6ef1b5bSJed Brown    fd = open(os.path.join(self.arch.arch,'include','petscconfiginfo.h'),'w')
498dc25a686SPierre Jolivet    fd.write('static const char *petscconfigureoptions = "'+self.framework.getOptionsString(['configModules', 'optionsModule']).replace('\"','\\"').replace('\\ ','\\\\ ')+'";\n')
499f8833479SBarry Smith    fd.close()
500f8833479SBarry Smith    return
501f8833479SBarry Smith
5022a4161d9SMatthew G Knepley  def dumpMachineInfo(self):
5032a4161d9SMatthew G Knepley    import platform
504a970bd74SBernhard M. Wiedemann    import datetime
5052a4161d9SMatthew G Knepley    import time
50640373944SSatish Balay    import script
507ca77dbeeSGeoffrey Irving    def escape(s):
508e08ecd42SSatish Balay      return s.replace('"',r'\"').replace(r'\ ',r'\\ ') # novermin
509c6ef1b5bSJed Brown    fd = open(os.path.join(self.arch.arch,'include','petscmachineinfo.h'),'w')
5102a4161d9SMatthew G Knepley    fd.write('static const char *petscmachineinfo = \"\\n\"\n')
5112a4161d9SMatthew G Knepley    fd.write('\"-----------------------------------------\\n\"\n')
512a970bd74SBernhard M. Wiedemann    buildhost = platform.node()
513a970bd74SBernhard M. Wiedemann    if os.environ.get('SOURCE_DATE_EPOCH'):
514a970bd74SBernhard M. Wiedemann      buildhost = "reproducible"
515a970bd74SBernhard M. Wiedemann    buildtime = datetime.datetime.utcfromtimestamp(int(os.environ.get('SOURCE_DATE_EPOCH', time.time())))
516a970bd74SBernhard M. Wiedemann    fd.write('\"Libraries compiled on %s on %s \\n\"\n' % (buildtime, buildhost))
51760acdfe7SSatish Balay    fd.write('\"Machine characteristics: %s\\n\"\n' % (platform.platform()))
5185188cb68SSatish Balay    fd.write('\"Using PETSc directory: %s\\n\"\n' % (escape(self.installdir.petscDir)))
5195188cb68SSatish Balay    fd.write('\"Using PETSc arch: %s\\n\"\n' % (escape(self.installdir.petscArch)))
520cdec380aSBarry Smith    fd.write('\"-----------------------------------------\\n\";\n')
5212a4161d9SMatthew G Knepley    fd.write('static const char *petsccompilerinfo = \"\\n\"\n')
5222a4161d9SMatthew G Knepley    self.setCompilers.pushLanguage(self.languages.clanguage)
5235f27b2e0SBarry Smith    fd.write('\"Using C compiler: %s %s \\n\"\n' % (escape(self.setCompilers.getCompiler()), escape(self.setCompilers.getCompilerFlags())))
5242a4161d9SMatthew G Knepley    self.setCompilers.popLanguage()
5258782282cSMatthew G Knepley    if hasattr(self.compilers, 'FC'):
5262a4161d9SMatthew G Knepley      self.setCompilers.pushLanguage('FC')
5275f27b2e0SBarry Smith      fd.write('\"Using Fortran compiler: %s %s  %s\\n\"\n' % (escape(self.setCompilers.getCompiler()), escape(self.setCompilers.getCompilerFlags()), escape(self.setCompilers.CPPFLAGS)))
5282a4161d9SMatthew G Knepley      self.setCompilers.popLanguage()
529cdec380aSBarry Smith    fd.write('\"-----------------------------------------\\n\";\n')
5302a4161d9SMatthew G Knepley    fd.write('static const char *petsccompilerflagsinfo = \"\\n\"\n')
5315a21677cSJed Brown    fd.write('\"Using include paths: %s\\n\"\n' % (escape(self.PETSC_CC_INCLUDES_INSTALL.replace('${PETSC_DIR}', self.installdir.petscDir))))
532cdec380aSBarry Smith    fd.write('\"-----------------------------------------\\n\";\n')
5332a4161d9SMatthew G Knepley    fd.write('static const char *petsclinkerinfo = \"\\n\"\n')
5342a4161d9SMatthew G Knepley    self.setCompilers.pushLanguage(self.languages.clanguage)
535ca77dbeeSGeoffrey Irving    fd.write('\"Using C linker: %s\\n\"\n' % (escape(self.setCompilers.getLinker())))
5362a4161d9SMatthew G Knepley    self.setCompilers.popLanguage()
5378782282cSMatthew G Knepley    if hasattr(self.compilers, 'FC'):
5382a4161d9SMatthew G Knepley      self.setCompilers.pushLanguage('FC')
539ca77dbeeSGeoffrey Irving      fd.write('\"Using Fortran linker: %s\\n\"\n' % (escape(self.setCompilers.getLinker())))
5402a4161d9SMatthew G Knepley      self.setCompilers.popLanguage()
5415188cb68SSatish 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)))
542cdec380aSBarry Smith    fd.write('\"-----------------------------------------\\n\";\n')
5432a4161d9SMatthew G Knepley    fd.close()
5442a4161d9SMatthew G Knepley    return
545b2843cf1SBarry Smith
546b2843cf1SBarry Smith  def configurePrefetch(self):
547b2843cf1SBarry Smith    '''Sees if there are any prefetch functions supported'''
5487fca349cSMatthew G. Knepley    if config.setCompilers.Configure.isSolaris(self.log) or self.framework.argDB['with-ios'] or not self.framework.argDB['with-prefetch']:
54993f78423SSatish Balay      self.addDefine('Prefetch(a,b,c)', ' ')
55093f78423SSatish Balay      return
551ec284106SBarry Smith    self.pushLanguage(self.languages.clanguage)
55210699583SJed Brown    if self.checkLink('#include <xmmintrin.h>', 'void *v = 0;_mm_prefetch((const char*)v,_MM_HINT_NTA);\n'):
55350d8bf02SJed Brown      # The Intel Intrinsics manual [1] specifies the prototype
55450d8bf02SJed Brown      #
55550d8bf02SJed Brown      #   void _mm_prefetch(char const *a, int sel);
55650d8bf02SJed Brown      #
55750d8bf02SJed Brown      # but other vendors seem to insist on using subtly different
55850d8bf02SJed Brown      # prototypes, including void* for the pointer, and an enum for
55950d8bf02SJed Brown      # sel.  These are both reasonable changes, but negatively impact
56050d8bf02SJed Brown      # portability.
56150d8bf02SJed Brown      #
562a8d69d7bSBarry Smith      # [1] https://software.intel.com/file/6373
56350d8bf02SJed Brown      self.addDefine('HAVE_XMMINTRIN_H', 1)
56450d8bf02SJed Brown      self.addDefine('Prefetch(a,b,c)', '_mm_prefetch((const char*)(a),(c))')
56550d8bf02SJed Brown      self.addDefine('PREFETCH_HINT_NTA', '_MM_HINT_NTA')
56650d8bf02SJed Brown      self.addDefine('PREFETCH_HINT_T0',  '_MM_HINT_T0')
56750d8bf02SJed Brown      self.addDefine('PREFETCH_HINT_T1',  '_MM_HINT_T1')
56850d8bf02SJed Brown      self.addDefine('PREFETCH_HINT_T2',  '_MM_HINT_T2')
56950d8bf02SJed Brown    elif self.checkLink('#include <xmmintrin.h>', 'void *v = 0;_mm_prefetch(v,_MM_HINT_NTA);\n'):
57050d8bf02SJed Brown      self.addDefine('HAVE_XMMINTRIN_H', 1)
57150d8bf02SJed Brown      self.addDefine('Prefetch(a,b,c)', '_mm_prefetch((const void*)(a),(c))')
57250d8bf02SJed Brown      self.addDefine('PREFETCH_HINT_NTA', '_MM_HINT_NTA')
57350d8bf02SJed Brown      self.addDefine('PREFETCH_HINT_T0',  '_MM_HINT_T0')
57450d8bf02SJed Brown      self.addDefine('PREFETCH_HINT_T1',  '_MM_HINT_T1')
57550d8bf02SJed Brown      self.addDefine('PREFETCH_HINT_T2',  '_MM_HINT_T2')
57610699583SJed Brown    elif self.checkLink('', 'void *v = 0;__builtin_prefetch(v,0,0);\n'):
57710699583SJed Brown      # From GCC docs: void __builtin_prefetch(const void *addr,int rw,int locality)
57810699583SJed Brown      #
57910699583SJed Brown      #   The value of rw is a compile-time constant one or zero; one
58010699583SJed Brown      #   means that the prefetch is preparing for a write to the memory
58110699583SJed Brown      #   address and zero, the default, means that the prefetch is
58210699583SJed Brown      #   preparing for a read. The value locality must be a compile-time
58310699583SJed Brown      #   constant integer between zero and three. A value of zero means
58410699583SJed Brown      #   that the data has no temporal locality, so it need not be left
58510699583SJed Brown      #   in the cache after the access. A value of three means that the
58610699583SJed Brown      #   data has a high degree of temporal locality and should be left
58710699583SJed Brown      #   in all levels of cache possible. Values of one and two mean,
58810699583SJed Brown      #   respectively, a low or moderate degree of temporal locality.
58910699583SJed Brown      #
59010699583SJed Brown      # Here we adopt Intel's x86/x86-64 naming scheme for the locality
59110699583SJed Brown      # hints.  Using macros for these values in necessary since some
59210699583SJed Brown      # compilers require an enum.
59310699583SJed Brown      self.addDefine('Prefetch(a,b,c)', '__builtin_prefetch((a),(b),(c))')
59410699583SJed Brown      self.addDefine('PREFETCH_HINT_NTA', '0')
59510699583SJed Brown      self.addDefine('PREFETCH_HINT_T0',  '3')
59610699583SJed Brown      self.addDefine('PREFETCH_HINT_T1',  '2')
59710699583SJed Brown      self.addDefine('PREFETCH_HINT_T2',  '1')
598b2843cf1SBarry Smith    else:
599b2843cf1SBarry Smith      self.addDefine('Prefetch(a,b,c)', ' ')
6007d490b44SBarry Smith    self.popLanguage()
601b2843cf1SBarry Smith
60249fe22e6SSatish Balay  def delGenFiles(self):
60349fe22e6SSatish Balay    '''Delete generated files'''
60449fe22e6SSatish Balay    delfile = os.path.join(self.arch.arch,'lib','petsc','conf','files')
60549fe22e6SSatish Balay    try:
60649fe22e6SSatish Balay      os.unlink(delfile)
60749fe22e6SSatish Balay    except: pass
60849fe22e6SSatish Balay
60909bc878fSSatish Balay  def configureAtoll(self):
61009bc878fSSatish Balay    '''Checks if atoll exists'''
61122164b4cSPierre Jolivet    if self.checkLink('#define _POSIX_C_SOURCE 200112L\n#include <stdlib.h>','long v = atoll("25");\n(void)v') or self.checkLink ('#include <stdlib.h>','long v = atoll("25");\n(void)v'):
61209bc878fSSatish Balay       self.addDefine('HAVE_ATOLL', '1')
61309bc878fSSatish Balay
6142400fdedSBarry Smith  def configureUnused(self):
6152400fdedSBarry Smith    '''Sees if __attribute((unused)) is supported'''
6161adaff47SSean Farley    if self.framework.argDB['with-ios']:
6172400fdedSBarry Smith      self.addDefine('UNUSED', ' ')
6182400fdedSBarry Smith      return
6192400fdedSBarry Smith    self.pushLanguage(self.languages.clanguage)
62022164b4cSPierre Jolivet    if self.checkLink('__attribute((unused)) static int myfunc(__attribute((unused)) void *name){ return 1;}', 'int i = 0;\nint j = myfunc(&i);\n(void)j;\ntypedef void* atype;\n__attribute((unused))  atype a'):
6212400fdedSBarry Smith      self.addDefine('UNUSED', '__attribute((unused))')
6222400fdedSBarry Smith    else:
6232400fdedSBarry Smith      self.addDefine('UNUSED', ' ')
6242400fdedSBarry Smith    self.popLanguage()
6252400fdedSBarry Smith
62698ed35c3SBarry Smith  def configureIsatty(self):
62798ed35c3SBarry Smith    '''Check if the Unix C function isatty() works correctly
62898ed35c3SBarry Smith       Actually just assumes it does not work correctly on batch systems'''
62998ed35c3SBarry Smith    if not self.framework.argDB['with-batch']:
63098ed35c3SBarry Smith      self.addDefine('USE_ISATTY',1)
63198ed35c3SBarry Smith
6321ef8df7fSJed Brown  def configureDeprecated(self):
6331ef8df7fSJed Brown    '''Check if __attribute((deprecated)) is supported'''
6347efe37a1SJacob Faibussowitsch    def checkDeprecated(macro_base, src, is_intel):
6357efe37a1SJacob Faibussowitsch      '''
6367efe37a1SJacob Faibussowitsch      run through the various attribute deprecated combinations and define MACRO_BAS(why) to the result
6377efe37a1SJacob Faibussowitsch      it if it compiles.
6387efe37a1SJacob Faibussowitsch
6397efe37a1SJacob Faibussowitsch      If none of the combos work, defines MACRO_BASE(why) as empty
6407efe37a1SJacob Faibussowitsch      '''
6417efe37a1SJacob Faibussowitsch      full_macro_name = macro_base + '(why)'
6427efe37a1SJacob Faibussowitsch      for prefix in ('__attribute__', '__attribute','__declspec'):
6437efe37a1SJacob Faibussowitsch        if prefix == '__declspec':
6447efe37a1SJacob Faibussowitsch          # declspec does not have an extra set of brackets around the arguments
6457efe37a1SJacob Faibussowitsch          attr_bodies = ('deprecated(why)', 'deprecated')
6461ef8df7fSJed Brown        else:
6477efe37a1SJacob Faibussowitsch          attr_bodies = ('(deprecated(why))', '(deprecated)')
6487efe37a1SJacob Faibussowitsch
6497efe37a1SJacob Faibussowitsch        for attr_body in attr_bodies:
6507efe37a1SJacob Faibussowitsch          attr_def = '{}({})'.format(prefix, attr_body)
6517efe37a1SJacob Faibussowitsch          test_src = '\n'.join((
6527efe37a1SJacob Faibussowitsch            '#define {} {}'.format(full_macro_name, attr_def),
6537efe37a1SJacob Faibussowitsch            src.format(macro_base + '("asdasdadsasd")')
6547efe37a1SJacob Faibussowitsch          ))
6557efe37a1SJacob Faibussowitsch          if self.checkCompile(test_src, ''):
6567efe37a1SJacob Faibussowitsch            self.logPrint('configureDeprecated: \'{}\' appears to work'.format(attr_def))
6577efe37a1SJacob Faibussowitsch            if is_intel and '(why)' in attr_body:
6587efe37a1SJacob 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))
6597efe37a1SJacob Faibussowitsch              self.logPrint('*** WE WILL THEREFORE REJECT \'{}\' AND CONTINUE TESTING ***'.format(attr_def))
6607efe37a1SJacob Faibussowitsch              continue
6617efe37a1SJacob Faibussowitsch            self.addDefine(full_macro_name, attr_def)
6627efe37a1SJacob Faibussowitsch            return
6637efe37a1SJacob Faibussowitsch
6647efe37a1SJacob Faibussowitsch      self.addDefine(full_macro_name, ' ')
6657efe37a1SJacob Faibussowitsch      return
6667efe37a1SJacob Faibussowitsch
6677efe37a1SJacob Faibussowitsch    lang = self.languages.clanguage
6687efe37a1SJacob Faibussowitsch    with self.Language(lang):
6697efe37a1SJacob Faibussowitsch      is_intel = self.setCompilers.isIntel(self.getCompiler(lang=lang), self.log)
6707efe37a1SJacob Faibussowitsch      checkDeprecated('DEPRECATED_FUNCTION', '{} int myfunc(void) {{ return 1; }}', is_intel)
6717efe37a1SJacob Faibussowitsch      checkDeprecated('DEPRECATED_TYPEDEF', 'typedef int my_int {};', is_intel)
6727efe37a1SJacob Faibussowitsch      checkDeprecated('DEPRECATED_ENUM', 'enum E {{ oldval {}, newval }};', is_intel)
6737efe37a1SJacob Faibussowitsch      # I was unable to make a CPP macro that takes the old and new values as separate
6747efe37a1SJacob Faibussowitsch      # arguments and builds the message needed by _Pragma hence the deprecation message is
6757efe37a1SJacob Faibussowitsch      # handled as it is
67605de396fSBarry Smith      if self.checkCompile('#define TEST _Pragma("GCC warning \"Testing _Pragma\"") value'):
67705de396fSBarry Smith        self.addDefine('DEPRECATED_MACRO(why)', '_Pragma(why)')
67805de396fSBarry Smith      else:
67905de396fSBarry Smith        self.addDefine('DEPRECATED_MACRO(why)', ' ')
6801ef8df7fSJed Brown
68118f41590SBarry Smith  def configureAlign(self):
6827b7fc14bSLisandro Dalcin    '''Check if __attribute(aligned) is supported'''
6837b7fc14bSLisandro Dalcin    code = '''\
684752d89a4SSatish Balaystruct mystruct {int myint;} __attribute((aligned(16)));
6857b7fc14bSLisandro Dalcinchar assert_aligned[(sizeof(struct mystruct)==16)*2-1];
686752d89a4SSatish Balay'''
687752d89a4SSatish Balay    self.pushLanguage(self.languages.clanguage)
6887b7fc14bSLisandro Dalcin    if self.checkCompile(code):
689752d89a4SSatish Balay      self.addDefine('ATTRIBUTEALIGNED(size)', '__attribute((aligned(size)))')
690752d89a4SSatish Balay      self.addDefine('HAVE_ATTRIBUTEALIGNED', 1)
691752d89a4SSatish Balay    else:
6927b7fc14bSLisandro Dalcin      self.framework.logPrint('Incorrect attribute(aligned)')
693752d89a4SSatish Balay      self.addDefine('ATTRIBUTEALIGNED(size)', ' ')
6947b7fc14bSLisandro Dalcin    self.popLanguage()
695752d89a4SSatish Balay    return
69618f41590SBarry Smith
6979800092aSJed Brown  def configureExpect(self):
6989800092aSJed Brown    '''Sees if the __builtin_expect directive is supported'''
6999800092aSJed Brown    self.pushLanguage(self.languages.clanguage)
7009800092aSJed Brown    if self.checkLink('', 'if (__builtin_expect(0,1)) return 1;'):
7019800092aSJed Brown      self.addDefine('HAVE_BUILTIN_EXPECT', 1)
7029800092aSJed Brown    self.popLanguage()
7039800092aSJed Brown
70453c77d0aSJed Brown  def configureFunctionName(self):
705fbfcfee5SBarry Smith    '''Sees if the compiler supports __func__ or a variant.'''
7061ec50b02SJed Brown    def getFunctionName(lang):
707fbfcfee5SBarry Smith      name = '"unknown"'
7081ec50b02SJed Brown      self.pushLanguage(lang)
709b6ff4c76SKarl Rupp      for fname in ['__func__','__FUNCTION__','__extension__ __func__']:
7100117e5a1SSatish Balay        code = "if ("+fname+"[0] != 'm') return 1;"
7110117e5a1SSatish Balay        if self.checkCompile('',code) and self.checkLink('',code):
7120117e5a1SSatish Balay          name = fname
7130117e5a1SSatish Balay          break
7141ec50b02SJed Brown      self.popLanguage()
7151ec50b02SJed Brown      return name
7161ec50b02SJed Brown    langs = []
717628773c9SSatish Balay
718628773c9SSatish Balay    self.addDefine('FUNCTION_NAME_C', getFunctionName('C'))
7195f6e5f85SSatish Balay    if hasattr(self.compilers, 'CXX'):
720628773c9SSatish Balay      self.addDefine('FUNCTION_NAME_CXX', getFunctionName('Cxx'))
72153c77d0aSJed Brown
722753ebd1dSJed Brown  def configureIntptrt(self):
723f18a5f7eSJacob Faibussowitsch    '''Determine what to use for uintptr_t and intptr_t'''
724753ebd1dSJed Brown    def staticAssertSizeMatchesVoidStar(inc,typename):
725753ebd1dSJed Brown      # The declaration is an error if either array size is negative.
726753ebd1dSJed 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
727d26187a0SJed Brown      return self.checkCompile(inc, ('#define STATIC_ASSERT(cond) char negative_length_if_false[2*(!!(cond))-1]\n'
728979939cdSSatish Balay                                     + 'STATIC_ASSERT(sizeof(void*) == sizeof(%s));'%typename))
729f18a5f7eSJacob Faibussowitsch
730f18a5f7eSJacob Faibussowitsch    def generate_uintptr_guesses():
731f18a5f7eSJacob Faibussowitsch      for suff in ('max', '64', '32', '16'):
732f18a5f7eSJacob Faibussowitsch        yield '#include <stdint.h>', 'uint{}_t'.format(suff), 'PRIx{}'.format(suff.upper())
733f18a5f7eSJacob Faibussowitsch      yield '#include <stdlib.h>\n#include <string.h>', 'size_t', 'zx'
734f18a5f7eSJacob Faibussowitsch      yield '', 'unsigned long long', 'llx'
735f18a5f7eSJacob Faibussowitsch      yield '', 'unsigned long', 'lx'
736f18a5f7eSJacob Faibussowitsch      yield '', 'unsigned', 'x'
737f18a5f7eSJacob Faibussowitsch
738f18a5f7eSJacob Faibussowitsch    def generate_intptr_guesses():
739f18a5f7eSJacob Faibussowitsch      for suff in ('max', '64', '32', '16'):
740f18a5f7eSJacob Faibussowitsch        yield '#include <stdint.h>', 'int{}_t'.format(suff), 'PRIx{}'.format(suff.upper())
741f18a5f7eSJacob Faibussowitsch      yield '', 'long long', 'llx'
742f18a5f7eSJacob Faibussowitsch      yield '', 'long', 'lx'
743f18a5f7eSJacob Faibussowitsch      yield '', 'int', 'x'
744f18a5f7eSJacob Faibussowitsch
745f18a5f7eSJacob Faibussowitsch    def check(default_typename, generator):
746f18a5f7eSJacob Faibussowitsch      macro_name = default_typename.upper()
747f18a5f7eSJacob Faibussowitsch      with self.Language(self.languages.clanguage):
748f18a5f7eSJacob Faibussowitsch        if self.checkCompile(
749f18a5f7eSJacob Faibussowitsch            '#include <stdint.h>',
750f18a5f7eSJacob Faibussowitsch            'int x; {type_name} i = ({type_name})&x; (void)i'.format(type_name=default_typename)
751f18a5f7eSJacob Faibussowitsch        ):
752f18a5f7eSJacob Faibussowitsch          typename     = default_typename
753f18a5f7eSJacob Faibussowitsch          print_format = 'PRIxPTR'
754d26187a0SJed Brown        else:
755f18a5f7eSJacob Faibussowitsch          for include, typename, print_format in generator():
756f18a5f7eSJacob Faibussowitsch            if staticAssertSizeMatchesVoidStar(include, typename):
757f18a5f7eSJacob Faibussowitsch              break
758f18a5f7eSJacob Faibussowitsch          else:
759f18a5f7eSJacob Faibussowitsch            raise RuntimeError('Could not find any {} type matching void*'.format(macro_name))
760f18a5f7eSJacob Faibussowitsch      self.addDefine(macro_name         , typename)
761f18a5f7eSJacob Faibussowitsch      self.addDefine(macro_name + '_FMT', '\"#\" ' + print_format)
762f18a5f7eSJacob Faibussowitsch      return
763f18a5f7eSJacob Faibussowitsch
764f18a5f7eSJacob Faibussowitsch    check('uintptr_t', generate_uintptr_guesses)
765f18a5f7eSJacob Faibussowitsch    check('intptr_t', generate_intptr_guesses)
766f18a5f7eSJacob Faibussowitsch    return
767753ebd1dSJed Brown
768ed938b00SJed Brown  def configureRTLDDefault(self):
7697b65ca21SBarry Smith    '''Check for dynamic library feature'''
770bfef2c86SBarry Smith    if self.checkCompile('#include <dlfcn.h>\n void *ptr =  RTLD_DEFAULT;'):
7719fb7294dSPierre Jolivet      self.addDefine('HAVE_RTLD_DEFAULT','1')
772f8833479SBarry Smith    return
773f8833479SBarry Smith
774f8833479SBarry Smith  def configureSolaris(self):
775f8833479SBarry Smith    '''Solaris specific stuff'''
776f8833479SBarry Smith    if os.path.isdir(os.path.join('/usr','ucblib')):
777f8833479SBarry Smith      try:
778f8833479SBarry Smith        flag = getattr(self.setCompilers, self.language[-1]+'SharedLinkerFlag')
779f8833479SBarry Smith      except AttributeError:
780f8833479SBarry Smith        flag = None
781f8833479SBarry Smith      if flag is None:
782f8833479SBarry Smith        self.compilers.LIBS += ' -L/usr/ucblib'
783f8833479SBarry Smith      else:
784f8833479SBarry Smith        self.compilers.LIBS += ' '+flag+'/usr/ucblib'
785f8833479SBarry Smith    return
786f8833479SBarry Smith
7870f64ec89SBarry Smith  def configureDarwin(self):
7880f64ec89SBarry Smith    '''Log brew configuration for Apple systems'''
7890f64ec89SBarry Smith    try:
7900f64ec89SBarry Smith      self.executeShellCommand(['brew', 'config'], log = self.log)
7910f64ec89SBarry Smith      self.executeShellCommand(['brew', 'info', 'gcc'], log = self.log)
7920f64ec89SBarry Smith    except:
7930f64ec89SBarry Smith      pass
7940f64ec89SBarry Smith    return
7950f64ec89SBarry Smith
796f8833479SBarry Smith  def configureLinux(self):
797f8833479SBarry Smith    '''Linux specific stuff'''
7989f15855cSMatthew G Knepley    # TODO: Test for this by mallocing an odd number of floats and checking the address
799f8833479SBarry Smith    self.addDefine('HAVE_DOUBLE_ALIGN_MALLOC', 1)
800f8833479SBarry Smith    return
801f8833479SBarry Smith
802f8833479SBarry Smith  def configureWin32(self):
803f8833479SBarry Smith    '''Win32 non-cygwin specific stuff'''
804f8833479SBarry Smith    kernel32=0
8054e8afd12SMosè Giordano    if self.libraries.add('Kernel32.lib','GetComputerName',prototype='#include <windows.h>', call='GetComputerName(NULL,NULL);'):
806f8833479SBarry Smith      self.addDefine('HAVE_WINDOWS_H',1)
807f8833479SBarry Smith      self.addDefine('HAVE_GETCOMPUTERNAME',1)
808f8833479SBarry Smith      kernel32=1
8094e8afd12SMosè Giordano    elif self.libraries.add('kernel32','GetComputerName',prototype='#include <windows.h>', call='GetComputerName(NULL,NULL);'):
810f8833479SBarry Smith      self.addDefine('HAVE_WINDOWS_H',1)
811f8833479SBarry Smith      self.addDefine('HAVE_GETCOMPUTERNAME',1)
812f8833479SBarry Smith      kernel32=1
813f8833479SBarry Smith    if kernel32:
814eed94e11SSatish Balay      if self.framework.argDB['with-windows-graphics']:
815eed94e11SSatish Balay        self.addDefine('USE_WINDOWS_GRAPHICS',1)
8164e8afd12SMosè Giordano      if self.checkLink('#include <windows.h>','LoadLibrary(0)'):
817f8833479SBarry Smith        self.addDefine('HAVE_LOADLIBRARY',1)
8184e8afd12SMosè Giordano      if self.checkLink('#include <windows.h>','GetProcAddress(0,0)'):
819b50f6d9eSLisandro Dalcin        self.addDefine('HAVE_GETPROCADDRESS',1)
8204e8afd12SMosè Giordano      if self.checkLink('#include <windows.h>','FreeLibrary(0)'):
821b50f6d9eSLisandro Dalcin        self.addDefine('HAVE_FREELIBRARY',1)
8224e8afd12SMosè Giordano      if self.checkLink('#include <windows.h>','GetLastError()'):
823a21658a3SLisandro Dalcin        self.addDefine('HAVE_GETLASTERROR',1)
8244e8afd12SMosè Giordano      if self.checkLink('#include <windows.h>','SetLastError(0)'):
825a21658a3SLisandro Dalcin        self.addDefine('HAVE_SETLASTERROR',1)
8264e8afd12SMosè Giordano      if self.checkLink('#include <windows.h>\n','QueryPerformanceCounter(0);\n'):
827bea725cfSBarry Smith        self.addDefine('USE_MICROSOFT_TIME',1)
8284e8afd12SMosè Giordano    if self.libraries.add('Advapi32.lib','GetUserName',prototype='#include <windows.h>', call='GetUserName(NULL,NULL);'):
829f8833479SBarry Smith      self.addDefine('HAVE_GET_USER_NAME',1)
8304e8afd12SMosè Giordano    elif self.libraries.add('advapi32','GetUserName',prototype='#include <windows.h>', call='GetUserName(NULL,NULL);'):
831f8833479SBarry Smith      self.addDefine('HAVE_GET_USER_NAME',1)
832f8833479SBarry Smith
8334e8afd12SMosè Giordano    if not self.libraries.add('User32.lib','GetDC',prototype='#include <windows.h>',call='GetDC(0);'):
8344e8afd12SMosè Giordano      self.libraries.add('user32','GetDC',prototype='#include <windows.h>',call='GetDC(0);')
8354e8afd12SMosè Giordano    if not self.libraries.add('Gdi32.lib','CreateCompatibleDC',prototype='#include <windows.h>',call='CreateCompatibleDC(0);'):
8364e8afd12SMosè Giordano      self.libraries.add('gdi32','CreateCompatibleDC',prototype='#include <windows.h>',call='CreateCompatibleDC(0);')
837f8833479SBarry Smith
838f8833479SBarry Smith    self.types.check('int32_t', 'int')
83922164b4cSPierre Jolivet    if not self.checkCompile('#include <sys/types.h>\n','uid_t u;\n(void)u'):
840f8833479SBarry Smith      self.addTypedef('int', 'uid_t')
841f8833479SBarry Smith      self.addTypedef('int', 'gid_t')
84222164b4cSPierre Jolivet    if not self.checkLink('#if defined(PETSC_HAVE_UNISTD_H)\n#include <unistd.h>\n#endif\n','int a=R_OK;\n(void)a'):
843f8833479SBarry Smith      self.framework.addDefine('R_OK', '04')
844f8833479SBarry Smith      self.framework.addDefine('W_OK', '02')
845f8833479SBarry Smith      self.framework.addDefine('X_OK', '01')
846f8833479SBarry Smith    if not self.checkLink('#include <sys/stat.h>\n','int a=0;\nif (S_ISDIR(a)){}\n'):
847f8833479SBarry Smith      self.framework.addDefine('S_ISREG(a)', '(((a)&_S_IFMT) == _S_IFREG)')
848f8833479SBarry Smith      self.framework.addDefine('S_ISDIR(a)', '(((a)&_S_IFMT) == _S_IFDIR)')
8494e8afd12SMosè Giordano    if self.checkCompile('#include <windows.h>\n','LARGE_INTEGER a;\nDWORD b=a.u.HighPart;\n'):
850f8833479SBarry Smith      self.addDefine('HAVE_LARGE_INTEGER_U',1)
851f8833479SBarry Smith
852f8833479SBarry Smith    # Windows requires a Binary file creation flag when creating/opening binary files.  Is a better test in order?
8534e8afd12SMosè Giordano    if self.checkCompile('#include <windows.h>\n#include <fcntl.h>\n', 'int flags = O_BINARY;'):
854f8833479SBarry Smith      self.addDefine('HAVE_O_BINARY',1)
855f8833479SBarry Smith
856f8833479SBarry Smith    if self.compilers.CC.find('win32fe') >= 0:
857ad4212abSSatish Balay      self.addDefine('HAVE_WINDOWS_COMPILERS',1)
858f8833479SBarry Smith      self.addDefine('DIR_SEPARATOR','\'\\\\\'')
859f8833479SBarry Smith      self.addDefine('REPLACE_DIR_SEPARATOR','\'/\'')
860f8833479SBarry Smith      self.addDefine('CANNOT_START_DEBUGGER',1)
8615188cb68SSatish Balay      (petscdir,error,status) = self.executeShellCommand('cygpath -w '+self.installdir.petscDir, log = self.log)
86234531a4dSSatish Balay      self.addDefine('DIR','"'+petscdir.replace('\\','\\\\')+'"')
8635188cb68SSatish Balay      (petscdir,error,status) = self.executeShellCommand('cygpath -m '+self.installdir.petscDir, log = self.log)
864e433681fSSatish Balay      self.addMakeMacro('wPETSC_DIR',petscdir)
8654e00a515SSatish Balay      if self.dataFilesPath.datafilespath:
8664e00a515SSatish Balay        (datafilespath,error,status) = self.executeShellCommand('cygpath -m '+self.dataFilesPath.datafilespath, log = self.log)
8674e00a515SSatish Balay        self.addMakeMacro('DATAFILESPATH',datafilespath)
8684e00a515SSatish Balay
869f8833479SBarry Smith    else:
870f8833479SBarry Smith      self.addDefine('REPLACE_DIR_SEPARATOR','\'\\\\\'')
871f8833479SBarry Smith      self.addDefine('DIR_SEPARATOR','\'/\'')
8725188cb68SSatish Balay      self.addDefine('DIR','"'+self.installdir.petscDir+'"')
8735188cb68SSatish Balay      self.addMakeMacro('wPETSC_DIR',self.installdir.petscDir)
8744e00a515SSatish Balay      if self.dataFilesPath.datafilespath:
8754e00a515SSatish Balay        self.addMakeMacro('DATAFILESPATH',self.dataFilesPath.datafilespath)
8765188cb68SSatish Balay    self.addDefine('ARCH','"'+self.installdir.petscArch+'"')
877f8833479SBarry Smith    return
878f8833479SBarry Smith
879*9b81490aSJacob Faibussowitsch  def configureCoverageForLang(self, lang, extra_coverage_flags=None, extra_debug_flags=None):
8807ce81a4bSJacob Faibussowitsch    """
8817ce81a4bSJacob Faibussowitsch    Check that a compiler accepts code-coverage flags. If the compiler does accept code-coverage flags
8827ce81a4bSJacob Faibussowitsch    try to set debugging flags equivalent to -Og.
8837ce81a4bSJacob Faibussowitsch
8847ce81a4bSJacob Faibussowitsch    Arguments:
8857ce81a4bSJacob Faibussowitsch    - lang: the language to check the coverage flag for
8867ce81a4bSJacob Faibussowitsch    - extra_coverage_flags: a list of extra flags to use when checking the coverage flags
8877ce81a4bSJacob Faibussowitsch    - extra_debug_flags: a list of extra flags to try when setting debug flags
8887ce81a4bSJacob Faibussowitsch
8897ce81a4bSJacob Faibussowitsch    On success:
8907ce81a4bSJacob Faibussowitsch    - defines PETSC_USE_COVERAGE to 1
8917ce81a4bSJacob Faibussowitsch    """
892*9b81490aSJacob Faibussowitsch    try:
893*9b81490aSJacob Faibussowitsch      import inspect
894*9b81490aSJacob Faibussowitsch
895*9b81490aSJacob Faibussowitsch      FUNC_NAME = inspect.currentframe().f_code.co_name
896*9b81490aSJacob Faibussowitsch    except:
897*9b81490aSJacob Faibussowitsch      FUNC_NAME = 'Unknown'
898*9b81490aSJacob Faibussowitsch
8997ce81a4bSJacob Faibussowitsch    def log_print(msg, *args, **kwargs):
900*9b81490aSJacob Faibussowitsch      self.logPrint('{}(): {}'.format(FUNC_NAME, msg), *args, **kwargs)
9017ce81a4bSJacob Faibussowitsch      return
9027ce81a4bSJacob Faibussowitsch
9037ce81a4bSJacob Faibussowitsch    def quoted(string):
9047ce81a4bSJacob Faibussowitsch      return string.join(("'", "'"))
9057ce81a4bSJacob Faibussowitsch
9067ce81a4bSJacob Faibussowitsch    def make_flag_list(default, extra):
9077ce81a4bSJacob Faibussowitsch      ret = [default]
9087ce81a4bSJacob Faibussowitsch      if extra is not None:
9097ce81a4bSJacob Faibussowitsch        assert isinstance(extra, list)
9107ce81a4bSJacob Faibussowitsch        ret.extend(extra)
9117ce81a4bSJacob Faibussowitsch      return ret
9127ce81a4bSJacob Faibussowitsch
9137ce81a4bSJacob Faibussowitsch    log_print('Checking coverage flag for language {}'.format(lang))
914*9b81490aSJacob Faibussowitsch
915*9b81490aSJacob Faibussowitsch    compiler = self.getCompiler(lang=lang)
916*9b81490aSJacob Faibussowitsch    if self.setCompilers.isGNU(compiler, self.log):
917*9b81490aSJacob Faibussowitsch      compiler_kind = 'GNU'
918*9b81490aSJacob Faibussowitsch    elif self.setCompilers.isClang(compiler, self.log):
919*9b81490aSJacob Faibussowitsch      compiler_kind = 'clang'
920*9b81490aSJacob Faibussowitsch    else:
921*9b81490aSJacob Faibussowitsch      compiler_kind = ''
922*9b81490aSJacob Faibussowitsch
9237ce81a4bSJacob Faibussowitsch    if not self.argDB['with-coverage']:
9247ce81a4bSJacob Faibussowitsch      log_print('coverage was disabled from command line or default')
9257ce81a4bSJacob Faibussowitsch      return
9267ce81a4bSJacob Faibussowitsch
9277ce81a4bSJacob Faibussowitsch    # if not gnuish and we don't have a set of extra flags, bail
928*9b81490aSJacob Faibussowitsch    if compiler_kind not in {'GNU', 'clang'} and extra_coverage_flags is None:
9297ce81a4bSJacob 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)))
9307ce81a4bSJacob Faibussowitsch      return
9317ce81a4bSJacob Faibussowitsch
9327ce81a4bSJacob Faibussowitsch    coverage_flags = make_flag_list('--coverage', extra_coverage_flags)
9337ce81a4bSJacob Faibussowitsch    log_print('Checking set of coverage flags: {}'.format(coverage_flags))
9347ce81a4bSJacob Faibussowitsch
935*9b81490aSJacob Faibussowitsch    found = None
9367ce81a4bSJacob Faibussowitsch    with self.Language(lang):
9377ce81a4bSJacob Faibussowitsch      with self.setCompilers.Language(lang):
9387ce81a4bSJacob Faibussowitsch        for flag in coverage_flags:
939*9b81490aSJacob Faibussowitsch          # the linker also needs to see the coverage flag
940*9b81490aSJacob Faibussowitsch          with self.setCompilers.extraCompilerFlags([flag], compilerOnly=False) as skip_flags:
941*9b81490aSJacob Faibussowitsch            if not skip_flags and self.checkRun():
942*9b81490aSJacob Faibussowitsch              # flag was accepted
943*9b81490aSJacob Faibussowitsch              found = flag
9447ce81a4bSJacob Faibussowitsch              break
945*9b81490aSJacob Faibussowitsch
9467ce81a4bSJacob Faibussowitsch          log_print(
9477ce81a4bSJacob Faibussowitsch            'Compiler {} did not accept coverage flag {}'.format(quoted(compiler), quoted(flag))
9487ce81a4bSJacob Faibussowitsch          )
9497ce81a4bSJacob Faibussowitsch
950*9b81490aSJacob Faibussowitsch        if found is None:
951*9b81490aSJacob Faibussowitsch          log_print(
952*9b81490aSJacob Faibussowitsch            'Compiler {} did not accept ANY coverage flags: {}, bailing!'.format(
953*9b81490aSJacob Faibussowitsch              quoted(compiler), coverage_flags
954*9b81490aSJacob Faibussowitsch            )
955*9b81490aSJacob Faibussowitsch          )
9567ce81a4bSJacob Faibussowitsch          return
9577ce81a4bSJacob Faibussowitsch
958*9b81490aSJacob Faibussowitsch        # must do this exactly here since:
959*9b81490aSJacob Faibussowitsch        #
960*9b81490aSJacob Faibussowitsch        # 1. setCompilers.extraCompilerFlags() will reset the compiler flags on __exit__()
961*9b81490aSJacob Faibussowitsch        #    (so cannot do it in the loop)
962*9b81490aSJacob Faibussowitsch        # 2. we need to set the compiler flag while setCompilers.Language() is still in
963*9b81490aSJacob Faibussowitsch        #    effect (so cannot do it outside the with statements)
964*9b81490aSJacob Faibussowitsch        self.setCompilers.insertCompilerFlag(flag, False)
965*9b81490aSJacob Faibussowitsch
9667ce81a4bSJacob Faibussowitsch    if not self.functions.haveFunction('__gcov_dump'):
9677ce81a4bSJacob Faibussowitsch      self.functions.checkClassify(['__gcov_dump'])
9687ce81a4bSJacob Faibussowitsch
9697ce81a4bSJacob Faibussowitsch    # now check if we can override the optimization level. It is only kosher to do so if
9707ce81a4bSJacob Faibussowitsch    # the user did not explicitly set the optimization flags (via CFLAGS, CXXFLAGS,
9717ce81a4bSJacob Faibussowitsch    # CXXOPTFLAGS, etc). If they have done so, we sternly warn them about their lapse in
9727ce81a4bSJacob Faibussowitsch    # judgement
9737ce81a4bSJacob Faibussowitsch    with self.Language(lang):
9747ce81a4bSJacob Faibussowitsch      compiler_flags = self.getCompilerFlags()
9757ce81a4bSJacob Faibussowitsch
9767ce81a4bSJacob Faibussowitsch    user_set          = 0
9777ce81a4bSJacob Faibussowitsch    allowed_opt_flags = re.compile(r'|'.join((r'-O[01g]', r'-g[1-9]*')))
9787ce81a4bSJacob Faibussowitsch    for flagsname in [self.getCompilerFlagsName(lang), self.compilerFlags.getOptionalFlagsName(lang)]:
9797ce81a4bSJacob Faibussowitsch      if flagsname in self.argDB:
9807ce81a4bSJacob Faibussowitsch        opt_flags = [
9817ce81a4bSJacob Faibussowitsch          f for f in self.compilerFlags.findOptFlags(compiler_flags) if not allowed_opt_flags.match(f)
9827ce81a4bSJacob Faibussowitsch        ]
9837ce81a4bSJacob Faibussowitsch        if opt_flags:
9847ce81a4bSJacob 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)))
9857ce81a4bSJacob Faibussowitsch          user_set = 1
9867ce81a4bSJacob Faibussowitsch          break
9877ce81a4bSJacob Faibussowitsch
9887ce81a4bSJacob Faibussowitsch    if not user_set:
9897ce81a4bSJacob Faibussowitsch      debug_flags = make_flag_list('-Og', extra_debug_flags)
9907ce81a4bSJacob Faibussowitsch      with self.setCompilers.Language(lang):
9917ce81a4bSJacob Faibussowitsch        for flag in debug_flags:
9927ce81a4bSJacob Faibussowitsch          try:
9937ce81a4bSJacob Faibussowitsch            self.setCompilers.addCompilerFlag(flag)
9947ce81a4bSJacob Faibussowitsch          except RuntimeError:
9957ce81a4bSJacob Faibussowitsch            continue
9967ce81a4bSJacob Faibussowitsch          break
9977ce81a4bSJacob Faibussowitsch
9987ce81a4bSJacob Faibussowitsch    self.addDefine('USE_COVERAGE', 1)
9997ce81a4bSJacob Faibussowitsch    return
10007ce81a4bSJacob Faibussowitsch
1001*9b81490aSJacob Faibussowitsch  def configureCoverage(self):
1002*9b81490aSJacob Faibussowitsch    for LANG in ['C', 'Cxx', 'CUDA', 'HIP', 'SYCL', 'FC']:
1003*9b81490aSJacob Faibussowitsch      compilerName = LANG.upper() if LANG in {'Cxx', 'FC'} else LANG + 'C'
1004*9b81490aSJacob Faibussowitsch      if hasattr(self.setCompilers, compilerName):
1005*9b81490aSJacob Faibussowitsch        kwargs = {}
1006*9b81490aSJacob Faibussowitsch        if LANG in {'CUDA'}:
1007*9b81490aSJacob Faibussowitsch          # nvcc preprocesses the base file into a bunch of intermediate files, which are
1008*9b81490aSJacob Faibussowitsch          # then compiled by the host compiler. Why is this a problem?  Because the
1009*9b81490aSJacob Faibussowitsch          # generated coverage data is based on these preprocessed source files! So gcov
1010*9b81490aSJacob Faibussowitsch          # tries to read it later, but since its in the tmp directory it cannot. Thus we
1011*9b81490aSJacob Faibussowitsch          # need to keep them around (in a place we know about).
1012*9b81490aSJacob Faibussowitsch          nvcc_tmp_dir = os.path.join(self.petscdir.dir, self.arch.arch, 'nvcc_tmp')
1013*9b81490aSJacob Faibussowitsch          try:
1014*9b81490aSJacob Faibussowitsch            os.mkdir(nvcc_tmp_dir)
1015*9b81490aSJacob Faibussowitsch          except FileExistsError:
1016*9b81490aSJacob Faibussowitsch            pass
1017*9b81490aSJacob Faibussowitsch          kwargs['extra_coverage_flags'] = [
1018*9b81490aSJacob Faibussowitsch            '-Xcompiler --coverage -Xcompiler -fPIC --keep --keep-dir={}'.format(nvcc_tmp_dir)
1019*9b81490aSJacob Faibussowitsch          ]
1020*9b81490aSJacob Faibussowitsch          if self.kokkos.found:
1021*9b81490aSJacob Faibussowitsch            # yet again the kokkos nvcc_wrapper goes out of its way to be as useless as
1022*9b81490aSJacob Faibussowitsch            # possible. Its default arch (sm_35) is actually too low to compile kokkos,
1023*9b81490aSJacob Faibussowitsch            # for whatever reason this works if you dont use the --keep and --keep-dir
1024*9b81490aSJacob Faibussowitsch            # flags above.
1025*9b81490aSJacob Faibussowitsch            kwargs['extra_coverage_flags'].append('-arch=native')
1026*9b81490aSJacob Faibussowitsch            kwargs['extra_debug_flags'] = ['-Xcompiler -Og']
1027*9b81490aSJacob Faibussowitsch        self.executeTest(self.configureCoverageForLang, args=[LANG], kargs=kwargs)
1028*9b81490aSJacob Faibussowitsch    return
1029*9b81490aSJacob Faibussowitsch
10307ce81a4bSJacob Faibussowitsch  def configureCoverageExecutable(self):
10317ce81a4bSJacob Faibussowitsch    """
10327ce81a4bSJacob Faibussowitsch    Check that a code-coverage collecting tool exists and is on PATH.
10337ce81a4bSJacob Faibussowitsch
10347ce81a4bSJacob Faibussowitsch    On success:
10357ce81a4bSJacob Faibussowitsch    - Adds PETSC_COVERAGE_EXEC make macro containing the full path to the coverage tool executable.
10367ce81a4bSJacob Faibussowitsch
10377ce81a4bSJacob Faibussowitsch    Raises RuntimeError if:
10387ce81a4bSJacob Faibussowitsch    - User explicitly requests auto-detection of the coverage tool from command line, and this
10397ce81a4bSJacob Faibussowitsch      routine fails to guess the suitable tool name.
10407ce81a4bSJacob Faibussowitsch    - The routine fails to find the tool, and --with-coverage is true
10417ce81a4bSJacob Faibussowitsch    """
10427ce81a4bSJacob Faibussowitsch    def log_print(msg, *args, **kwargs):
10437ce81a4bSJacob Faibussowitsch      self.logPrint('checkCoverage: '+str(msg), *args, **kwargs)
10447ce81a4bSJacob Faibussowitsch      return
10457ce81a4bSJacob Faibussowitsch
10467ce81a4bSJacob Faibussowitsch    def quoted(string):
10477ce81a4bSJacob Faibussowitsch      return string.join(("'", "'"))
10487ce81a4bSJacob Faibussowitsch
10497ce81a4bSJacob Faibussowitsch    required         = bool(self.argDB['with-coverage'])
10507ce81a4bSJacob Faibussowitsch    arg_opt          = self.argDB['with-coverage-exec']
10517ce81a4bSJacob Faibussowitsch    use_default_path = True
10527ce81a4bSJacob Faibussowitsch    search_path      = ''
10537ce81a4bSJacob Faibussowitsch
10547ce81a4bSJacob Faibussowitsch    log_print('{} to find an executable'.format('REQUIRED' if required else 'NOT required'))
10557ce81a4bSJacob Faibussowitsch    if arg_opt in {'auto', 'default-auto', '1'}:
10567ce81a4bSJacob Faibussowitsch      # detect it based on the C language compiler, hopefully this does not clash!
1057095fb05fSJacob Faibussowitsch      lang     = self.setCompilers.languages.clanguage
1058095fb05fSJacob Faibussowitsch      compiler = self.getCompiler(lang=lang)
10597ce81a4bSJacob Faibussowitsch      log_print('User did not explicitly set coverage exec (got {}), trying to auto-detect based on compiler {}'.format(quoted(arg_opt), quoted(compiler)))
10607ce81a4bSJacob Faibussowitsch      if self.setCompilers.isGNU(compiler, self.log):
1061095fb05fSJacob Faibussowitsch        compiler_version_re = re.compile(r'[gG][cC\+\-]+[0-9]* \(.+\) (\d+)\.(\d+)\.(\d+)')
10627ce81a4bSJacob Faibussowitsch        exec_names          = ['gcov']
10637ce81a4bSJacob Faibussowitsch      elif self.setCompilers.isClang(compiler, self.log):
1064095fb05fSJacob Faibussowitsch        compiler_version_re = re.compile(r'clang version (\d+)\.(\d+)\.(\d+)')
10657ce81a4bSJacob Faibussowitsch        exec_names          = ['llvm-cov']
10667ce81a4bSJacob Faibussowitsch        if self.setCompilers.isDarwin(self.log):
10677ce81a4bSJacob Faibussowitsch          # macOS masquerades llvm-cov as just 'gcov', so we add this to the list in case
10687ce81a4bSJacob Faibussowitsch          # bare llvm-cov does not work
10697ce81a4bSJacob Faibussowitsch          exec_names.append('gcov')
10707ce81a4bSJacob Faibussowitsch      elif arg_opt == 'default-auto' and not required:
10717ce81a4bSJacob Faibussowitsch        # default-auto implies the user did not set it via command line!
10727ce81a4bSJacob 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)))
10737ce81a4bSJacob Faibussowitsch        return
10747ce81a4bSJacob Faibussowitsch      else:
10757ce81a4bSJacob Faibussowitsch        # implies 'auto' explicitly set by user, or we were required to find
10767ce81a4bSJacob Faibussowitsch        # something. either way we should error
10777ce81a4bSJacob Faibussowitsch        raise RuntimeError('Could not auto-detect coverage tool for {}, please set coverage tool name explicitly'.format(quoted(compiler)))
1078095fb05fSJacob Faibussowitsch
1079095fb05fSJacob Faibussowitsch      try:
1080095fb05fSJacob Faibussowitsch        compiler_version_str = self.compilerFlags.version[lang]
1081095fb05fSJacob Faibussowitsch      except KeyError:
1082095fb05fSJacob Faibussowitsch        compiler_version_str = 'Unknown'
1083095fb05fSJacob Faibussowitsch
1084095fb05fSJacob Faibussowitsch      log_print('Searching version string {} (for compiler {}) using pattern {}'.format(quoted(compiler_version_str), quoted(compiler), quoted(compiler_version_re.pattern)))
1085095fb05fSJacob Faibussowitsch      compiler_version = compiler_version_re.search(compiler_version_str)
1086095fb05fSJacob Faibussowitsch      if compiler_version is not None:
1087095fb05fSJacob Faibussowitsch        log_print('Found major = {}, minor = {}, patch = {}'.format(compiler_version.group(1), compiler_version.group(2), compiler_version.group(3)))
1088095fb05fSJacob Faibussowitsch        # form [llvm-cov-14, llvm-cov-14.0, llvm-cov, etc.]
1089095fb05fSJacob Faibussowitsch        cov_exec_name = exec_names[0]
1090095fb05fSJacob Faibussowitsch        exec_names    = [
1091095fb05fSJacob Faibussowitsch          # llvm-cov-14
1092095fb05fSJacob Faibussowitsch          '{}-{}'.format(cov_exec_name, compiler_version.group(1)),
1093095fb05fSJacob Faibussowitsch           # llvm-cov-14.0
1094095fb05fSJacob Faibussowitsch          '{}-{}.{}'.format(cov_exec_name, compiler_version.group(1), compiler_version.group(2))
1095095fb05fSJacob Faibussowitsch        ] + exec_names
10967ce81a4bSJacob Faibussowitsch    else:
10977ce81a4bSJacob Faibussowitsch      log_print('User explicitly set coverage exec as {}'.format(quoted(arg_opt)))
10987ce81a4bSJacob Faibussowitsch      par_dir = os.path.dirname(arg_opt)
10997ce81a4bSJacob Faibussowitsch      if os.path.exists(par_dir):
11007ce81a4bSJacob Faibussowitsch        # arg_opt is path-like, we should only search the provided directory when we go
11017ce81a4bSJacob Faibussowitsch        # looking for the tool
11027ce81a4bSJacob Faibussowitsch        use_default_path = False
11037ce81a4bSJacob Faibussowitsch        search_path      = par_dir
11047ce81a4bSJacob Faibussowitsch      exec_names = [arg_opt]
11057ce81a4bSJacob Faibussowitsch
11067ce81a4bSJacob Faibussowitsch    make_macro_name = 'PETSC_COVERAGE_EXEC'
11077ce81a4bSJacob Faibussowitsch    log_print('Checking for coverage tool(s):\n{}'.format('\n'.join('- '+t for t in exec_names)))
11087ce81a4bSJacob Faibussowitsch    found_exec = self.getExecutables(
11097ce81a4bSJacob Faibussowitsch      exec_names,
11107ce81a4bSJacob Faibussowitsch      path=search_path, getFullPath=True, useDefaultPath=use_default_path, resultName=make_macro_name
11117ce81a4bSJacob Faibussowitsch    )
11127ce81a4bSJacob Faibussowitsch
11137ce81a4bSJacob Faibussowitsch    if found_exec is None:
11147ce81a4bSJacob Faibussowitsch      # didn't find the coverage tool
11157ce81a4bSJacob Faibussowitsch      if required:
11167ce81a4bSJacob Faibussowitsch        raise RuntimeError('Coverage tool(s) {} could not be found. Please provide explicit path to coverage tool'.format(exec_names))
11177ce81a4bSJacob Faibussowitsch      return
11187ce81a4bSJacob Faibussowitsch
11197ce81a4bSJacob Faibussowitsch    found_exec_name = os.path.basename(found_exec)
11207ce81a4bSJacob Faibussowitsch    if 'llvm-cov' in found_exec_name and 'gcov' not in found_exec_name:
11217ce81a4bSJacob Faibussowitsch      # llvm-cov needs to be called as 'llvm-cov gcov' to work
11227ce81a4bSJacob Faibussowitsch      self.addMakeMacro(make_macro_name, found_exec + ' gcov')
11237ce81a4bSJacob Faibussowitsch    return
11247ce81a4bSJacob Faibussowitsch
1125689a5dfaSJacob Faibussowitsch  def configureStrictPetscErrorCode(self):
1126689a5dfaSJacob Faibussowitsch    """
1127689a5dfaSJacob Faibussowitsch    Enables or disables strict PetscErrorCode checking.
1128689a5dfaSJacob Faibussowitsch
1129689a5dfaSJacob Faibussowitsch    If --with-strict-petscerrorcode = 1:
1130689a5dfaSJacob Faibussowitsch    - defines PETSC_USE_STRICT_PETSCERRORCODE to 1
1131689a5dfaSJacob Faibussowitsch
1132689a5dfaSJacob Faibussowitsch    Else:
1133689a5dfaSJacob Faibussowitsch    - deletes any prior PETSC_USE_STRICT_PETSCERRORCODE definitions (if they exist)
1134689a5dfaSJacob Faibussowitsch    """
1135689a5dfaSJacob Faibussowitsch    define_name = 'USE_STRICT_PETSCERRORCODE'
1136689a5dfaSJacob Faibussowitsch    if self.argDB['with-strict-petscerrorcode']:
1137689a5dfaSJacob Faibussowitsch      self.addDefine(define_name, 1)
1138689a5dfaSJacob Faibussowitsch    else:
1139689a5dfaSJacob Faibussowitsch      # in case it was somehow added previously
1140689a5dfaSJacob Faibussowitsch      self.delDefine(define_name)
1141689a5dfaSJacob Faibussowitsch    return
1142689a5dfaSJacob Faibussowitsch
1143f8833479SBarry Smith#-----------------------------------------------------------------------------------------------------
1144b10d012aSSatish Balay  def configureCygwinBrokenPipe(self):
1145b10d012aSSatish Balay    '''Cygwin version <= 1.7.18 had issues with pipes and long commands invoked from gnu-make
1146b10d012aSSatish Balay    http://cygwin.com/ml/cygwin/2013-05/msg00340.html '''
11477fca349cSMatthew G. Knepley    if config.setCompilers.Configure.isCygwin(self.log):
1148b10d012aSSatish Balay      import platform
1149b10d012aSSatish Balay      import re
1150b10d012aSSatish Balay      r=re.compile("([0-9]+).([0-9]+).([0-9]+)")
1151b10d012aSSatish Balay      m=r.match(platform.release())
1152b10d012aSSatish Balay      major=int(m.group(1))
1153b10d012aSSatish Balay      minor=int(m.group(2))
1154b10d012aSSatish Balay      subminor=int(m.group(3))
1155b10d012aSSatish Balay      if ((major < 1) or (major == 1 and minor < 7) or (major == 1 and minor == 7 and subminor <= 18)):
1156b10d012aSSatish Balay        self.addMakeMacro('PETSC_CYGWIN_BROKEN_PIPE','1')
1157b10d012aSSatish Balay    return
1158b10d012aSSatish Balay
1159b10d012aSSatish Balay#-----------------------------------------------------------------------------------------------------
1160569865ddSSatish Balay  def configureDefaultArch(self):
1161af0996ceSBarry Smith    conffile = os.path.join('lib','petsc','conf', 'petscvariables')
1162569865ddSSatish Balay    if self.framework.argDB['with-default-arch']:
1163c6ef1b5bSJed Brown      fd = open(conffile, 'w')
1164569865ddSSatish Balay      fd.write('PETSC_ARCH='+self.arch.arch+'\n')
1165da93591fSBarry Smith      fd.write('PETSC_DIR='+self.petscdir.dir+'\n')
1166b9b902edSJed Brown      fd.write('include '+os.path.join('$(PETSC_DIR)','$(PETSC_ARCH)','lib','petsc','conf','petscvariables')+'\n')
1167569865ddSSatish Balay      fd.close()
1168569865ddSSatish Balay      self.framework.actions.addArgument('PETSc', 'Build', 'Set default architecture to '+self.arch.arch+' in '+conffile)
1169569865ddSSatish Balay    elif os.path.isfile(conffile):
1170569865ddSSatish Balay      try:
1171569865ddSSatish Balay        os.unlink(conffile)
1172569865ddSSatish Balay      except:
1173569865ddSSatish Balay        raise RuntimeError('Unable to remove file '+conffile+'. Did a different user create it?')
1174569865ddSSatish Balay    return
1175569865ddSSatish Balay
1176569865ddSSatish Balay#-----------------------------------------------------------------------------------------------------
1177f8833479SBarry Smith  def configureScript(self):
1178f8833479SBarry Smith    '''Output a script in the conf directory which will reproduce the configuration'''
1179f8833479SBarry Smith    import nargs
1180495bf1a9SSatish Balay    import sys
1181af0996ceSBarry Smith    scriptName = os.path.join(self.arch.arch,'lib','petsc','conf', 'reconfigure-'+self.arch.arch+'.py')
1182f8833479SBarry Smith    args = dict([(nargs.Arg.parseArgument(arg)[0], arg) for arg in self.framework.clArgs])
1183e97fc2efSSatish Balay    if 'with-clean' in args:
1184e97fc2efSSatish Balay      del args['with-clean']
1185d418e2d7SSatish Balay    if 'force' in args:
1186d418e2d7SSatish Balay      del args['force']
1187f8833479SBarry Smith    if 'configModules' in args:
11881063a081SSatish Balay      if nargs.Arg.parseArgument(args['configModules'])[1] == 'PETSc.Configure':
1189f8833479SBarry Smith        del args['configModules']
1190f8833479SBarry Smith    if 'optionsModule' in args:
119123a19ef1SSatish Balay      if nargs.Arg.parseArgument(args['optionsModule'])[1] == 'config.compilerOptions':
1192f8833479SBarry Smith        del args['optionsModule']
1193f8833479SBarry Smith    if not 'PETSC_ARCH' in args:
11941063a081SSatish Balay      args['PETSC_ARCH'] = 'PETSC_ARCH='+str(self.arch.arch)
1195c6ef1b5bSJed Brown    f = open(scriptName, 'w')
1196495bf1a9SSatish Balay    f.write('#!'+sys.executable+'\n')
1197f8833479SBarry Smith    f.write('if __name__ == \'__main__\':\n')
1198f8833479SBarry Smith    f.write('  import sys\n')
11997561c02cSSatish Balay    f.write('  import os\n')
12007561c02cSSatish Balay    f.write('  sys.path.insert(0, os.path.abspath(\'config\'))\n')
1201f8833479SBarry Smith    f.write('  import configure\n')
12021063a081SSatish Balay    # pretty print repr(args.values())
12031063a081SSatish Balay    f.write('  configure_options = [\n')
12048bec23c5SJed Brown    for itm in sorted(args.values()):
12051063a081SSatish Balay      f.write('    \''+str(itm)+'\',\n')
12061063a081SSatish Balay    f.write('  ]\n')
1207f8833479SBarry Smith    f.write('  configure.petsc_configure(configure_options)\n')
1208f8833479SBarry Smith    f.close()
1209f8833479SBarry Smith    try:
12105b6bfdb9SJed Brown      os.chmod(scriptName, 0o775)
12115b6bfdb9SJed Brown    except OSError as e:
1212f8833479SBarry Smith      self.framework.logPrint('Unable to make reconfigure script executable:\n'+str(e))
1213f8833479SBarry Smith    self.framework.actions.addArgument('PETSc', 'File creation', 'Created '+scriptName+' for automatic reconfiguration')
1214f8833479SBarry Smith    return
1215f8833479SBarry Smith
1216f8833479SBarry Smith  def configureInstall(self):
1217f8833479SBarry Smith    '''Setup the directories for installation'''
1218f8833479SBarry Smith    if self.framework.argDB['prefix']:
12195b4fc442SVaclav Hapla      self.addMakeRule('print_mesg_after_build','',
12205b4fc442SVaclav Hapla       ['-@echo "========================================="',
12215b4fc442SVaclav Hapla        '-@echo "Now to install the libraries do:"',
12225b4fc442SVaclav Hapla        '-@echo "%s${MAKE_USER} PETSC_DIR=${PETSC_DIR} PETSC_ARCH=${PETSC_ARCH} install"' % self.installdir.installSudo,
1223315b77e6SSatish Balay        '-@echo "========================================="'])
1224f8833479SBarry Smith    else:
12255b4fc442SVaclav Hapla      self.addMakeRule('print_mesg_after_build','',
12265b4fc442SVaclav Hapla       ['-@echo "========================================="',
12275b4fc442SVaclav Hapla        '-@echo "Now to check if the libraries are working do:"',
12285b4fc442SVaclav Hapla        '-@echo "${MAKE_USER} PETSC_DIR=${PETSC_DIR} PETSC_ARCH=${PETSC_ARCH} check"',
1229315b77e6SSatish Balay        '-@echo "========================================="'])
1230f8833479SBarry Smith      return
1231f8833479SBarry Smith
123228bb2e72SSatish Balay  def postProcessPackages(self):
123328bb2e72SSatish Balay    postPackages=[]
123428bb2e72SSatish Balay    for i in self.framework.packages:
123528bb2e72SSatish Balay      if hasattr(i,'postProcess'): postPackages.append(i)
123628bb2e72SSatish Balay    if postPackages:
1237e64d19dfSSatish Balay      # ctetgen needs petsc conf files. so attempt to create them early
1238a77eb93bSSatish Balay      self.framework.dumpConfFiles()
1239d9293e7bSBarry Smith      # tacky fix for dependency of Aluimia on Pflotran; requested via petsc-dev Matt provide a correct fix
1240d9293e7bSBarry Smith      for i in postPackages:
1241d9293e7bSBarry Smith        if i.name.upper() in ['PFLOTRAN']:
1242d9293e7bSBarry Smith          i.postProcess()
1243d9293e7bSBarry Smith          postPackages.remove(i)
124428bb2e72SSatish Balay      for i in postPackages: i.postProcess()
1245aa5c8b8eSBarry Smith      for i in postPackages:
1246aa5c8b8eSBarry Smith        if i.installedpetsc:
1247aa5c8b8eSBarry Smith          self.installed = 1
1248aa5c8b8eSBarry Smith          break
124928bb2e72SSatish Balay    return
1250f8833479SBarry Smith
1251f8833479SBarry Smith  def configure(self):
1252bf3e94a3SBarry Smith    if 'package-prefix-hash' in self.argDB:
1253bf3e94a3SBarry Smith      # turn off prefix if it was only used to for installing external packages.
1254bf3e94a3SBarry Smith      self.framework.argDB['prefix'] = ''
1255bf3e94a3SBarry Smith      self.dir = os.path.abspath(os.path.join(self.petscdir.dir, self.arch.arch))
1256bf3e94a3SBarry Smith      self.installdir.dir = self.dir
1257bf3e94a3SBarry Smith      self.installdir.petscDir = self.petscdir.dir
1258bf3e94a3SBarry Smith      self.petscDir = self.petscdir.dir
1259bf3e94a3SBarry Smith      self.petscArch = self.arch.arch
1260bf3e94a3SBarry Smith      self.addMakeMacro('PREFIXDIR',self.dir)
1261bf3e94a3SBarry Smith      self.confDir = os.path.abspath(os.path.join(self.petscdir.dir, self.arch.arch))
1262bf3e94a3SBarry Smith
1263f8833479SBarry Smith    if not os.path.samefile(self.petscdir.dir, os.getcwd()):
1264f8833479SBarry Smith      raise RuntimeError('Wrong PETSC_DIR option specified: '+str(self.petscdir.dir) + '\n  Configure invoked in: '+os.path.realpath(os.getcwd()))
1265550489e3SMatthew 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):
12663552d8fbSSatish Balay      raise RuntimeError('Incorrect option --prefix='+self.framework.argDB['prefix']+' specified. It cannot be same as PETSC_DIR!')
12678fd0dbdbSBarry Smith    if self.framework.argDB['prefix'] and self.framework.argDB['prefix'].find(' ') > -1:
12688fd0dbdbSBarry 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')
1269c16c35a9SSatish 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)):
1270c16c35a9SSatish Balay      raise RuntimeError('Incorrect option --prefix='+self.framework.argDB['prefix']+' specified. It cannot be same as PETSC_DIR/PETSC_ARCH!')
1271f16c1317SJed Brown    self.framework.header          = os.path.join(self.arch.arch,'include','petscconf.h')
1272f16c1317SJed Brown    self.framework.cHeader         = os.path.join(self.arch.arch,'include','petscfix.h')
1273bf113f49SJacob Faibussowitsch    self.framework.poisonheader    = os.path.join(self.arch.arch,'include','petscconf_poison.h')
12749c735a01SStefano Zampini    self.framework.pkgheader       = os.path.join(self.arch.arch,'include','petscpkg_version.h')
1275af0996ceSBarry Smith    self.framework.makeMacroHeader = os.path.join(self.arch.arch,'lib','petsc','conf','petscvariables')
1276af0996ceSBarry Smith    self.framework.makeRuleHeader  = os.path.join(self.arch.arch,'lib','petsc','conf','petscrules')
1277f8833479SBarry Smith    if self.libraries.math is None:
1278f8833479SBarry Smith      raise RuntimeError('PETSc requires a functional math library. Please send configure.log to petsc-maint@mcs.anl.gov.')
1279f8833479SBarry Smith    if self.languages.clanguage == 'Cxx' and not hasattr(self.compilers, 'CXX'):
1280f8833479SBarry Smith      raise RuntimeError('Cannot set C language to C++ without a functional C++ compiler.')
1281ed938b00SJed Brown    self.executeTest(self.configureRTLDDefault)
1282b2843cf1SBarry Smith    self.executeTest(self.configurePrefetch)
12832400fdedSBarry Smith    self.executeTest(self.configureUnused)
12841ef8df7fSJed Brown    self.executeTest(self.configureDeprecated)
128598ed35c3SBarry Smith    self.executeTest(self.configureIsatty)
1286e8e972b2SVaclav Hapla    self.executeTest(self.configureExpect)
1287e8e972b2SVaclav Hapla    self.executeTest(self.configureAlign)
1288e8e972b2SVaclav Hapla    self.executeTest(self.configureFunctionName)
1289e8e972b2SVaclav Hapla    self.executeTest(self.configureIntptrt)
1290f8833479SBarry Smith    self.executeTest(self.configureSolaris)
1291f8833479SBarry Smith    self.executeTest(self.configureLinux)
12920f64ec89SBarry Smith    self.executeTest(self.configureDarwin)
1293f8833479SBarry Smith    self.executeTest(self.configureWin32)
1294b10d012aSSatish Balay    self.executeTest(self.configureCygwinBrokenPipe)
1295569865ddSSatish Balay    self.executeTest(self.configureDefaultArch)
1296f8833479SBarry Smith    self.executeTest(self.configureScript)
1297f8833479SBarry Smith    self.executeTest(self.configureInstall)
129809bc878fSSatish Balay    self.executeTest(self.configureAtoll)
1299*9b81490aSJacob Faibussowitsch    self.executeTest(self.configureCoverage)
13007ce81a4bSJacob Faibussowitsch    self.executeTest(self.configureCoverageExecutable)
1301689a5dfaSJacob Faibussowitsch    self.executeTest(self.configureStrictPetscErrorCode)
1302f8833479SBarry Smith
1303f8833479SBarry Smith    self.Dump()
1304f8833479SBarry Smith    self.dumpConfigInfo()
13052a4161d9SMatthew G Knepley    self.dumpMachineInfo()
130649fe22e6SSatish Balay    self.delGenFiles()
130740277576SBarry Smith    # need to save the current state of BuildSystem so that postProcess() packages can read it in and perhaps run make install
130840277576SBarry Smith    self.framework.storeSubstitutions(self.framework.argDB)
1309492432c8SJed Brown    self.framework.argDB['configureCache'] = pickle.dumps(self.framework)
131040277576SBarry Smith    self.framework.argDB.save(force = True)
13118244ab14SJed Brown    self.DumpPkgconfig('PETSc.pc')
13128244ab14SJed Brown    self.DumpPkgconfig('petsc.pc')
1313351d3a41SMatthew G Knepley    self.DumpModule()
1314f7ad81e1SBarry Smith    self.postProcessPackages()
1315f8833479SBarry Smith    self.framework.log.write('================================================================================\n')
1316f8833479SBarry Smith    self.logClear()
1317f8833479SBarry Smith    return
1318