xref: /petsc/config/PETSc/Configure.py (revision 095fb05fba1caf9b11dcec8f3d0f42e5f5a166fb)
1f8833479SBarry Smithimport config.base
2f8833479SBarry Smith
3f8833479SBarry Smithimport os
46dd73af6SBarry Smithimport sys
5f8833479SBarry Smithimport re
6492432c8SJed Brownimport pickle
7f8833479SBarry Smith
8f8833479SBarry Smithclass Configure(config.base.Configure):
9f8833479SBarry Smith  def __init__(self, framework):
10f8833479SBarry Smith    config.base.Configure.__init__(self, framework)
11f8833479SBarry Smith    self.headerPrefix = 'PETSC'
12f8833479SBarry Smith    self.substPrefix  = 'PETSC'
13aa5c8b8eSBarry Smith    self.installed    = 0 # 1 indicates that Configure itself has already compiled and installed PETSc
144161761dSBarry Smith    self.found        = 1
15f8833479SBarry Smith    return
16f8833479SBarry Smith
177c939e48SSatish Balay  def __str2__(self):
185b4fc442SVaclav Hapla    desc = ['  Using GNU make: ' + self.make.make]
197ce81a4bSJacob Faibussowitsch    if self.defines.get('USE_COVERAGE'):
207ce81a4bSJacob Faibussowitsch      desc.extend([
217ce81a4bSJacob Faibussowitsch        '  Code coverage: yes',
227ce81a4bSJacob Faibussowitsch        '  Using code coverage executable: {}'.format(self.getMakeMacro('PETSC_COVERAGE_EXEC'))
237ce81a4bSJacob Faibussowitsch      ])
24aa5c8b8eSBarry Smith    if not self.installed:
25a0022257SSatish Balay      desc.append('xxx=========================================================================xxx')
26dc0529c6SBarry Smith      desc.append(' Configure stage complete. Now build PETSc libraries with:')
275b4fc442SVaclav Hapla      desc.append('   %s PETSC_DIR=%s PETSC_ARCH=%s all' % (self.make.make_user, self.petscdir.dir, self.arch.arch))
28a0022257SSatish Balay      desc.append('xxx=========================================================================xxx')
29aa5c8b8eSBarry Smith    else:
30aa5c8b8eSBarry Smith      desc.append('xxx=========================================================================xxx')
31aa5c8b8eSBarry Smith      desc.append(' Installation complete. You do not need to run make to compile or install the software')
32aa5c8b8eSBarry Smith      desc.append('xxx=========================================================================xxx')
337ce81a4bSJacob Faibussowitsch    desc.append('')
347ce81a4bSJacob Faibussowitsch    return '\n'.join(desc)
35f8833479SBarry Smith
36f8833479SBarry Smith  def setupHelp(self, help):
37f8833479SBarry Smith    import nargs
38ce0b2093SBarry Smith    help.addArgument('PETSc',  '-prefix=<dir>',                              nargs.Arg(None, '', 'Specifiy location to install PETSc (eg. /usr/local)'))
397deb5ab3SBarry Smith    help.addArgument('PETSc',  '-with-prefetch=<bool>',                      nargs.ArgBool(None, 1,'Enable checking for prefetch instructions'))
40eed94e11SSatish Balay    help.addArgument('Windows','-with-windows-graphics=<bool>',              nargs.ArgBool(None, 1,'Enable check for Windows Graphics'))
41569865ddSSatish Balay    help.addArgument('PETSc', '-with-default-arch=<bool>',                   nargs.ArgBool(None, 1, 'Allow using the last configured arch without setting PETSC_ARCH'))
4257cb31baSSatish Balay    help.addArgument('PETSc','-with-single-library=<bool>',                  nargs.ArgBool(None, 1,'Put all PETSc code into the single -lpetsc library'))
43cb297985SSatish Balay    help.addArgument('PETSc','-with-fortran-bindings=<bool>',                nargs.ArgBool(None, 1,'Build PETSc fortran bindings in the library and corresponding module files'))
44525d6f2eSBarry Smith    help.addArgument('PETSc', '-with-ios=<bool>',                            nargs.ArgBool(None, 0, 'Build an iPhone/iPad version of PETSc library'))
4513f9d092SSatish Balay    help.addArgument('PETSc', '-with-display=<x11display>',                  nargs.Arg(None, '', 'Specifiy DISPLAY env variable for use with matlab test)'))
462c30b4dfSSatish Balay    help.addArgument('PETSc', '-with-package-scripts=<pyscripts>',           nargs.ArgFileList(None,None,'Specify configure package scripts for user provided packages'))
477ce81a4bSJacob Faibussowitsch    help.addArgument('PETSc', '-with-coverage=<bool>',                       nargs.ArgFuzzyBool(None, value=0, help='Enable or disable code-coverage collection'))
487ce81a4bSJacob 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'))
49a0c7f9aaSSamuel Khuvis    help.addArgument('PETSc', '-with-tau-perfstubs=<bool>',                  nargs.ArgBool(None, 1,'Enable TAU profiler stubs'))
50689a5dfaSJacob 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'))
51f8833479SBarry Smith    return
52f8833479SBarry Smith
536dd73af6SBarry Smith  def registerPythonFile(self,filename,directory):
546dd73af6SBarry Smith    ''' Add a python file to the framework and registers its headerprefix, ... externalpackagedir
556dd73af6SBarry Smith        directory is the directory where the file relative to the BuildSystem or config path in python notation with . '''
566dd73af6SBarry Smith    (utilityName, ext) = os.path.splitext(filename)
576dd73af6SBarry Smith    if not utilityName.startswith('.') and not utilityName.startswith('#') and ext == '.py' and not utilityName == '__init__':
586dd73af6SBarry Smith      if directory: directory = directory+'.'
596dd73af6SBarry Smith      utilityObj                             = self.framework.require(directory+utilityName, self)
606dd73af6SBarry Smith      utilityObj.headerPrefix                = self.headerPrefix
616dd73af6SBarry Smith      utilityObj.archProvider                = self.arch
626dd73af6SBarry Smith      utilityObj.languageProvider            = self.languages
636dd73af6SBarry Smith      utilityObj.installDirProvider          = self.installdir
646dd73af6SBarry Smith      utilityObj.externalPackagesDirProvider = self.externalpackagesdir
656dd73af6SBarry Smith      utilityObj.precisionProvider           = self.scalartypes
666dd73af6SBarry Smith      utilityObj.indexProvider               = self.indexTypes
676dd73af6SBarry Smith      setattr(self, utilityName.lower(), utilityObj)
6851294b80SMatthew G. Knepley      return utilityObj
6951294b80SMatthew G. Knepley    return None
706dd73af6SBarry Smith
71f8833479SBarry Smith  def setupDependencies(self, framework):
72f8833479SBarry Smith    config.base.Configure.setupDependencies(self, framework)
73dca78d2bSSatish Balay    self.programs      = framework.require('config.programs',           self)
74f8833479SBarry Smith    self.setCompilers  = framework.require('config.setCompilers',       self)
757ce81a4bSJacob Faibussowitsch    self.compilerFlags = framework.require('config.compilerFlags',      self)
7630b8aa07SMatthew G. Knepley    self.compilers     = framework.require('config.compilers',          self)
779d310bb7SBarry Smith    self.arch          = framework.require('PETSc.options.arch',        self.setCompilers)
789d310bb7SBarry Smith    self.petscdir      = framework.require('PETSc.options.petscdir',    self.arch)
799d310bb7SBarry Smith    self.installdir    = framework.require('PETSc.options.installDir',  self)
804e00a515SSatish Balay    self.dataFilesPath = framework.require('PETSc.options.dataFilesPath',self)
816dd73af6SBarry Smith    self.scalartypes   = framework.require('PETSc.options.scalarTypes', self)
826dd73af6SBarry Smith    self.indexTypes    = framework.require('PETSc.options.indexTypes',  self)
839d310bb7SBarry Smith    self.languages     = framework.require('PETSc.options.languages',   self.setCompilers)
8430b8aa07SMatthew G. Knepley    self.indexTypes    = framework.require('PETSc.options.indexTypes',  self.compilers)
85f8833479SBarry Smith    self.types         = framework.require('config.types',              self)
86f8833479SBarry Smith    self.headers       = framework.require('config.headers',            self)
87f8833479SBarry Smith    self.functions     = framework.require('config.functions',          self)
88f8833479SBarry Smith    self.libraries     = framework.require('config.libraries',          self)
89cd37d877SShri Abhyankar    self.atomics       = framework.require('config.atomics',            self)
909481793eSSatish Balay    self.make          = framework.require('config.packages.make',      self)
919552296fSBarry Smith    self.blasLapack    = framework.require('config.packages.BlasLapack',self)
92e6b0c433SBarry Smith    self.mpi           = framework.require('config.packages.MPI',       self)
930542e31aSBarry Smith    self.fortran       = framework.require('config.compilersFortran',   self)
940542e31aSBarry Smith    self.externalpackagesdir = framework.require('PETSc.options.externalpackagesdir',self)
9549d43ecaSSatish Balay
9609a6cbfcSBernhard M. Wiedemann    for utility in sorted(os.listdir(os.path.join('config','PETSc','options'))):
976dd73af6SBarry Smith      self.registerPythonFile(utility,'PETSc.options')
989d310bb7SBarry Smith
9909a6cbfcSBernhard M. Wiedemann    for utility in sorted(os.listdir(os.path.join('config','BuildSystem','config','utilities'))):
1006dd73af6SBarry Smith      self.registerPythonFile(utility,'config.utilities')
10106e08bc7SBarry Smith
10209a6cbfcSBernhard M. Wiedemann    for package in sorted(os.listdir(os.path.join('config', 'BuildSystem', 'config', 'packages'))):
10351294b80SMatthew G. Knepley      obj = self.registerPythonFile(package,'config.packages')
10451294b80SMatthew G. Knepley      if obj:
10551294b80SMatthew G. Knepley        obj.archProvider                = self.framework.requireModule(obj.archProvider, obj)
10651294b80SMatthew G. Knepley        obj.languageProvider            = self.framework.requireModule(obj.languageProvider, obj)
10751294b80SMatthew G. Knepley        obj.installDirProvider          = self.framework.requireModule(obj.installDirProvider, obj)
10851294b80SMatthew G. Knepley        obj.externalPackagesDirProvider = self.framework.requireModule(obj.externalPackagesDirProvider, obj)
10951294b80SMatthew G. Knepley        obj.precisionProvider           = self.framework.requireModule(obj.precisionProvider, obj)
11051294b80SMatthew G. Knepley        obj.indexProvider               = self.framework.requireModule(obj.indexProvider, obj)
1116dd73af6SBarry Smith
1125faf1eacSMatthew G. Knepley    # Force blaslapack and opencl to depend on scalarType so precision is set before BlasLapack is built
1139d310bb7SBarry Smith    framework.require('PETSc.options.scalarTypes', self.f2cblaslapack)
1149d310bb7SBarry Smith    framework.require('PETSc.options.scalarTypes', self.fblaslapack)
1159d310bb7SBarry Smith    framework.require('PETSc.options.scalarTypes', self.blaslapack)
1165faf1eacSMatthew G. Knepley    framework.require('PETSc.options.scalarTypes', self.opencl)
117f8833479SBarry Smith
118dca78d2bSSatish Balay    self.programs.headerPrefix     = self.headerPrefix
1195b3958d3SJacob Faibussowitsch    self.setCompilers.headerPrefix = self.headerPrefix
120f8833479SBarry Smith    self.compilers.headerPrefix    = self.headerPrefix
1210542e31aSBarry Smith    self.fortran.headerPrefix      = self.headerPrefix
122f8833479SBarry Smith    self.types.headerPrefix        = self.headerPrefix
123f8833479SBarry Smith    self.headers.headerPrefix      = self.headerPrefix
124f8833479SBarry Smith    self.functions.headerPrefix    = self.headerPrefix
125f8833479SBarry Smith    self.libraries.headerPrefix    = self.headerPrefix
1266dd73af6SBarry Smith
1272c30b4dfSSatish Balay    # Register user provided package scripts
1282c30b4dfSSatish Balay    if 'with-package-scripts' in self.framework.argDB:
1292c30b4dfSSatish Balay      for script in self.framework.argDB['with-package-scripts']:
1302c30b4dfSSatish Balay        if os.path.splitext(script)[1] != '.py':
1312c30b4dfSSatish Balay          raise RuntimeError('Only python scripts compatible with configure package script format should be specified! Invalid option -with-package-scripts='+script)
1322c30b4dfSSatish Balay        self.framework.logPrint('User is registering a new package script: '+script)
1332c30b4dfSSatish Balay        dname,fname = os.path.split(script)
1342c30b4dfSSatish Balay        if dname: sys.path.append(dname)
1352c30b4dfSSatish Balay        self.registerPythonFile(fname,'')
1366dd73af6SBarry Smith
1376dd73af6SBarry Smith    # test for a variety of basic headers and functions
1384211eb48SBarry Smith    headersC = map(lambda name: name+'.h',['setjmp','dos','fcntl','float','io','malloc','pwd','strings',
139ace159c0SJed Brown                                            'unistd','machine/endian','sys/param','sys/procfs','sys/resource',
1402475b7caSBarry Smith                                            'sys/systeminfo','sys/times','sys/utsname',
1417e4f0192SMosè Giordano                                            'sys/socket','sys/wait','netinet/in','netdb','direct','time','Ws2tcpip','sys/types',
14218da0197SPierre Jolivet                                            'WindowsX','float','ieeefp','stdint','inttypes','immintrin'])
14345082d64SJed Brown    functions = ['access','_access','clock','drand48','getcwd','_getcwd','getdomainname','gethostname',
144d7976ebaSJed Brown                 'getwd','posix_memalign','popen','PXFGETARG','rand','getpagesize',
1454211eb48SBarry Smith                 'readlink','realpath','usleep','sleep','_sleep',
1462475b7caSBarry Smith                 'uname','snprintf','_snprintf','lseek','_lseek','time','fork','stricmp',
1472475b7caSBarry Smith                 'strcasecmp','bzero','dlopen','dlsym','dlclose','dlerror',
14867f4e542SPierre Jolivet                 '_set_output_format','_mkdir','socket','gethostbyname','fpresetsticky',
14918da0197SPierre Jolivet                 'fpsetsticky','__gcov_dump']
150b0651e32SBarry Smith    libraries = [(['fpe'],'handle_sigfpes')]
151b0651e32SBarry Smith    librariessock = [(['socket','nsl'],'socket')]
152f8833479SBarry Smith    self.headers.headers.extend(headersC)
153f8833479SBarry Smith    self.functions.functions.extend(functions)
154b0651e32SBarry Smith    self.libraries.libraries.extend(libraries)
155b0651e32SBarry Smith    if not hasattr(self,'socket'):
156b0651e32SBarry Smith      self.libraries.libraries.extend(librariessock)
157f8833479SBarry Smith    return
158f8833479SBarry Smith
1598244ab14SJed Brown  def DumpPkgconfig(self, petsc_pc):
160262119f8SBarry Smith    ''' Create a pkg-config file '''
161262119f8SBarry Smith    if not os.path.exists(os.path.join(self.petscdir.dir,self.arch.arch,'lib','pkgconfig')):
162262119f8SBarry Smith      os.makedirs(os.path.join(self.petscdir.dir,self.arch.arch,'lib','pkgconfig'))
1632eefe1c6SJed Brown    with open(os.path.join(self.petscdir.dir,self.arch.arch,'lib','pkgconfig',petsc_pc),'w') as fd:
1645e3311eeSJed Brown      cflags_inc = ['-I${includedir}']
165262119f8SBarry Smith      if self.framework.argDB['prefix']:
1665bb5b263SMatthew G. Knepley        fd.write('prefix='+self.installdir.dir+'\n')
167262119f8SBarry Smith      else:
168e1e675deSJed Brown        fd.write('prefix='+os.path.join(self.petscdir.dir, self.arch.arch)+'\n')
169e1e675deSJed Brown        cflags_inc.append('-I' + os.path.join(self.petscdir.dir, 'include'))
170262119f8SBarry Smith      fd.write('exec_prefix=${prefix}\n')
171262119f8SBarry Smith      fd.write('includedir=${prefix}/include\n')
1725e3311eeSJed Brown      fd.write('libdir=${prefix}/lib\n')
173262119f8SBarry Smith
1742eefe1c6SJed Brown      with self.setCompilers.Language('C'):
175262119f8SBarry Smith        fd.write('ccompiler='+self.setCompilers.getCompiler()+'\n')
176756c7f9fSJed Brown        fd.write('cflags_extra='+self.setCompilers.getCompilerFlags().strip()+'\n')
177756c7f9fSJed Brown        fd.write('cflags_dep='+self.compilers.dependenciesGenerationFlag.get('C','')+'\n')
178756c7f9fSJed Brown        fd.write('ldflag_rpath='+self.setCompilers.CSharedLinkerFlag+'\n')
17903e383c8SJed Brown      if hasattr(self.compilers, 'CXX'):
1802eefe1c6SJed Brown        with self.setCompilers.Language('C++'):
181262119f8SBarry Smith          fd.write('cxxcompiler='+self.setCompilers.getCompiler()+'\n')
182756c7f9fSJed Brown          fd.write('cxxflags_extra='+self.setCompilers.getCompilerFlags().strip()+'\n')
183262119f8SBarry Smith      if hasattr(self.compilers, 'FC'):
1842eefe1c6SJed Brown        with self.setCompilers.Language('FC'):
185262119f8SBarry Smith          fd.write('fcompiler='+self.setCompilers.getCompiler()+'\n')
186756c7f9fSJed Brown          fd.write('fflags_extra='+self.setCompilers.getCompilerFlags().strip()+'\n')
18750520af6SPatrick Sanan      if hasattr(self.compilers, 'CUDAC'):
18850520af6SPatrick Sanan        with self.setCompilers.Language('CUDA'):
18950520af6SPatrick Sanan          fd.write('cudacompiler='+self.setCompilers.getCompiler()+'\n')
19050520af6SPatrick Sanan          fd.write('cudaflags_extra='+self.setCompilers.getCompilerFlags().strip()+'\n')
1918f561fa3SPatrick Sanan          p = self.framework.require('config.packages.cuda')
19250520af6SPatrick Sanan          fd.write('cudalib='+self.libraries.toStringNoDupes(p.lib)+'\n')
19350520af6SPatrick Sanan          fd.write('cudainclude='+self.headers.toStringNoDupes(p.include)+'\n')
1947ba7a817SBarry Smith          if hasattr(self.setCompilers,'CUDA_CXX'):
1957ba7a817SBarry Smith            fd.write('cuda_cxx='+self.setCompilers.CUDA_CXX+'\n')
1967ba7a817SBarry Smith            fd.write('cuda_cxxflags='+self.setCompilers.CUDA_CXXFLAGS+'\n')
197262119f8SBarry Smith
198262119f8SBarry Smith      fd.write('\n')
199262119f8SBarry Smith      fd.write('Name: PETSc\n')
200262119f8SBarry Smith      fd.write('Description: Library to solve ODEs and algebraic equations\n')
201351d3a41SMatthew G Knepley      fd.write('Version: %s\n' % self.petscdir.version)
2025e3311eeSJed Brown      fd.write('Cflags: ' + ' '.join([self.setCompilers.CPPFLAGS] + cflags_inc) + '\n')
20337371b91SJed Brown      fd.write('Libs: '+self.libraries.toStringNoDupes(['-L${libdir}', self.petsclib], with_rpath=False)+'\n')
2048ebf8858SJed Brown      # Remove RPATH flags from library list.  User can add them using
2058ebf8858SJed Brown      # pkg-config --variable=ldflag_rpath and pkg-config --libs-only-L
206de8f682fSSatish 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')
207262119f8SBarry Smith    return
208262119f8SBarry Smith
209351d3a41SMatthew G Knepley  def DumpModule(self):
210351d3a41SMatthew G Knepley    ''' Create a module file '''
211af0996ceSBarry Smith    if not os.path.exists(os.path.join(self.petscdir.dir,self.arch.arch,'lib','petsc','conf','modules')):
212af0996ceSBarry Smith      os.makedirs(os.path.join(self.petscdir.dir,self.arch.arch,'lib','petsc','conf','modules'))
213af0996ceSBarry Smith    if not os.path.exists(os.path.join(self.petscdir.dir,self.arch.arch,'lib','petsc','conf','modules','petsc')):
214af0996ceSBarry Smith      os.makedirs(os.path.join(self.petscdir.dir,self.arch.arch,'lib','petsc','conf','modules','petsc'))
215351d3a41SMatthew G Knepley    if self.framework.argDB['prefix']:
2165bb5b263SMatthew G. Knepley      installdir  = self.installdir.dir
21755d606a3SSatish Balay      installarch = ''
21855d606a3SSatish Balay      installpath = os.path.join(installdir,'bin')
219351d3a41SMatthew G Knepley    else:
220351d3a41SMatthew G Knepley      installdir  = self.petscdir.dir
22155d606a3SSatish Balay      installarch = self.arch.arch
22255d606a3SSatish Balay      installpath = os.path.join(installdir,installarch,'bin')+':'+os.path.join(installdir,'bin')
223af0996ceSBarry Smith    fd = open(os.path.join(self.petscdir.dir,self.arch.arch,'lib','petsc','conf','modules','petsc',self.petscdir.version),'w')
224351d3a41SMatthew G Knepley    fd.write('''\
225351d3a41SMatthew G Knepley#%%Module
226351d3a41SMatthew G Knepley
227351d3a41SMatthew G Knepleyproc ModulesHelp { } {
228351d3a41SMatthew G Knepley    puts stderr "This module sets the path and environment variables for petsc-%s"
229a17b96a8SKyle Gerard Felker    puts stderr "     see https://petsc.org/ for more information      "
230351d3a41SMatthew G Knepley    puts stderr ""
231351d3a41SMatthew G Knepley}
232351d3a41SMatthew G Knepleymodule-whatis "PETSc - Portable, Extensible Toolkit for Scientific Computation"
233351d3a41SMatthew G Knepley
234dd486775SJed Brownset petsc_dir   "%s"
235dd486775SJed Brownset petsc_arch  "%s"
236351d3a41SMatthew G Knepley
237dd486775SJed Brownsetenv PETSC_ARCH "$petsc_arch"
238dd486775SJed Brownsetenv PETSC_DIR "$petsc_dir"
239dd486775SJed Brownprepend-path PATH "%s"
24055d606a3SSatish Balay''' % (self.petscdir.version, installdir, installarch, installpath))
241351d3a41SMatthew G Knepley    fd.close()
242351d3a41SMatthew G Knepley    return
243351d3a41SMatthew G Knepley
244f8833479SBarry Smith  def Dump(self):
245f8833479SBarry Smith    ''' Actually put the values into the configuration files '''
246f8833479SBarry Smith    # eventually everything between -- should be gone
24717f368bcSBarry Smith    if self.mpi.usingMPIUni:
24817f368bcSBarry Smith      #
24917f368bcSBarry Smith      # Remove any MPI/MPICH include files that may have been put here by previous runs of ./configure
2507908f030SMatthew 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)
25117f368bcSBarry Smith
252b5f71184SBarry Smith    self.logPrintDivider()
253b5f71184SBarry Smith    # Test for compiler-specific macros that need to be defined.
254b5f71184SBarry Smith    if self.setCompilers.isCrayVector('CC', self.log):
255b5f71184SBarry Smith      self.addDefine('HAVE_CRAY_VECTOR','1')
256b5f71184SBarry Smith
257b5f71184SBarry Smith    if self.functions.haveFunction('gethostbyname') and self.functions.haveFunction('socket') and self.headers.haveHeader('netinet/in.h'):
258b5f71184SBarry Smith      self.addDefine('USE_SOCKET_VIEWER','1')
259b5f71184SBarry Smith      if self.checkCompile('#include <sys/socket.h>','setsockopt(0,SOL_SOCKET,SO_REUSEADDR,0,0)'):
260b5f71184SBarry Smith        self.addDefine('HAVE_SO_REUSEADDR','1')
261b5f71184SBarry Smith
262b5f71184SBarry Smith    self.logPrintDivider()
2635f27b2e0SBarry Smith    self.setCompilers.pushLanguage('C')
2645f27b2e0SBarry Smith    compiler = self.setCompilers.getCompiler()
265217fe27eSSatish Balay    if [s for s in ['mpicc','mpiicc'] if os.path.basename(compiler).find(s)>=0]:
2665f27b2e0SBarry Smith      try:
2675f27b2e0SBarry Smith        output   = self.executeShellCommand(compiler + ' -show', log = self.log)[0]
2685f27b2e0SBarry Smith        compiler = output.split(' ')[0]
269f424265bSStefano Zampini        self.addDefine('MPICC_SHOW','"'+output.strip().replace('\n','\\\\n').replace('"','')+'"')
2705f27b2e0SBarry Smith      except:
271c9872b61SBarry Smith        self.addDefine('MPICC_SHOW','"Unavailable"')
272c9872b61SBarry Smith    else:
273c9872b61SBarry Smith      self.addDefine('MPICC_SHOW','"Unavailable"')
2745f27b2e0SBarry Smith    self.setCompilers.popLanguage()
275f8833479SBarry Smith#-----------------------------------------------------------------------------------------------------
276f8833479SBarry Smith
277f8833479SBarry Smith    # Sometimes we need C compiler, even if built with C++
278f8833479SBarry Smith    self.setCompilers.pushLanguage('C')
279e4c30378SBarry Smith    # do not use getCompilerFlags() because that automatically includes the CPPFLAGS so one ends up with duplication flags in makefile usage
280e4c30378SBarry Smith    self.addMakeMacro('CC_FLAGS',self.setCompilers.CFLAGS)
281f8833479SBarry Smith    self.setCompilers.popLanguage()
282f8833479SBarry Smith
28334f774f6SJed Brown    # And sometimes we need a C++ compiler even when PETSc is built with C
28434f774f6SJed Brown    if hasattr(self.compilers, 'CXX'):
28534f774f6SJed Brown      self.setCompilers.pushLanguage('Cxx')
28629921a8fSScott Kruger      self.addDefine('HAVE_CXX','1')
2870b119762SSatish Balay      self.addMakeMacro('CXXPP_FLAGS',self.setCompilers.CXXPPFLAGS)
288e4c30378SBarry Smith      # do not use getCompilerFlags() because that automatically includes the CXXPPFLAGS so one ends up with duplication flags in makefile usage
289e4c30378SBarry Smith      self.addMakeMacro('CXX_FLAGS',self.setCompilers.CXXFLAGS+' '+self.setCompilers.CXX_CXXFLAGS)
2902f4326f3SSatish Balay      cxx_linker = self.setCompilers.getLinker()
2912f4326f3SSatish Balay      self.addMakeMacro('CXX_LINKER',cxx_linker)
2922f4326f3SSatish Balay      self.addMakeMacro('CXX_LINKER_FLAGS',self.setCompilers.getLinkerFlags())
29334f774f6SJed Brown      self.setCompilers.popLanguage()
29434f774f6SJed Brown
295f8833479SBarry Smith    # C preprocessor values
2961315f054SBarry Smith    self.addMakeMacro('CPP_FLAGS',self.setCompilers.CPPFLAGS)
297f8833479SBarry Smith
298f8833479SBarry Smith    # compiler values
299f8833479SBarry Smith    self.setCompilers.pushLanguage(self.languages.clanguage)
300f8833479SBarry Smith    self.addMakeMacro('PCC',self.setCompilers.getCompiler())
301e4c30378SBarry Smith    # do not use getCompilerFlags() because that automatically includes the preprocessor flags so one ends up with duplication flags in makefile usage
302e4c30378SBarry Smith    if self.languages.clanguage == 'C':
303e4c30378SBarry Smith      self.addMakeMacro('PCC_FLAGS','$(CC_FLAGS)')
304e4c30378SBarry Smith    else:
305e4c30378SBarry Smith      self.addMakeMacro('PCC_FLAGS','$(CXX_FLAGS)')
306f8833479SBarry Smith    self.setCompilers.popLanguage()
307f8833479SBarry Smith    # .o or .obj
308f8833479SBarry Smith    self.addMakeMacro('CC_SUFFIX','o')
309f8833479SBarry Smith
310f8833479SBarry Smith    # executable linker values
311f8833479SBarry Smith    self.setCompilers.pushLanguage(self.languages.clanguage)
312f8833479SBarry Smith    pcc_linker = self.setCompilers.getLinker()
313f8833479SBarry Smith    self.addMakeMacro('PCC_LINKER',pcc_linker)
314c84a332bSSatish Balay    self.addMakeMacro('PCC_LINKER_FLAGS',self.setCompilers.getLinkerFlags())
315f8833479SBarry Smith    self.setCompilers.popLanguage()
316f8833479SBarry Smith    # '' for Unix, .exe for Windows
317f8833479SBarry Smith    self.addMakeMacro('CC_LINKER_SUFFIX','')
318f8833479SBarry Smith
319f8833479SBarry Smith    if hasattr(self.compilers, 'FC'):
320cb297985SSatish Balay      if self.framework.argDB['with-fortran-bindings']:
321257f4e5aSSatish Balay        if not self.fortran.fortranIsF90:
322257f4e5aSSatish Balay          raise RuntimeError('Error! Fortran compiler "'+self.compilers.FC+'" does not support F90! PETSc fortran bindings require a F90 compiler')
323cb297985SSatish Balay        self.addDefine('HAVE_FORTRAN','1')
324f8833479SBarry Smith      self.setCompilers.pushLanguage('FC')
325f8833479SBarry Smith      # need FPPFLAGS in config/setCompilers
3260b119762SSatish Balay      self.addMakeMacro('FPP_FLAGS',self.setCompilers.FPPFLAGS)
327f8833479SBarry Smith
328f8833479SBarry Smith      # compiler values
329f8833479SBarry Smith      self.addMakeMacro('FC_FLAGS',self.setCompilers.getCompilerFlags())
330f8833479SBarry Smith      self.setCompilers.popLanguage()
331f8833479SBarry Smith      # .o or .obj
332f8833479SBarry Smith      self.addMakeMacro('FC_SUFFIX','o')
333f8833479SBarry Smith
334f8833479SBarry Smith      # executable linker values
335f8833479SBarry Smith      self.setCompilers.pushLanguage('FC')
336a9acdec7SBarry Smith      self.addMakeMacro('FC_LINKER',self.setCompilers.getLinker())
3376d53d35eSSatish Balay      self.addMakeMacro('FC_LINKER_FLAGS',self.setCompilers.getLinkerFlags())
338bb82cf9cSSatish Balay      self.setCompilers.popLanguage()
3395d631499SMatthew Knepley
3405d631499SMatthew Knepley      # F90 Modules
3415d631499SMatthew Knepley      if self.setCompilers.fortranModuleIncludeFlag:
3425d631499SMatthew Knepley        self.addMakeMacro('FC_MODULE_FLAG', self.setCompilers.fortranModuleIncludeFlag)
3436ddd6694SSatish Balay      else: # for non-f90 compilers like g77
3446ddd6694SSatish Balay        self.addMakeMacro('FC_MODULE_FLAG', '-I')
345a324c51cSMatthew G Knepley      if self.setCompilers.fortranModuleIncludeFlag:
346a324c51cSMatthew G Knepley        self.addMakeMacro('FC_MODULE_OUTPUT_FLAG', self.setCompilers.fortranModuleOutputFlag)
347f8833479SBarry Smith    else:
348f8833479SBarry Smith      self.addMakeMacro('FC','')
349f8833479SBarry Smith
35046a3958fSBarry Smith    if hasattr(self.compilers, 'CUDAC'):
3517ff2890cSSatish Balay      self.setCompilers.pushLanguage('CUDA')
352d93a25ecSSatish Balay      self.addMakeMacro('CUDAC_FLAGS',self.setCompilers.getCompilerFlags())
35350dcbc5aSJunchao Zhang      self.addMakeMacro('CUDAPP_FLAGS',self.setCompilers.CUDAPPFLAGS)
3547ff2890cSSatish Balay      self.setCompilers.popLanguage()
3557ff2890cSSatish Balay
356694a2f0eSJunchao Zhang    if hasattr(self.compilers, 'HIPC'):
35728f796eaSScott Kruger      self.setCompilers.pushLanguage('HIP')
358694a2f0eSJunchao Zhang      self.addMakeMacro('HIPC_FLAGS',self.setCompilers.getCompilerFlags())
3597fb1458fSStefano Zampini      self.addMakeMacro('HIPPP_FLAGS',self.setCompilers.HIPPPFLAGS)
36028f796eaSScott Kruger      self.setCompilers.popLanguage()
36128f796eaSScott Kruger
36250dcbc5aSJunchao Zhang    if hasattr(self.compilers, 'SYCLC'):
36328f796eaSScott Kruger      self.setCompilers.pushLanguage('SYCL')
36450dcbc5aSJunchao Zhang      self.addMakeMacro('SYCLC_FLAGS',self.setCompilers.getCompilerFlags())
36550dcbc5aSJunchao Zhang      self.addMakeMacro('SYCLPP_FLAGS',self.setCompilers.SYCLPPFLAGS)
36628f796eaSScott Kruger      self.setCompilers.popLanguage()
36728f796eaSScott Kruger
368f8833479SBarry Smith    # shared library linker values
369f8833479SBarry Smith    self.setCompilers.pushLanguage(self.languages.clanguage)
370f8833479SBarry Smith    # need to fix BuildSystem to collect these separately
371f8833479SBarry Smith    self.addMakeMacro('SL_LINKER',self.setCompilers.getLinker())
37270db8aa6SSatish Balay    self.addMakeMacro('SL_LINKER_FLAGS','${PCC_LINKER_FLAGS}')
373f8833479SBarry Smith    self.setCompilers.popLanguage()
374f8833479SBarry Smith    # One of 'a', 'so', 'lib', 'dll', 'dylib' (perhaps others also?) depending on the library generator and architecture
375f8833479SBarry Smith    # Note: . is not included in this macro, consistent with AR_LIB_SUFFIX
376f8833479SBarry Smith    if self.setCompilers.sharedLibraryExt == self.setCompilers.AR_LIB_SUFFIX:
377f8833479SBarry Smith      self.addMakeMacro('SL_LINKER_SUFFIX', '')
37846bc77b6SBarry Smith      self.addDefine('SLSUFFIX','""')
379f8833479SBarry Smith    else:
380f8833479SBarry Smith      self.addMakeMacro('SL_LINKER_SUFFIX', self.setCompilers.sharedLibraryExt)
38146bc77b6SBarry Smith      self.addDefine('SLSUFFIX','"'+self.setCompilers.sharedLibraryExt+'"')
382bb82cf9cSSatish Balay
38323e93537SBarry Smith    self.addMakeMacro('SL_LINKER_LIBS','${PETSC_EXTERNAL_LIB_BASIC}')
384bb82cf9cSSatish Balay
385f8833479SBarry Smith#-----------------------------------------------------------------------------------------------------
386f8833479SBarry Smith
387f8833479SBarry Smith    # CONLY or CPP. We should change the PETSc makefiles to do this better
388f8833479SBarry Smith    if self.languages.clanguage == 'C': lang = 'CONLY'
389f8833479SBarry Smith    else: lang = 'CXXONLY'
390f8833479SBarry Smith    self.addMakeMacro('PETSC_LANGUAGE',lang)
391f8833479SBarry Smith
392f8833479SBarry Smith    # real or complex
393f8833479SBarry Smith    self.addMakeMacro('PETSC_SCALAR',self.scalartypes.scalartype)
394f8833479SBarry Smith    # double or float
395f8833479SBarry Smith    self.addMakeMacro('PETSC_PRECISION',self.scalartypes.precision)
396f8833479SBarry Smith
397f8833479SBarry Smith    if self.framework.argDB['with-batch']:
398f8833479SBarry Smith      self.addMakeMacro('PETSC_WITH_BATCH','1')
399f8833479SBarry Smith
400f8833479SBarry Smith#-----------------------------------------------------------------------------------------------------
401a6cc6bb1SBarry Smith    # print include and lib for makefiles
402b5f71184SBarry Smith    self.logPrintDivider()
403f8833479SBarry Smith    self.framework.packages.reverse()
4045a21677cSJed Brown    petscincludes = [os.path.join(self.petscdir.dir,'include'),os.path.join(self.petscdir.dir,self.arch.arch,'include')]
4058749e224SJunchao Zhang    petscincludes_install = [os.path.join(self.installdir.dir, 'include')] if self.framework.argDB['prefix'] else petscincludes
4065a21677cSJed Brown    includes = []
407de8f682fSSatish Balay    self.packagelibs = []
408f8833479SBarry Smith    for i in self.framework.packages:
4097f0ff1afSBarry Smith      if not i.required:
4103972cb20SJacob Faibussowitsch        if i.devicePackage:
4113972cb20SJacob Faibussowitsch          self.addDefine('HAVE_DEVICE',1)
412eeb16384SBarry Smith        self.addDefine('HAVE_'+i.PACKAGE.replace('-','_'), 1)  # ONLY list package if it is used directly by PETSc (and not only by another package)
413f8833479SBarry Smith      if not isinstance(i.lib, list):
414f8833479SBarry Smith        i.lib = [i.lib]
415de8f682fSSatish Balay      if i.linkedbypetsc: self.packagelibs.extend(i.lib)
416eeb16384SBarry Smith      self.addMakeMacro(i.PACKAGE.replace('-','_')+'_LIB', self.libraries.toStringNoDupes(i.lib))
417f8833479SBarry Smith      if hasattr(i,'include'):
418f8833479SBarry Smith        if not isinstance(i.include,list):
419f8833479SBarry Smith          i.include = [i.include]
420ac9e4c42SSatish Balay        includes.extend(i.include)
421eeb16384SBarry Smith        self.addMakeMacro(i.PACKAGE.replace('-','_')+'_INCLUDE',self.headers.toStringNoDupes(i.include))
4222df986feSBarry Smith    if self.framework.argDB['with-single-library']:
423e282ce78SJed Brown      self.petsclib = '-lpetsc'
42491bb3077SSatish Balay    else:
425e282ce78SJed Brown      self.petsclib = '-lpetscts -lpetscsnes -lpetscksp -lpetscdm -lpetscmat -lpetscvec -lpetscsys'
426de8f682fSSatish Balay    self.complibs = self.compilers.flibs+self.compilers.cxxlibs+self.compilers.LIBS.split()
4275a21677cSJed Brown    self.PETSC_WITH_EXTERNAL_LIB = self.libraries.toStringNoDupes(['-L${PETSC_DIR}/${PETSC_ARCH}/lib', self.petsclib]+self.packagelibs+self.complibs)
428de8f682fSSatish Balay    self.PETSC_EXTERNAL_LIB_BASIC = self.libraries.toStringNoDupes(self.packagelibs+self.complibs)
429de8f682fSSatish Balay
430de8f682fSSatish Balay    self.addMakeMacro('PETSC_EXTERNAL_LIB_BASIC',self.PETSC_EXTERNAL_LIB_BASIC)
4315a21677cSJed Brown    allincludes = petscincludes + includes
4325a21677cSJed Brown    allincludes_install = petscincludes_install + includes
4335a21677cSJed Brown    self.PETSC_CC_INCLUDES = self.headers.toStringNoDupes(allincludes)
4345a21677cSJed Brown    self.PETSC_CC_INCLUDES_INSTALL = self.headers.toStringNoDupes(allincludes_install)
4355a21677cSJed Brown    self.addMakeMacro('PETSC_CC_INCLUDES',self.PETSC_CC_INCLUDES)
4365a21677cSJed Brown    self.addMakeMacro('PETSC_CC_INCLUDES_INSTALL', self.PETSC_CC_INCLUDES_INSTALL)
437cbd5cc15SBarry Smith    if hasattr(self.compilers, 'FC'):
4385a21677cSJed Brown      def modinc(includes):
4390542e31aSBarry Smith        return includes if self.fortran.fortranIsF90 else []
4405a21677cSJed Brown      self.addMakeMacro('PETSC_FC_INCLUDES',self.headers.toStringNoDupes(allincludes,modinc(allincludes)))
4415a21677cSJed Brown      self.addMakeMacro('PETSC_FC_INCLUDES_INSTALL',self.headers.toStringNoDupes(allincludes_install,modinc(allincludes_install)))
442f8833479SBarry Smith
4435bb5b263SMatthew G. Knepley    self.addDefine('LIB_DIR','"'+os.path.join(self.installdir.dir,'lib')+'"')
444f8833479SBarry Smith
4450f3b21c2SBarry Smith    if self.framework.argDB['with-single-library']:
4460f3b21c2SBarry Smith      # overrides the values set in conf/variables
4470f3b21c2SBarry Smith      self.addMakeMacro('LIBNAME','${INSTALL_LIB_DIR}/libpetsc.${AR_LIB_SUFFIX}')
44857cb31baSSatish Balay      self.addMakeMacro('SHLIBS','libpetsc')
449bccf1c12SBarry Smith      self.addMakeMacro('PETSC_LIB_BASIC','-lpetsc')
450797063a9SSatish Balay      self.addMakeMacro('PETSC_KSP_LIB_BASIC','-lpetsc')
451797063a9SSatish Balay      self.addMakeMacro('PETSC_TS_LIB_BASIC','-lpetsc')
452b0a7d7e7SSatish Balay      self.addMakeMacro('PETSC_TAO_LIB_BASIC','-lpetsc')
453de8f682fSSatish Balay      self.addMakeMacro('PETSC_WITH_EXTERNAL_LIB',self.PETSC_WITH_EXTERNAL_LIB)
454bb84e0fdSBarry Smith      self.addDefine('USE_SINGLE_LIBRARY', '1')
4552df986feSBarry Smith      if self.sharedlibraries.useShared:
456ea820d49SSatish Balay        self.addMakeMacro('PETSC_SYS_LIB','${C_SH_LIB_PATH} ${PETSC_WITH_EXTERNAL_LIB}')
457ea820d49SSatish Balay        self.addMakeMacro('PETSC_VEC_LIB','${C_SH_LIB_PATH} ${PETSC_WITH_EXTERNAL_LIB}')
458ea820d49SSatish Balay        self.addMakeMacro('PETSC_MAT_LIB','${C_SH_LIB_PATH} ${PETSC_WITH_EXTERNAL_LIB}')
459ea820d49SSatish Balay        self.addMakeMacro('PETSC_DM_LIB','${C_SH_LIB_PATH} ${PETSC_WITH_EXTERNAL_LIB}')
460ea820d49SSatish Balay        self.addMakeMacro('PETSC_KSP_LIB','${C_SH_LIB_PATH} ${PETSC_WITH_EXTERNAL_LIB}')
461ea820d49SSatish Balay        self.addMakeMacro('PETSC_SNES_LIB','${C_SH_LIB_PATH} ${PETSC_WITH_EXTERNAL_LIB}')
462ea820d49SSatish Balay        self.addMakeMacro('PETSC_TS_LIB','${C_SH_LIB_PATH} ${PETSC_WITH_EXTERNAL_LIB}')
463b0a7d7e7SSatish Balay        self.addMakeMacro('PETSC_TAO_LIB','${C_SH_LIB_PATH} ${PETSC_WITH_EXTERNAL_LIB}')
464fdb87e33SJed Brown        self.addMakeMacro('PETSC_CHARACTERISTIC_LIB','${C_SH_LIB_PATH} ${PETSC_WITH_EXTERNAL_LIB}')
465ea820d49SSatish Balay        self.addMakeMacro('PETSC_LIB','${C_SH_LIB_PATH} ${PETSC_WITH_EXTERNAL_LIB}')
466ea820d49SSatish Balay        self.addMakeMacro('PETSC_CONTRIB','${C_SH_LIB_PATH} ${PETSC_WITH_EXTERNAL_LIB}')
4672df986feSBarry Smith      else:
468ea820d49SSatish Balay        self.addMakeMacro('PETSC_SYS_LIB','${PETSC_WITH_EXTERNAL_LIB}')
469ea820d49SSatish Balay        self.addMakeMacro('PETSC_VEC_LIB','${PETSC_WITH_EXTERNAL_LIB}')
470ea820d49SSatish Balay        self.addMakeMacro('PETSC_MAT_LIB','${PETSC_WITH_EXTERNAL_LIB}')
471ea820d49SSatish Balay        self.addMakeMacro('PETSC_DM_LIB','${PETSC_WITH_EXTERNAL_LIB}')
472ea820d49SSatish Balay        self.addMakeMacro('PETSC_KSP_LIB','${PETSC_WITH_EXTERNAL_LIB}')
473ea820d49SSatish Balay        self.addMakeMacro('PETSC_SNES_LIB','${PETSC_WITH_EXTERNAL_LIB}')
474ea820d49SSatish Balay        self.addMakeMacro('PETSC_TS_LIB','${PETSC_WITH_EXTERNAL_LIB}')
475b0a7d7e7SSatish Balay        self.addMakeMacro('PETSC_TAO_LIB','${PETSC_WITH_EXTERNAL_LIB}')
476fdb87e33SJed Brown        self.addMakeMacro('PETSC_CHARACTERISTIC_LIB','${PETSC_WITH_EXTERNAL_LIB}')
477ea820d49SSatish Balay        self.addMakeMacro('PETSC_LIB','${PETSC_WITH_EXTERNAL_LIB}')
478ea820d49SSatish Balay        self.addMakeMacro('PETSC_CONTRIB','${PETSC_WITH_EXTERNAL_LIB}')
4790f3b21c2SBarry Smith
480f8833479SBarry Smith    if not os.path.exists(os.path.join(self.petscdir.dir,self.arch.arch,'lib')):
481f8833479SBarry Smith      os.makedirs(os.path.join(self.petscdir.dir,self.arch.arch,'lib'))
482f8833479SBarry Smith
48313f9d092SSatish Balay# add a makefile endtry for display
48413f9d092SSatish Balay    if self.framework.argDB['with-display']:
48513f9d092SSatish Balay      self.addMakeMacro('DISPLAY',self.framework.argDB['with-display'])
48613f9d092SSatish Balay
487f8833479SBarry Smith    # add a makefile entry for configure options
488f8833479SBarry Smith    self.addMakeMacro('CONFIGURE_OPTIONS', self.framework.getOptionsString(['configModules', 'optionsModule']).replace('\"','\\"'))
489a0c7f9aaSSamuel Khuvis
490a0c7f9aaSSamuel Khuvis    if self.framework.argDB['with-tau-perfstubs']:
491a0c7f9aaSSamuel Khuvis      self.addDefine('HAVE_TAU_PERFSTUBS',1)
492f8833479SBarry Smith    return
493f8833479SBarry Smith
494f8833479SBarry Smith  def dumpConfigInfo(self):
495f8833479SBarry Smith    import time
496c6ef1b5bSJed Brown    fd = open(os.path.join(self.arch.arch,'include','petscconfiginfo.h'),'w')
497dc25a686SPierre Jolivet    fd.write('static const char *petscconfigureoptions = "'+self.framework.getOptionsString(['configModules', 'optionsModule']).replace('\"','\\"').replace('\\ ','\\\\ ')+'";\n')
498f8833479SBarry Smith    fd.close()
499f8833479SBarry Smith    return
500f8833479SBarry Smith
5012a4161d9SMatthew G Knepley  def dumpMachineInfo(self):
5022a4161d9SMatthew G Knepley    import platform
503a970bd74SBernhard M. Wiedemann    import datetime
5042a4161d9SMatthew G Knepley    import time
50540373944SSatish Balay    import script
506ca77dbeeSGeoffrey Irving    def escape(s):
507e08ecd42SSatish Balay      return s.replace('"',r'\"').replace(r'\ ',r'\\ ') # novermin
508c6ef1b5bSJed Brown    fd = open(os.path.join(self.arch.arch,'include','petscmachineinfo.h'),'w')
5092a4161d9SMatthew G Knepley    fd.write('static const char *petscmachineinfo = \"\\n\"\n')
5102a4161d9SMatthew G Knepley    fd.write('\"-----------------------------------------\\n\"\n')
511a970bd74SBernhard M. Wiedemann    buildhost = platform.node()
512a970bd74SBernhard M. Wiedemann    if os.environ.get('SOURCE_DATE_EPOCH'):
513a970bd74SBernhard M. Wiedemann      buildhost = "reproducible"
514a970bd74SBernhard M. Wiedemann    buildtime = datetime.datetime.utcfromtimestamp(int(os.environ.get('SOURCE_DATE_EPOCH', time.time())))
515a970bd74SBernhard M. Wiedemann    fd.write('\"Libraries compiled on %s on %s \\n\"\n' % (buildtime, buildhost))
51660acdfe7SSatish Balay    fd.write('\"Machine characteristics: %s\\n\"\n' % (platform.platform()))
5175188cb68SSatish Balay    fd.write('\"Using PETSc directory: %s\\n\"\n' % (escape(self.installdir.petscDir)))
5185188cb68SSatish Balay    fd.write('\"Using PETSc arch: %s\\n\"\n' % (escape(self.installdir.petscArch)))
519cdec380aSBarry Smith    fd.write('\"-----------------------------------------\\n\";\n')
5202a4161d9SMatthew G Knepley    fd.write('static const char *petsccompilerinfo = \"\\n\"\n')
5212a4161d9SMatthew G Knepley    self.setCompilers.pushLanguage(self.languages.clanguage)
5225f27b2e0SBarry Smith    fd.write('\"Using C compiler: %s %s \\n\"\n' % (escape(self.setCompilers.getCompiler()), escape(self.setCompilers.getCompilerFlags())))
5232a4161d9SMatthew G Knepley    self.setCompilers.popLanguage()
5248782282cSMatthew G Knepley    if hasattr(self.compilers, 'FC'):
5252a4161d9SMatthew G Knepley      self.setCompilers.pushLanguage('FC')
5265f27b2e0SBarry Smith      fd.write('\"Using Fortran compiler: %s %s  %s\\n\"\n' % (escape(self.setCompilers.getCompiler()), escape(self.setCompilers.getCompilerFlags()), escape(self.setCompilers.CPPFLAGS)))
5272a4161d9SMatthew G Knepley      self.setCompilers.popLanguage()
528cdec380aSBarry Smith    fd.write('\"-----------------------------------------\\n\";\n')
5292a4161d9SMatthew G Knepley    fd.write('static const char *petsccompilerflagsinfo = \"\\n\"\n')
5305a21677cSJed Brown    fd.write('\"Using include paths: %s\\n\"\n' % (escape(self.PETSC_CC_INCLUDES_INSTALL.replace('${PETSC_DIR}', self.installdir.petscDir))))
531cdec380aSBarry Smith    fd.write('\"-----------------------------------------\\n\";\n')
5322a4161d9SMatthew G Knepley    fd.write('static const char *petsclinkerinfo = \"\\n\"\n')
5332a4161d9SMatthew G Knepley    self.setCompilers.pushLanguage(self.languages.clanguage)
534ca77dbeeSGeoffrey Irving    fd.write('\"Using C linker: %s\\n\"\n' % (escape(self.setCompilers.getLinker())))
5352a4161d9SMatthew G Knepley    self.setCompilers.popLanguage()
5368782282cSMatthew G Knepley    if hasattr(self.compilers, 'FC'):
5372a4161d9SMatthew G Knepley      self.setCompilers.pushLanguage('FC')
538ca77dbeeSGeoffrey Irving      fd.write('\"Using Fortran linker: %s\\n\"\n' % (escape(self.setCompilers.getLinker())))
5392a4161d9SMatthew G Knepley      self.setCompilers.popLanguage()
5405188cb68SSatish 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)))
541cdec380aSBarry Smith    fd.write('\"-----------------------------------------\\n\";\n')
5422a4161d9SMatthew G Knepley    fd.close()
5432a4161d9SMatthew G Knepley    return
544b2843cf1SBarry Smith
545b2843cf1SBarry Smith  def configurePrefetch(self):
546b2843cf1SBarry Smith    '''Sees if there are any prefetch functions supported'''
5477fca349cSMatthew G. Knepley    if config.setCompilers.Configure.isSolaris(self.log) or self.framework.argDB['with-ios'] or not self.framework.argDB['with-prefetch']:
54893f78423SSatish Balay      self.addDefine('Prefetch(a,b,c)', ' ')
54993f78423SSatish Balay      return
550ec284106SBarry Smith    self.pushLanguage(self.languages.clanguage)
55110699583SJed Brown    if self.checkLink('#include <xmmintrin.h>', 'void *v = 0;_mm_prefetch((const char*)v,_MM_HINT_NTA);\n'):
55250d8bf02SJed Brown      # The Intel Intrinsics manual [1] specifies the prototype
55350d8bf02SJed Brown      #
55450d8bf02SJed Brown      #   void _mm_prefetch(char const *a, int sel);
55550d8bf02SJed Brown      #
55650d8bf02SJed Brown      # but other vendors seem to insist on using subtly different
55750d8bf02SJed Brown      # prototypes, including void* for the pointer, and an enum for
55850d8bf02SJed Brown      # sel.  These are both reasonable changes, but negatively impact
55950d8bf02SJed Brown      # portability.
56050d8bf02SJed Brown      #
561a8d69d7bSBarry Smith      # [1] https://software.intel.com/file/6373
56250d8bf02SJed Brown      self.addDefine('HAVE_XMMINTRIN_H', 1)
56350d8bf02SJed Brown      self.addDefine('Prefetch(a,b,c)', '_mm_prefetch((const char*)(a),(c))')
56450d8bf02SJed Brown      self.addDefine('PREFETCH_HINT_NTA', '_MM_HINT_NTA')
56550d8bf02SJed Brown      self.addDefine('PREFETCH_HINT_T0',  '_MM_HINT_T0')
56650d8bf02SJed Brown      self.addDefine('PREFETCH_HINT_T1',  '_MM_HINT_T1')
56750d8bf02SJed Brown      self.addDefine('PREFETCH_HINT_T2',  '_MM_HINT_T2')
56850d8bf02SJed Brown    elif self.checkLink('#include <xmmintrin.h>', 'void *v = 0;_mm_prefetch(v,_MM_HINT_NTA);\n'):
56950d8bf02SJed Brown      self.addDefine('HAVE_XMMINTRIN_H', 1)
57050d8bf02SJed Brown      self.addDefine('Prefetch(a,b,c)', '_mm_prefetch((const void*)(a),(c))')
57150d8bf02SJed Brown      self.addDefine('PREFETCH_HINT_NTA', '_MM_HINT_NTA')
57250d8bf02SJed Brown      self.addDefine('PREFETCH_HINT_T0',  '_MM_HINT_T0')
57350d8bf02SJed Brown      self.addDefine('PREFETCH_HINT_T1',  '_MM_HINT_T1')
57450d8bf02SJed Brown      self.addDefine('PREFETCH_HINT_T2',  '_MM_HINT_T2')
57510699583SJed Brown    elif self.checkLink('', 'void *v = 0;__builtin_prefetch(v,0,0);\n'):
57610699583SJed Brown      # From GCC docs: void __builtin_prefetch(const void *addr,int rw,int locality)
57710699583SJed Brown      #
57810699583SJed Brown      #   The value of rw is a compile-time constant one or zero; one
57910699583SJed Brown      #   means that the prefetch is preparing for a write to the memory
58010699583SJed Brown      #   address and zero, the default, means that the prefetch is
58110699583SJed Brown      #   preparing for a read. The value locality must be a compile-time
58210699583SJed Brown      #   constant integer between zero and three. A value of zero means
58310699583SJed Brown      #   that the data has no temporal locality, so it need not be left
58410699583SJed Brown      #   in the cache after the access. A value of three means that the
58510699583SJed Brown      #   data has a high degree of temporal locality and should be left
58610699583SJed Brown      #   in all levels of cache possible. Values of one and two mean,
58710699583SJed Brown      #   respectively, a low or moderate degree of temporal locality.
58810699583SJed Brown      #
58910699583SJed Brown      # Here we adopt Intel's x86/x86-64 naming scheme for the locality
59010699583SJed Brown      # hints.  Using macros for these values in necessary since some
59110699583SJed Brown      # compilers require an enum.
59210699583SJed Brown      self.addDefine('Prefetch(a,b,c)', '__builtin_prefetch((a),(b),(c))')
59310699583SJed Brown      self.addDefine('PREFETCH_HINT_NTA', '0')
59410699583SJed Brown      self.addDefine('PREFETCH_HINT_T0',  '3')
59510699583SJed Brown      self.addDefine('PREFETCH_HINT_T1',  '2')
59610699583SJed Brown      self.addDefine('PREFETCH_HINT_T2',  '1')
597b2843cf1SBarry Smith    else:
598b2843cf1SBarry Smith      self.addDefine('Prefetch(a,b,c)', ' ')
5997d490b44SBarry Smith    self.popLanguage()
600b2843cf1SBarry Smith
60149fe22e6SSatish Balay  def delGenFiles(self):
60249fe22e6SSatish Balay    '''Delete generated files'''
60349fe22e6SSatish Balay    delfile = os.path.join(self.arch.arch,'lib','petsc','conf','files')
60449fe22e6SSatish Balay    try:
60549fe22e6SSatish Balay      os.unlink(delfile)
60649fe22e6SSatish Balay    except: pass
60749fe22e6SSatish Balay
60809bc878fSSatish Balay  def configureAtoll(self):
60909bc878fSSatish Balay    '''Checks if atoll exists'''
61022164b4cSPierre 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'):
61109bc878fSSatish Balay       self.addDefine('HAVE_ATOLL', '1')
61209bc878fSSatish Balay
6132400fdedSBarry Smith  def configureUnused(self):
6142400fdedSBarry Smith    '''Sees if __attribute((unused)) is supported'''
6151adaff47SSean Farley    if self.framework.argDB['with-ios']:
6162400fdedSBarry Smith      self.addDefine('UNUSED', ' ')
6172400fdedSBarry Smith      return
6182400fdedSBarry Smith    self.pushLanguage(self.languages.clanguage)
61922164b4cSPierre 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'):
6202400fdedSBarry Smith      self.addDefine('UNUSED', '__attribute((unused))')
6212400fdedSBarry Smith    else:
6222400fdedSBarry Smith      self.addDefine('UNUSED', ' ')
6232400fdedSBarry Smith    self.popLanguage()
6242400fdedSBarry Smith
62598ed35c3SBarry Smith  def configureIsatty(self):
62698ed35c3SBarry Smith    '''Check if the Unix C function isatty() works correctly
62798ed35c3SBarry Smith       Actually just assumes it does not work correctly on batch systems'''
62898ed35c3SBarry Smith    if not self.framework.argDB['with-batch']:
62998ed35c3SBarry Smith      self.addDefine('USE_ISATTY',1)
63098ed35c3SBarry Smith
6311ef8df7fSJed Brown  def configureDeprecated(self):
6321ef8df7fSJed Brown    '''Check if __attribute((deprecated)) is supported'''
6337efe37a1SJacob Faibussowitsch    def checkDeprecated(macro_base, src, is_intel):
6347efe37a1SJacob Faibussowitsch      '''
6357efe37a1SJacob Faibussowitsch      run through the various attribute deprecated combinations and define MACRO_BAS(why) to the result
6367efe37a1SJacob Faibussowitsch      it if it compiles.
6377efe37a1SJacob Faibussowitsch
6387efe37a1SJacob Faibussowitsch      If none of the combos work, defines MACRO_BASE(why) as empty
6397efe37a1SJacob Faibussowitsch      '''
6407efe37a1SJacob Faibussowitsch      full_macro_name = macro_base + '(why)'
6417efe37a1SJacob Faibussowitsch      for prefix in ('__attribute__', '__attribute','__declspec'):
6427efe37a1SJacob Faibussowitsch        if prefix == '__declspec':
6437efe37a1SJacob Faibussowitsch          # declspec does not have an extra set of brackets around the arguments
6447efe37a1SJacob Faibussowitsch          attr_bodies = ('deprecated(why)', 'deprecated')
6451ef8df7fSJed Brown        else:
6467efe37a1SJacob Faibussowitsch          attr_bodies = ('(deprecated(why))', '(deprecated)')
6477efe37a1SJacob Faibussowitsch
6487efe37a1SJacob Faibussowitsch        for attr_body in attr_bodies:
6497efe37a1SJacob Faibussowitsch          attr_def = '{}({})'.format(prefix, attr_body)
6507efe37a1SJacob Faibussowitsch          test_src = '\n'.join((
6517efe37a1SJacob Faibussowitsch            '#define {} {}'.format(full_macro_name, attr_def),
6527efe37a1SJacob Faibussowitsch            src.format(macro_base + '("asdasdadsasd")')
6537efe37a1SJacob Faibussowitsch          ))
6547efe37a1SJacob Faibussowitsch          if self.checkCompile(test_src, ''):
6557efe37a1SJacob Faibussowitsch            self.logPrint('configureDeprecated: \'{}\' appears to work'.format(attr_def))
6567efe37a1SJacob Faibussowitsch            if is_intel and '(why)' in attr_body:
6577efe37a1SJacob 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))
6587efe37a1SJacob Faibussowitsch              self.logPrint('*** WE WILL THEREFORE REJECT \'{}\' AND CONTINUE TESTING ***'.format(attr_def))
6597efe37a1SJacob Faibussowitsch              continue
6607efe37a1SJacob Faibussowitsch            self.addDefine(full_macro_name, attr_def)
6617efe37a1SJacob Faibussowitsch            return
6627efe37a1SJacob Faibussowitsch
6637efe37a1SJacob Faibussowitsch      self.addDefine(full_macro_name, ' ')
6647efe37a1SJacob Faibussowitsch      return
6657efe37a1SJacob Faibussowitsch
6667efe37a1SJacob Faibussowitsch    lang = self.languages.clanguage
6677efe37a1SJacob Faibussowitsch    with self.Language(lang):
6687efe37a1SJacob Faibussowitsch      is_intel = self.setCompilers.isIntel(self.getCompiler(lang=lang), self.log)
6697efe37a1SJacob Faibussowitsch      checkDeprecated('DEPRECATED_FUNCTION', '{} int myfunc(void) {{ return 1; }}', is_intel)
6707efe37a1SJacob Faibussowitsch      checkDeprecated('DEPRECATED_TYPEDEF', 'typedef int my_int {};', is_intel)
6717efe37a1SJacob Faibussowitsch      checkDeprecated('DEPRECATED_ENUM', 'enum E {{ oldval {}, newval }};', is_intel)
6727efe37a1SJacob Faibussowitsch      # I was unable to make a CPP macro that takes the old and new values as separate
6737efe37a1SJacob Faibussowitsch      # arguments and builds the message needed by _Pragma hence the deprecation message is
6747efe37a1SJacob Faibussowitsch      # handled as it is
67505de396fSBarry Smith      if self.checkCompile('#define TEST _Pragma("GCC warning \"Testing _Pragma\"") value'):
67605de396fSBarry Smith        self.addDefine('DEPRECATED_MACRO(why)', '_Pragma(why)')
67705de396fSBarry Smith      else:
67805de396fSBarry Smith        self.addDefine('DEPRECATED_MACRO(why)', ' ')
6791ef8df7fSJed Brown
68018f41590SBarry Smith  def configureAlign(self):
6817b7fc14bSLisandro Dalcin    '''Check if __attribute(aligned) is supported'''
6827b7fc14bSLisandro Dalcin    code = '''\
683752d89a4SSatish Balaystruct mystruct {int myint;} __attribute((aligned(16)));
6847b7fc14bSLisandro Dalcinchar assert_aligned[(sizeof(struct mystruct)==16)*2-1];
685752d89a4SSatish Balay'''
686752d89a4SSatish Balay    self.pushLanguage(self.languages.clanguage)
6877b7fc14bSLisandro Dalcin    if self.checkCompile(code):
688752d89a4SSatish Balay      self.addDefine('ATTRIBUTEALIGNED(size)', '__attribute((aligned(size)))')
689752d89a4SSatish Balay      self.addDefine('HAVE_ATTRIBUTEALIGNED', 1)
690752d89a4SSatish Balay    else:
6917b7fc14bSLisandro Dalcin      self.framework.logPrint('Incorrect attribute(aligned)')
692752d89a4SSatish Balay      self.addDefine('ATTRIBUTEALIGNED(size)', ' ')
6937b7fc14bSLisandro Dalcin    self.popLanguage()
694752d89a4SSatish Balay    return
69518f41590SBarry Smith
6969800092aSJed Brown  def configureExpect(self):
6979800092aSJed Brown    '''Sees if the __builtin_expect directive is supported'''
6989800092aSJed Brown    self.pushLanguage(self.languages.clanguage)
6999800092aSJed Brown    if self.checkLink('', 'if (__builtin_expect(0,1)) return 1;'):
7009800092aSJed Brown      self.addDefine('HAVE_BUILTIN_EXPECT', 1)
7019800092aSJed Brown    self.popLanguage()
7029800092aSJed Brown
70353c77d0aSJed Brown  def configureFunctionName(self):
704fbfcfee5SBarry Smith    '''Sees if the compiler supports __func__ or a variant.'''
7051ec50b02SJed Brown    def getFunctionName(lang):
706fbfcfee5SBarry Smith      name = '"unknown"'
7071ec50b02SJed Brown      self.pushLanguage(lang)
708b6ff4c76SKarl Rupp      for fname in ['__func__','__FUNCTION__','__extension__ __func__']:
7090117e5a1SSatish Balay        code = "if ("+fname+"[0] != 'm') return 1;"
7100117e5a1SSatish Balay        if self.checkCompile('',code) and self.checkLink('',code):
7110117e5a1SSatish Balay          name = fname
7120117e5a1SSatish Balay          break
7131ec50b02SJed Brown      self.popLanguage()
7141ec50b02SJed Brown      return name
7151ec50b02SJed Brown    langs = []
716628773c9SSatish Balay
717628773c9SSatish Balay    self.addDefine('FUNCTION_NAME_C', getFunctionName('C'))
7185f6e5f85SSatish Balay    if hasattr(self.compilers, 'CXX'):
719628773c9SSatish Balay      self.addDefine('FUNCTION_NAME_CXX', getFunctionName('Cxx'))
72053c77d0aSJed Brown
721753ebd1dSJed Brown  def configureIntptrt(self):
722f18a5f7eSJacob Faibussowitsch    '''Determine what to use for uintptr_t and intptr_t'''
723753ebd1dSJed Brown    def staticAssertSizeMatchesVoidStar(inc,typename):
724753ebd1dSJed Brown      # The declaration is an error if either array size is negative.
725753ebd1dSJed 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
726d26187a0SJed Brown      return self.checkCompile(inc, ('#define STATIC_ASSERT(cond) char negative_length_if_false[2*(!!(cond))-1]\n'
727979939cdSSatish Balay                                     + 'STATIC_ASSERT(sizeof(void*) == sizeof(%s));'%typename))
728f18a5f7eSJacob Faibussowitsch
729f18a5f7eSJacob Faibussowitsch    def generate_uintptr_guesses():
730f18a5f7eSJacob Faibussowitsch      for suff in ('max', '64', '32', '16'):
731f18a5f7eSJacob Faibussowitsch        yield '#include <stdint.h>', 'uint{}_t'.format(suff), 'PRIx{}'.format(suff.upper())
732f18a5f7eSJacob Faibussowitsch      yield '#include <stdlib.h>\n#include <string.h>', 'size_t', 'zx'
733f18a5f7eSJacob Faibussowitsch      yield '', 'unsigned long long', 'llx'
734f18a5f7eSJacob Faibussowitsch      yield '', 'unsigned long', 'lx'
735f18a5f7eSJacob Faibussowitsch      yield '', 'unsigned', 'x'
736f18a5f7eSJacob Faibussowitsch
737f18a5f7eSJacob Faibussowitsch    def generate_intptr_guesses():
738f18a5f7eSJacob Faibussowitsch      for suff in ('max', '64', '32', '16'):
739f18a5f7eSJacob Faibussowitsch        yield '#include <stdint.h>', 'int{}_t'.format(suff), 'PRIx{}'.format(suff.upper())
740f18a5f7eSJacob Faibussowitsch      yield '', 'long long', 'llx'
741f18a5f7eSJacob Faibussowitsch      yield '', 'long', 'lx'
742f18a5f7eSJacob Faibussowitsch      yield '', 'int', 'x'
743f18a5f7eSJacob Faibussowitsch
744f18a5f7eSJacob Faibussowitsch    def check(default_typename, generator):
745f18a5f7eSJacob Faibussowitsch      macro_name = default_typename.upper()
746f18a5f7eSJacob Faibussowitsch      with self.Language(self.languages.clanguage):
747f18a5f7eSJacob Faibussowitsch        if self.checkCompile(
748f18a5f7eSJacob Faibussowitsch            '#include <stdint.h>',
749f18a5f7eSJacob Faibussowitsch            'int x; {type_name} i = ({type_name})&x; (void)i'.format(type_name=default_typename)
750f18a5f7eSJacob Faibussowitsch        ):
751f18a5f7eSJacob Faibussowitsch          typename     = default_typename
752f18a5f7eSJacob Faibussowitsch          print_format = 'PRIxPTR'
753d26187a0SJed Brown        else:
754f18a5f7eSJacob Faibussowitsch          for include, typename, print_format in generator():
755f18a5f7eSJacob Faibussowitsch            if staticAssertSizeMatchesVoidStar(include, typename):
756f18a5f7eSJacob Faibussowitsch              break
757f18a5f7eSJacob Faibussowitsch          else:
758f18a5f7eSJacob Faibussowitsch            raise RuntimeError('Could not find any {} type matching void*'.format(macro_name))
759f18a5f7eSJacob Faibussowitsch      self.addDefine(macro_name         , typename)
760f18a5f7eSJacob Faibussowitsch      self.addDefine(macro_name + '_FMT', '\"#\" ' + print_format)
761f18a5f7eSJacob Faibussowitsch      return
762f18a5f7eSJacob Faibussowitsch
763f18a5f7eSJacob Faibussowitsch    check('uintptr_t', generate_uintptr_guesses)
764f18a5f7eSJacob Faibussowitsch    check('intptr_t', generate_intptr_guesses)
765f18a5f7eSJacob Faibussowitsch    return
766753ebd1dSJed Brown
767ed938b00SJed Brown  def configureRTLDDefault(self):
7687b65ca21SBarry Smith    '''Check for dynamic library feature'''
769bfef2c86SBarry Smith    if self.checkCompile('#include <dlfcn.h>\n void *ptr =  RTLD_DEFAULT;'):
7709fb7294dSPierre Jolivet      self.addDefine('HAVE_RTLD_DEFAULT','1')
771f8833479SBarry Smith    return
772f8833479SBarry Smith
773f8833479SBarry Smith  def configureSolaris(self):
774f8833479SBarry Smith    '''Solaris specific stuff'''
775f8833479SBarry Smith    if os.path.isdir(os.path.join('/usr','ucblib')):
776f8833479SBarry Smith      try:
777f8833479SBarry Smith        flag = getattr(self.setCompilers, self.language[-1]+'SharedLinkerFlag')
778f8833479SBarry Smith      except AttributeError:
779f8833479SBarry Smith        flag = None
780f8833479SBarry Smith      if flag is None:
781f8833479SBarry Smith        self.compilers.LIBS += ' -L/usr/ucblib'
782f8833479SBarry Smith      else:
783f8833479SBarry Smith        self.compilers.LIBS += ' '+flag+'/usr/ucblib'
784f8833479SBarry Smith    return
785f8833479SBarry Smith
7860f64ec89SBarry Smith  def configureDarwin(self):
7870f64ec89SBarry Smith    '''Log brew configuration for Apple systems'''
7880f64ec89SBarry Smith    try:
7890f64ec89SBarry Smith      self.executeShellCommand(['brew', 'config'], log = self.log)
7900f64ec89SBarry Smith      self.executeShellCommand(['brew', 'info', 'gcc'], log = self.log)
7910f64ec89SBarry Smith    except:
7920f64ec89SBarry Smith      pass
7930f64ec89SBarry Smith    return
7940f64ec89SBarry Smith
795f8833479SBarry Smith  def configureLinux(self):
796f8833479SBarry Smith    '''Linux specific stuff'''
7979f15855cSMatthew G Knepley    # TODO: Test for this by mallocing an odd number of floats and checking the address
798f8833479SBarry Smith    self.addDefine('HAVE_DOUBLE_ALIGN_MALLOC', 1)
799f8833479SBarry Smith    return
800f8833479SBarry Smith
801f8833479SBarry Smith  def configureWin32(self):
802f8833479SBarry Smith    '''Win32 non-cygwin specific stuff'''
803f8833479SBarry Smith    kernel32=0
8044e8afd12SMosè Giordano    if self.libraries.add('Kernel32.lib','GetComputerName',prototype='#include <windows.h>', call='GetComputerName(NULL,NULL);'):
805f8833479SBarry Smith      self.addDefine('HAVE_WINDOWS_H',1)
806f8833479SBarry Smith      self.addDefine('HAVE_GETCOMPUTERNAME',1)
807f8833479SBarry Smith      kernel32=1
8084e8afd12SMosè Giordano    elif self.libraries.add('kernel32','GetComputerName',prototype='#include <windows.h>', call='GetComputerName(NULL,NULL);'):
809f8833479SBarry Smith      self.addDefine('HAVE_WINDOWS_H',1)
810f8833479SBarry Smith      self.addDefine('HAVE_GETCOMPUTERNAME',1)
811f8833479SBarry Smith      kernel32=1
812f8833479SBarry Smith    if kernel32:
813eed94e11SSatish Balay      if self.framework.argDB['with-windows-graphics']:
814eed94e11SSatish Balay        self.addDefine('USE_WINDOWS_GRAPHICS',1)
8154e8afd12SMosè Giordano      if self.checkLink('#include <windows.h>','LoadLibrary(0)'):
816f8833479SBarry Smith        self.addDefine('HAVE_LOADLIBRARY',1)
8174e8afd12SMosè Giordano      if self.checkLink('#include <windows.h>','GetProcAddress(0,0)'):
818b50f6d9eSLisandro Dalcin        self.addDefine('HAVE_GETPROCADDRESS',1)
8194e8afd12SMosè Giordano      if self.checkLink('#include <windows.h>','FreeLibrary(0)'):
820b50f6d9eSLisandro Dalcin        self.addDefine('HAVE_FREELIBRARY',1)
8214e8afd12SMosè Giordano      if self.checkLink('#include <windows.h>','GetLastError()'):
822a21658a3SLisandro Dalcin        self.addDefine('HAVE_GETLASTERROR',1)
8234e8afd12SMosè Giordano      if self.checkLink('#include <windows.h>','SetLastError(0)'):
824a21658a3SLisandro Dalcin        self.addDefine('HAVE_SETLASTERROR',1)
8254e8afd12SMosè Giordano      if self.checkLink('#include <windows.h>\n','QueryPerformanceCounter(0);\n'):
826bea725cfSBarry Smith        self.addDefine('USE_MICROSOFT_TIME',1)
8274e8afd12SMosè Giordano    if self.libraries.add('Advapi32.lib','GetUserName',prototype='#include <windows.h>', call='GetUserName(NULL,NULL);'):
828f8833479SBarry Smith      self.addDefine('HAVE_GET_USER_NAME',1)
8294e8afd12SMosè Giordano    elif self.libraries.add('advapi32','GetUserName',prototype='#include <windows.h>', call='GetUserName(NULL,NULL);'):
830f8833479SBarry Smith      self.addDefine('HAVE_GET_USER_NAME',1)
831f8833479SBarry Smith
8324e8afd12SMosè Giordano    if not self.libraries.add('User32.lib','GetDC',prototype='#include <windows.h>',call='GetDC(0);'):
8334e8afd12SMosè Giordano      self.libraries.add('user32','GetDC',prototype='#include <windows.h>',call='GetDC(0);')
8344e8afd12SMosè Giordano    if not self.libraries.add('Gdi32.lib','CreateCompatibleDC',prototype='#include <windows.h>',call='CreateCompatibleDC(0);'):
8354e8afd12SMosè Giordano      self.libraries.add('gdi32','CreateCompatibleDC',prototype='#include <windows.h>',call='CreateCompatibleDC(0);')
836f8833479SBarry Smith
837f8833479SBarry Smith    self.types.check('int32_t', 'int')
83822164b4cSPierre Jolivet    if not self.checkCompile('#include <sys/types.h>\n','uid_t u;\n(void)u'):
839f8833479SBarry Smith      self.addTypedef('int', 'uid_t')
840f8833479SBarry Smith      self.addTypedef('int', 'gid_t')
84122164b4cSPierre 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'):
842f8833479SBarry Smith      self.framework.addDefine('R_OK', '04')
843f8833479SBarry Smith      self.framework.addDefine('W_OK', '02')
844f8833479SBarry Smith      self.framework.addDefine('X_OK', '01')
845f8833479SBarry Smith    if not self.checkLink('#include <sys/stat.h>\n','int a=0;\nif (S_ISDIR(a)){}\n'):
846f8833479SBarry Smith      self.framework.addDefine('S_ISREG(a)', '(((a)&_S_IFMT) == _S_IFREG)')
847f8833479SBarry Smith      self.framework.addDefine('S_ISDIR(a)', '(((a)&_S_IFMT) == _S_IFDIR)')
8484e8afd12SMosè Giordano    if self.checkCompile('#include <windows.h>\n','LARGE_INTEGER a;\nDWORD b=a.u.HighPart;\n'):
849f8833479SBarry Smith      self.addDefine('HAVE_LARGE_INTEGER_U',1)
850f8833479SBarry Smith
851f8833479SBarry Smith    # Windows requires a Binary file creation flag when creating/opening binary files.  Is a better test in order?
8524e8afd12SMosè Giordano    if self.checkCompile('#include <windows.h>\n#include <fcntl.h>\n', 'int flags = O_BINARY;'):
853f8833479SBarry Smith      self.addDefine('HAVE_O_BINARY',1)
854f8833479SBarry Smith
855f8833479SBarry Smith    if self.compilers.CC.find('win32fe') >= 0:
856ad4212abSSatish Balay      self.addDefine('HAVE_WINDOWS_COMPILERS',1)
857f8833479SBarry Smith      self.addDefine('DIR_SEPARATOR','\'\\\\\'')
858f8833479SBarry Smith      self.addDefine('REPLACE_DIR_SEPARATOR','\'/\'')
859f8833479SBarry Smith      self.addDefine('CANNOT_START_DEBUGGER',1)
8605188cb68SSatish Balay      (petscdir,error,status) = self.executeShellCommand('cygpath -w '+self.installdir.petscDir, log = self.log)
86134531a4dSSatish Balay      self.addDefine('DIR','"'+petscdir.replace('\\','\\\\')+'"')
8625188cb68SSatish Balay      (petscdir,error,status) = self.executeShellCommand('cygpath -m '+self.installdir.petscDir, log = self.log)
863e433681fSSatish Balay      self.addMakeMacro('wPETSC_DIR',petscdir)
8644e00a515SSatish Balay      if self.dataFilesPath.datafilespath:
8654e00a515SSatish Balay        (datafilespath,error,status) = self.executeShellCommand('cygpath -m '+self.dataFilesPath.datafilespath, log = self.log)
8664e00a515SSatish Balay        self.addMakeMacro('DATAFILESPATH',datafilespath)
8674e00a515SSatish Balay
868f8833479SBarry Smith    else:
869f8833479SBarry Smith      self.addDefine('REPLACE_DIR_SEPARATOR','\'\\\\\'')
870f8833479SBarry Smith      self.addDefine('DIR_SEPARATOR','\'/\'')
8715188cb68SSatish Balay      self.addDefine('DIR','"'+self.installdir.petscDir+'"')
8725188cb68SSatish Balay      self.addMakeMacro('wPETSC_DIR',self.installdir.petscDir)
8734e00a515SSatish Balay      if self.dataFilesPath.datafilespath:
8744e00a515SSatish Balay        self.addMakeMacro('DATAFILESPATH',self.dataFilesPath.datafilespath)
8755188cb68SSatish Balay    self.addDefine('ARCH','"'+self.installdir.petscArch+'"')
876f8833479SBarry Smith    return
877f8833479SBarry Smith
8787ce81a4bSJacob Faibussowitsch  def configureCoverage(self, lang, extra_coverage_flags=None, extra_debug_flags=None):
8797ce81a4bSJacob Faibussowitsch    """
8807ce81a4bSJacob Faibussowitsch    Check that a compiler accepts code-coverage flags. If the compiler does accept code-coverage flags
8817ce81a4bSJacob Faibussowitsch    try to set debugging flags equivalent to -Og.
8827ce81a4bSJacob Faibussowitsch
8837ce81a4bSJacob Faibussowitsch    Arguments:
8847ce81a4bSJacob Faibussowitsch    - lang: the language to check the coverage flag for
8857ce81a4bSJacob Faibussowitsch    - extra_coverage_flags: a list of extra flags to use when checking the coverage flags
8867ce81a4bSJacob Faibussowitsch    - extra_debug_flags: a list of extra flags to try when setting debug flags
8877ce81a4bSJacob Faibussowitsch
8887ce81a4bSJacob Faibussowitsch    On success:
8897ce81a4bSJacob Faibussowitsch    - defines PETSC_USE_COVERAGE to 1
8907ce81a4bSJacob Faibussowitsch    """
8917ce81a4bSJacob Faibussowitsch    def log_print(msg, *args, **kwargs):
8927ce81a4bSJacob Faibussowitsch      self.logPrint('checkCoverage: '+str(msg), *args, **kwargs)
8937ce81a4bSJacob Faibussowitsch      return
8947ce81a4bSJacob Faibussowitsch
8957ce81a4bSJacob Faibussowitsch    def quoted(string):
8967ce81a4bSJacob Faibussowitsch      return string.join(("'", "'"))
8977ce81a4bSJacob Faibussowitsch
8987ce81a4bSJacob Faibussowitsch    def make_flag_list(default, extra):
8997ce81a4bSJacob Faibussowitsch      ret = [default]
9007ce81a4bSJacob Faibussowitsch      if extra is not None:
9017ce81a4bSJacob Faibussowitsch        assert isinstance(extra, list)
9027ce81a4bSJacob Faibussowitsch        ret.extend(extra)
9037ce81a4bSJacob Faibussowitsch      return ret
9047ce81a4bSJacob Faibussowitsch
9057ce81a4bSJacob Faibussowitsch    log_print('Checking coverage flag for language {}'.format(lang))
9067ce81a4bSJacob Faibussowitsch    if not self.argDB['with-coverage']:
9077ce81a4bSJacob Faibussowitsch      log_print('coverage was disabled from command line or default')
9087ce81a4bSJacob Faibussowitsch      return
9097ce81a4bSJacob Faibussowitsch
9107ce81a4bSJacob Faibussowitsch    compiler  = self.getCompiler(lang=lang)
9117ce81a4bSJacob Faibussowitsch    is_gnuish = self.setCompilers.isGNU(compiler, self.log) or self.setCompilers.isClang(compiler, self.log)
9127ce81a4bSJacob Faibussowitsch
9137ce81a4bSJacob Faibussowitsch    # if not gnuish and we don't have a set of extra flags, bail
9147ce81a4bSJacob Faibussowitsch    if not is_gnuish and extra_coverage_flags is None:
9157ce81a4bSJacob 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)))
9167ce81a4bSJacob Faibussowitsch      return
9177ce81a4bSJacob Faibussowitsch
9187ce81a4bSJacob Faibussowitsch    coverage_flags = make_flag_list('--coverage', extra_coverage_flags)
9197ce81a4bSJacob Faibussowitsch    log_print('Checking set of coverage flags: {}'.format(coverage_flags))
9207ce81a4bSJacob Faibussowitsch
9217ce81a4bSJacob Faibussowitsch    found = 0
9227ce81a4bSJacob Faibussowitsch    with self.Language(lang):
9237ce81a4bSJacob Faibussowitsch      with self.setCompilers.Language(lang):
9247ce81a4bSJacob Faibussowitsch        for flag in coverage_flags:
9257ce81a4bSJacob Faibussowitsch          if self.setCompilers.checkCompilerFlag(flag) and self.checkLink():
9267ce81a4bSJacob Faibussowitsch            # compilerOnly = False, the linker also needs to see the coverage flag
9277ce81a4bSJacob Faibussowitsch            self.setCompilers.insertCompilerFlag(flag, False)
9287ce81a4bSJacob Faibussowitsch            found = 1
9297ce81a4bSJacob Faibussowitsch            break
9307ce81a4bSJacob Faibussowitsch          log_print(
9317ce81a4bSJacob Faibussowitsch            'Compiler {} did not accept coverage flag {}'.format(quoted(compiler), quoted(flag))
9327ce81a4bSJacob Faibussowitsch          )
9337ce81a4bSJacob Faibussowitsch
9347ce81a4bSJacob Faibussowitsch    if not found:
9357ce81a4bSJacob Faibussowitsch      log_print('Compiler {} did not accept ANY coverage flags: {}, bailing!'.format(quoted(compiler), coverage_flags))
9367ce81a4bSJacob Faibussowitsch      return
9377ce81a4bSJacob Faibussowitsch
9387ce81a4bSJacob Faibussowitsch    if not self.functions.haveFunction('__gcov_dump'):
9397ce81a4bSJacob Faibussowitsch      self.functions.checkClassify(['__gcov_dump'])
9407ce81a4bSJacob Faibussowitsch
9417ce81a4bSJacob Faibussowitsch    # now check if we can override the optimization level. It is only kosher to do so if
9427ce81a4bSJacob Faibussowitsch    # the user did not explicitly set the optimization flags (via CFLAGS, CXXFLAGS,
9437ce81a4bSJacob Faibussowitsch    # CXXOPTFLAGS, etc). If they have done so, we sternly warn them about their lapse in
9447ce81a4bSJacob Faibussowitsch    # judgement
9457ce81a4bSJacob Faibussowitsch    with self.Language(lang):
9467ce81a4bSJacob Faibussowitsch      compiler_flags = self.getCompilerFlags()
9477ce81a4bSJacob Faibussowitsch
9487ce81a4bSJacob Faibussowitsch    user_set          = 0
9497ce81a4bSJacob Faibussowitsch    allowed_opt_flags = re.compile(r'|'.join((r'-O[01g]', r'-g[1-9]*')))
9507ce81a4bSJacob Faibussowitsch    for flagsname in [self.getCompilerFlagsName(lang), self.compilerFlags.getOptionalFlagsName(lang)]:
9517ce81a4bSJacob Faibussowitsch      if flagsname in self.argDB:
9527ce81a4bSJacob Faibussowitsch        opt_flags = [
9537ce81a4bSJacob Faibussowitsch          f for f in self.compilerFlags.findOptFlags(compiler_flags) if not allowed_opt_flags.match(f)
9547ce81a4bSJacob Faibussowitsch        ]
9557ce81a4bSJacob Faibussowitsch        if opt_flags:
9567ce81a4bSJacob 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)))
9577ce81a4bSJacob Faibussowitsch          user_set = 1
9587ce81a4bSJacob Faibussowitsch          break
9597ce81a4bSJacob Faibussowitsch
9607ce81a4bSJacob Faibussowitsch    if not user_set:
9617ce81a4bSJacob Faibussowitsch      debug_flags = make_flag_list('-Og', extra_debug_flags)
9627ce81a4bSJacob Faibussowitsch      with self.setCompilers.Language(lang):
9637ce81a4bSJacob Faibussowitsch        for flag in debug_flags:
9647ce81a4bSJacob Faibussowitsch          try:
9657ce81a4bSJacob Faibussowitsch            self.setCompilers.addCompilerFlag(flag)
9667ce81a4bSJacob Faibussowitsch          except RuntimeError:
9677ce81a4bSJacob Faibussowitsch            continue
9687ce81a4bSJacob Faibussowitsch          break
9697ce81a4bSJacob Faibussowitsch
9707ce81a4bSJacob Faibussowitsch    self.addDefine('USE_COVERAGE', 1)
9717ce81a4bSJacob Faibussowitsch    return
9727ce81a4bSJacob Faibussowitsch
9737ce81a4bSJacob Faibussowitsch  def configureCoverageExecutable(self):
9747ce81a4bSJacob Faibussowitsch    """
9757ce81a4bSJacob Faibussowitsch    Check that a code-coverage collecting tool exists and is on PATH.
9767ce81a4bSJacob Faibussowitsch
9777ce81a4bSJacob Faibussowitsch    On success:
9787ce81a4bSJacob Faibussowitsch    - Adds PETSC_COVERAGE_EXEC make macro containing the full path to the coverage tool executable.
9797ce81a4bSJacob Faibussowitsch
9807ce81a4bSJacob Faibussowitsch    Raises RuntimeError if:
9817ce81a4bSJacob Faibussowitsch    - User explicitly requests auto-detection of the coverage tool from command line, and this
9827ce81a4bSJacob Faibussowitsch      routine fails to guess the suitable tool name.
9837ce81a4bSJacob Faibussowitsch    - The routine fails to find the tool, and --with-coverage is true
9847ce81a4bSJacob Faibussowitsch    """
9857ce81a4bSJacob Faibussowitsch    def log_print(msg, *args, **kwargs):
9867ce81a4bSJacob Faibussowitsch      self.logPrint('checkCoverage: '+str(msg), *args, **kwargs)
9877ce81a4bSJacob Faibussowitsch      return
9887ce81a4bSJacob Faibussowitsch
9897ce81a4bSJacob Faibussowitsch    def quoted(string):
9907ce81a4bSJacob Faibussowitsch      return string.join(("'", "'"))
9917ce81a4bSJacob Faibussowitsch
9927ce81a4bSJacob Faibussowitsch    required         = bool(self.argDB['with-coverage'])
9937ce81a4bSJacob Faibussowitsch    arg_opt          = self.argDB['with-coverage-exec']
9947ce81a4bSJacob Faibussowitsch    use_default_path = True
9957ce81a4bSJacob Faibussowitsch    search_path      = ''
9967ce81a4bSJacob Faibussowitsch
9977ce81a4bSJacob Faibussowitsch    log_print('{} to find an executable'.format('REQUIRED' if required else 'NOT required'))
9987ce81a4bSJacob Faibussowitsch    if arg_opt in {'auto', 'default-auto', '1'}:
9997ce81a4bSJacob Faibussowitsch      # detect it based on the C language compiler, hopefully this does not clash!
1000*095fb05fSJacob Faibussowitsch      lang     = self.setCompilers.languages.clanguage
1001*095fb05fSJacob Faibussowitsch      compiler = self.getCompiler(lang=lang)
10027ce81a4bSJacob Faibussowitsch      log_print('User did not explicitly set coverage exec (got {}), trying to auto-detect based on compiler {}'.format(quoted(arg_opt), quoted(compiler)))
10037ce81a4bSJacob Faibussowitsch      if self.setCompilers.isGNU(compiler, self.log):
1004*095fb05fSJacob Faibussowitsch        compiler_version_re = re.compile(r'[gG][cC\+\-]+[0-9]* \(.+\) (\d+)\.(\d+)\.(\d+)')
10057ce81a4bSJacob Faibussowitsch        exec_names          = ['gcov']
10067ce81a4bSJacob Faibussowitsch      elif self.setCompilers.isClang(compiler, self.log):
1007*095fb05fSJacob Faibussowitsch        compiler_version_re = re.compile(r'clang version (\d+)\.(\d+)\.(\d+)')
10087ce81a4bSJacob Faibussowitsch        exec_names          = ['llvm-cov']
10097ce81a4bSJacob Faibussowitsch        if self.setCompilers.isDarwin(self.log):
10107ce81a4bSJacob Faibussowitsch          # macOS masquerades llvm-cov as just 'gcov', so we add this to the list in case
10117ce81a4bSJacob Faibussowitsch          # bare llvm-cov does not work
10127ce81a4bSJacob Faibussowitsch          exec_names.append('gcov')
10137ce81a4bSJacob Faibussowitsch      elif arg_opt == 'default-auto' and not required:
10147ce81a4bSJacob Faibussowitsch        # default-auto implies the user did not set it via command line!
10157ce81a4bSJacob 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)))
10167ce81a4bSJacob Faibussowitsch        return
10177ce81a4bSJacob Faibussowitsch      else:
10187ce81a4bSJacob Faibussowitsch        # implies 'auto' explicitly set by user, or we were required to find
10197ce81a4bSJacob Faibussowitsch        # something. either way we should error
10207ce81a4bSJacob Faibussowitsch        raise RuntimeError('Could not auto-detect coverage tool for {}, please set coverage tool name explicitly'.format(quoted(compiler)))
1021*095fb05fSJacob Faibussowitsch
1022*095fb05fSJacob Faibussowitsch      try:
1023*095fb05fSJacob Faibussowitsch        compiler_version_str = self.compilerFlags.version[lang]
1024*095fb05fSJacob Faibussowitsch      except KeyError:
1025*095fb05fSJacob Faibussowitsch        compiler_version_str = 'Unknown'
1026*095fb05fSJacob Faibussowitsch
1027*095fb05fSJacob Faibussowitsch      log_print('Searching version string {} (for compiler {}) using pattern {}'.format(quoted(compiler_version_str), quoted(compiler), quoted(compiler_version_re.pattern)))
1028*095fb05fSJacob Faibussowitsch      compiler_version = compiler_version_re.search(compiler_version_str)
1029*095fb05fSJacob Faibussowitsch      if compiler_version is not None:
1030*095fb05fSJacob Faibussowitsch        log_print('Found major = {}, minor = {}, patch = {}'.format(compiler_version.group(1), compiler_version.group(2), compiler_version.group(3)))
1031*095fb05fSJacob Faibussowitsch        # form [llvm-cov-14, llvm-cov-14.0, llvm-cov, etc.]
1032*095fb05fSJacob Faibussowitsch        cov_exec_name = exec_names[0]
1033*095fb05fSJacob Faibussowitsch        exec_names    = [
1034*095fb05fSJacob Faibussowitsch          # llvm-cov-14
1035*095fb05fSJacob Faibussowitsch          '{}-{}'.format(cov_exec_name, compiler_version.group(1)),
1036*095fb05fSJacob Faibussowitsch           # llvm-cov-14.0
1037*095fb05fSJacob Faibussowitsch          '{}-{}.{}'.format(cov_exec_name, compiler_version.group(1), compiler_version.group(2))
1038*095fb05fSJacob Faibussowitsch        ] + exec_names
10397ce81a4bSJacob Faibussowitsch    else:
10407ce81a4bSJacob Faibussowitsch      log_print('User explicitly set coverage exec as {}'.format(quoted(arg_opt)))
10417ce81a4bSJacob Faibussowitsch      par_dir = os.path.dirname(arg_opt)
10427ce81a4bSJacob Faibussowitsch      if os.path.exists(par_dir):
10437ce81a4bSJacob Faibussowitsch        # arg_opt is path-like, we should only search the provided directory when we go
10447ce81a4bSJacob Faibussowitsch        # looking for the tool
10457ce81a4bSJacob Faibussowitsch        use_default_path = False
10467ce81a4bSJacob Faibussowitsch        search_path      = par_dir
10477ce81a4bSJacob Faibussowitsch      exec_names = [arg_opt]
10487ce81a4bSJacob Faibussowitsch
10497ce81a4bSJacob Faibussowitsch    make_macro_name = 'PETSC_COVERAGE_EXEC'
10507ce81a4bSJacob Faibussowitsch    log_print('Checking for coverage tool(s):\n{}'.format('\n'.join('- '+t for t in exec_names)))
10517ce81a4bSJacob Faibussowitsch    found_exec = self.getExecutables(
10527ce81a4bSJacob Faibussowitsch      exec_names,
10537ce81a4bSJacob Faibussowitsch      path=search_path, getFullPath=True, useDefaultPath=use_default_path, resultName=make_macro_name
10547ce81a4bSJacob Faibussowitsch    )
10557ce81a4bSJacob Faibussowitsch
10567ce81a4bSJacob Faibussowitsch    if found_exec is None:
10577ce81a4bSJacob Faibussowitsch      # didn't find the coverage tool
10587ce81a4bSJacob Faibussowitsch      if required:
10597ce81a4bSJacob Faibussowitsch        raise RuntimeError('Coverage tool(s) {} could not be found. Please provide explicit path to coverage tool'.format(exec_names))
10607ce81a4bSJacob Faibussowitsch      return
10617ce81a4bSJacob Faibussowitsch
10627ce81a4bSJacob Faibussowitsch    found_exec_name = os.path.basename(found_exec)
10637ce81a4bSJacob Faibussowitsch    if 'llvm-cov' in found_exec_name and 'gcov' not in found_exec_name:
10647ce81a4bSJacob Faibussowitsch      # llvm-cov needs to be called as 'llvm-cov gcov' to work
10657ce81a4bSJacob Faibussowitsch      self.addMakeMacro(make_macro_name, found_exec + ' gcov')
10667ce81a4bSJacob Faibussowitsch    return
10677ce81a4bSJacob Faibussowitsch
1068689a5dfaSJacob Faibussowitsch  def configureStrictPetscErrorCode(self):
1069689a5dfaSJacob Faibussowitsch    """
1070689a5dfaSJacob Faibussowitsch    Enables or disables strict PetscErrorCode checking.
1071689a5dfaSJacob Faibussowitsch
1072689a5dfaSJacob Faibussowitsch    If --with-strict-petscerrorcode = 1:
1073689a5dfaSJacob Faibussowitsch    - defines PETSC_USE_STRICT_PETSCERRORCODE to 1
1074689a5dfaSJacob Faibussowitsch
1075689a5dfaSJacob Faibussowitsch    Else:
1076689a5dfaSJacob Faibussowitsch    - deletes any prior PETSC_USE_STRICT_PETSCERRORCODE definitions (if they exist)
1077689a5dfaSJacob Faibussowitsch    """
1078689a5dfaSJacob Faibussowitsch    define_name = 'USE_STRICT_PETSCERRORCODE'
1079689a5dfaSJacob Faibussowitsch    if self.argDB['with-strict-petscerrorcode']:
1080689a5dfaSJacob Faibussowitsch      self.addDefine(define_name, 1)
1081689a5dfaSJacob Faibussowitsch    else:
1082689a5dfaSJacob Faibussowitsch      # in case it was somehow added previously
1083689a5dfaSJacob Faibussowitsch      self.delDefine(define_name)
1084689a5dfaSJacob Faibussowitsch    return
1085689a5dfaSJacob Faibussowitsch
1086f8833479SBarry Smith#-----------------------------------------------------------------------------------------------------
1087b10d012aSSatish Balay  def configureCygwinBrokenPipe(self):
1088b10d012aSSatish Balay    '''Cygwin version <= 1.7.18 had issues with pipes and long commands invoked from gnu-make
1089b10d012aSSatish Balay    http://cygwin.com/ml/cygwin/2013-05/msg00340.html '''
10907fca349cSMatthew G. Knepley    if config.setCompilers.Configure.isCygwin(self.log):
1091b10d012aSSatish Balay      import platform
1092b10d012aSSatish Balay      import re
1093b10d012aSSatish Balay      r=re.compile("([0-9]+).([0-9]+).([0-9]+)")
1094b10d012aSSatish Balay      m=r.match(platform.release())
1095b10d012aSSatish Balay      major=int(m.group(1))
1096b10d012aSSatish Balay      minor=int(m.group(2))
1097b10d012aSSatish Balay      subminor=int(m.group(3))
1098b10d012aSSatish Balay      if ((major < 1) or (major == 1 and minor < 7) or (major == 1 and minor == 7 and subminor <= 18)):
1099b10d012aSSatish Balay        self.addMakeMacro('PETSC_CYGWIN_BROKEN_PIPE','1')
1100b10d012aSSatish Balay    return
1101b10d012aSSatish Balay
1102b10d012aSSatish Balay#-----------------------------------------------------------------------------------------------------
1103569865ddSSatish Balay  def configureDefaultArch(self):
1104af0996ceSBarry Smith    conffile = os.path.join('lib','petsc','conf', 'petscvariables')
1105569865ddSSatish Balay    if self.framework.argDB['with-default-arch']:
1106c6ef1b5bSJed Brown      fd = open(conffile, 'w')
1107569865ddSSatish Balay      fd.write('PETSC_ARCH='+self.arch.arch+'\n')
1108da93591fSBarry Smith      fd.write('PETSC_DIR='+self.petscdir.dir+'\n')
1109b9b902edSJed Brown      fd.write('include '+os.path.join('$(PETSC_DIR)','$(PETSC_ARCH)','lib','petsc','conf','petscvariables')+'\n')
1110569865ddSSatish Balay      fd.close()
1111569865ddSSatish Balay      self.framework.actions.addArgument('PETSc', 'Build', 'Set default architecture to '+self.arch.arch+' in '+conffile)
1112569865ddSSatish Balay    elif os.path.isfile(conffile):
1113569865ddSSatish Balay      try:
1114569865ddSSatish Balay        os.unlink(conffile)
1115569865ddSSatish Balay      except:
1116569865ddSSatish Balay        raise RuntimeError('Unable to remove file '+conffile+'. Did a different user create it?')
1117569865ddSSatish Balay    return
1118569865ddSSatish Balay
1119569865ddSSatish Balay#-----------------------------------------------------------------------------------------------------
1120f8833479SBarry Smith  def configureScript(self):
1121f8833479SBarry Smith    '''Output a script in the conf directory which will reproduce the configuration'''
1122f8833479SBarry Smith    import nargs
1123495bf1a9SSatish Balay    import sys
1124af0996ceSBarry Smith    scriptName = os.path.join(self.arch.arch,'lib','petsc','conf', 'reconfigure-'+self.arch.arch+'.py')
1125f8833479SBarry Smith    args = dict([(nargs.Arg.parseArgument(arg)[0], arg) for arg in self.framework.clArgs])
1126e97fc2efSSatish Balay    if 'with-clean' in args:
1127e97fc2efSSatish Balay      del args['with-clean']
1128d418e2d7SSatish Balay    if 'force' in args:
1129d418e2d7SSatish Balay      del args['force']
1130f8833479SBarry Smith    if 'configModules' in args:
11311063a081SSatish Balay      if nargs.Arg.parseArgument(args['configModules'])[1] == 'PETSc.Configure':
1132f8833479SBarry Smith        del args['configModules']
1133f8833479SBarry Smith    if 'optionsModule' in args:
113423a19ef1SSatish Balay      if nargs.Arg.parseArgument(args['optionsModule'])[1] == 'config.compilerOptions':
1135f8833479SBarry Smith        del args['optionsModule']
1136f8833479SBarry Smith    if not 'PETSC_ARCH' in args:
11371063a081SSatish Balay      args['PETSC_ARCH'] = 'PETSC_ARCH='+str(self.arch.arch)
1138c6ef1b5bSJed Brown    f = open(scriptName, 'w')
1139495bf1a9SSatish Balay    f.write('#!'+sys.executable+'\n')
1140f8833479SBarry Smith    f.write('if __name__ == \'__main__\':\n')
1141f8833479SBarry Smith    f.write('  import sys\n')
11427561c02cSSatish Balay    f.write('  import os\n')
11437561c02cSSatish Balay    f.write('  sys.path.insert(0, os.path.abspath(\'config\'))\n')
1144f8833479SBarry Smith    f.write('  import configure\n')
11451063a081SSatish Balay    # pretty print repr(args.values())
11461063a081SSatish Balay    f.write('  configure_options = [\n')
11478bec23c5SJed Brown    for itm in sorted(args.values()):
11481063a081SSatish Balay      f.write('    \''+str(itm)+'\',\n')
11491063a081SSatish Balay    f.write('  ]\n')
1150f8833479SBarry Smith    f.write('  configure.petsc_configure(configure_options)\n')
1151f8833479SBarry Smith    f.close()
1152f8833479SBarry Smith    try:
11535b6bfdb9SJed Brown      os.chmod(scriptName, 0o775)
11545b6bfdb9SJed Brown    except OSError as e:
1155f8833479SBarry Smith      self.framework.logPrint('Unable to make reconfigure script executable:\n'+str(e))
1156f8833479SBarry Smith    self.framework.actions.addArgument('PETSc', 'File creation', 'Created '+scriptName+' for automatic reconfiguration')
1157f8833479SBarry Smith    return
1158f8833479SBarry Smith
1159f8833479SBarry Smith  def configureInstall(self):
1160f8833479SBarry Smith    '''Setup the directories for installation'''
1161f8833479SBarry Smith    if self.framework.argDB['prefix']:
11625b4fc442SVaclav Hapla      self.addMakeRule('print_mesg_after_build','',
11635b4fc442SVaclav Hapla       ['-@echo "========================================="',
11645b4fc442SVaclav Hapla        '-@echo "Now to install the libraries do:"',
11655b4fc442SVaclav Hapla        '-@echo "%s${MAKE_USER} PETSC_DIR=${PETSC_DIR} PETSC_ARCH=${PETSC_ARCH} install"' % self.installdir.installSudo,
1166315b77e6SSatish Balay        '-@echo "========================================="'])
1167f8833479SBarry Smith    else:
11685b4fc442SVaclav Hapla      self.addMakeRule('print_mesg_after_build','',
11695b4fc442SVaclav Hapla       ['-@echo "========================================="',
11705b4fc442SVaclav Hapla        '-@echo "Now to check if the libraries are working do:"',
11715b4fc442SVaclav Hapla        '-@echo "${MAKE_USER} PETSC_DIR=${PETSC_DIR} PETSC_ARCH=${PETSC_ARCH} check"',
1172315b77e6SSatish Balay        '-@echo "========================================="'])
1173f8833479SBarry Smith      return
1174f8833479SBarry Smith
117528bb2e72SSatish Balay  def postProcessPackages(self):
117628bb2e72SSatish Balay    postPackages=[]
117728bb2e72SSatish Balay    for i in self.framework.packages:
117828bb2e72SSatish Balay      if hasattr(i,'postProcess'): postPackages.append(i)
117928bb2e72SSatish Balay    if postPackages:
1180e64d19dfSSatish Balay      # ctetgen needs petsc conf files. so attempt to create them early
1181a77eb93bSSatish Balay      self.framework.dumpConfFiles()
1182d9293e7bSBarry Smith      # tacky fix for dependency of Aluimia on Pflotran; requested via petsc-dev Matt provide a correct fix
1183d9293e7bSBarry Smith      for i in postPackages:
1184d9293e7bSBarry Smith        if i.name.upper() in ['PFLOTRAN']:
1185d9293e7bSBarry Smith          i.postProcess()
1186d9293e7bSBarry Smith          postPackages.remove(i)
118728bb2e72SSatish Balay      for i in postPackages: i.postProcess()
1188aa5c8b8eSBarry Smith      for i in postPackages:
1189aa5c8b8eSBarry Smith        if i.installedpetsc:
1190aa5c8b8eSBarry Smith          self.installed = 1
1191aa5c8b8eSBarry Smith          break
119228bb2e72SSatish Balay    return
1193f8833479SBarry Smith
1194f8833479SBarry Smith  def configure(self):
1195bf3e94a3SBarry Smith    if 'package-prefix-hash' in self.argDB:
1196bf3e94a3SBarry Smith      # turn off prefix if it was only used to for installing external packages.
1197bf3e94a3SBarry Smith      self.framework.argDB['prefix'] = ''
1198bf3e94a3SBarry Smith      self.dir = os.path.abspath(os.path.join(self.petscdir.dir, self.arch.arch))
1199bf3e94a3SBarry Smith      self.installdir.dir = self.dir
1200bf3e94a3SBarry Smith      self.installdir.petscDir = self.petscdir.dir
1201bf3e94a3SBarry Smith      self.petscDir = self.petscdir.dir
1202bf3e94a3SBarry Smith      self.petscArch = self.arch.arch
1203bf3e94a3SBarry Smith      self.addMakeMacro('PREFIXDIR',self.dir)
1204bf3e94a3SBarry Smith      self.confDir = os.path.abspath(os.path.join(self.petscdir.dir, self.arch.arch))
1205bf3e94a3SBarry Smith
1206f8833479SBarry Smith    if not os.path.samefile(self.petscdir.dir, os.getcwd()):
1207f8833479SBarry Smith      raise RuntimeError('Wrong PETSC_DIR option specified: '+str(self.petscdir.dir) + '\n  Configure invoked in: '+os.path.realpath(os.getcwd()))
1208550489e3SMatthew 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):
12093552d8fbSSatish Balay      raise RuntimeError('Incorrect option --prefix='+self.framework.argDB['prefix']+' specified. It cannot be same as PETSC_DIR!')
12108fd0dbdbSBarry Smith    if self.framework.argDB['prefix'] and self.framework.argDB['prefix'].find(' ') > -1:
12118fd0dbdbSBarry 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')
1212c16c35a9SSatish 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)):
1213c16c35a9SSatish Balay      raise RuntimeError('Incorrect option --prefix='+self.framework.argDB['prefix']+' specified. It cannot be same as PETSC_DIR/PETSC_ARCH!')
1214f16c1317SJed Brown    self.framework.header          = os.path.join(self.arch.arch,'include','petscconf.h')
1215f16c1317SJed Brown    self.framework.cHeader         = os.path.join(self.arch.arch,'include','petscfix.h')
1216bf113f49SJacob Faibussowitsch    self.framework.poisonheader    = os.path.join(self.arch.arch,'include','petscconf_poison.h')
12179c735a01SStefano Zampini    self.framework.pkgheader       = os.path.join(self.arch.arch,'include','petscpkg_version.h')
1218af0996ceSBarry Smith    self.framework.makeMacroHeader = os.path.join(self.arch.arch,'lib','petsc','conf','petscvariables')
1219af0996ceSBarry Smith    self.framework.makeRuleHeader  = os.path.join(self.arch.arch,'lib','petsc','conf','petscrules')
1220f8833479SBarry Smith    if self.libraries.math is None:
1221f8833479SBarry Smith      raise RuntimeError('PETSc requires a functional math library. Please send configure.log to petsc-maint@mcs.anl.gov.')
1222f8833479SBarry Smith    if self.languages.clanguage == 'Cxx' and not hasattr(self.compilers, 'CXX'):
1223f8833479SBarry Smith      raise RuntimeError('Cannot set C language to C++ without a functional C++ compiler.')
1224ed938b00SJed Brown    self.executeTest(self.configureRTLDDefault)
1225b2843cf1SBarry Smith    self.executeTest(self.configurePrefetch)
12262400fdedSBarry Smith    self.executeTest(self.configureUnused)
12271ef8df7fSJed Brown    self.executeTest(self.configureDeprecated)
122898ed35c3SBarry Smith    self.executeTest(self.configureIsatty)
1229e8e972b2SVaclav Hapla    self.executeTest(self.configureExpect)
1230e8e972b2SVaclav Hapla    self.executeTest(self.configureAlign)
1231e8e972b2SVaclav Hapla    self.executeTest(self.configureFunctionName)
1232e8e972b2SVaclav Hapla    self.executeTest(self.configureIntptrt)
1233f8833479SBarry Smith    self.executeTest(self.configureSolaris)
1234f8833479SBarry Smith    self.executeTest(self.configureLinux)
12350f64ec89SBarry Smith    self.executeTest(self.configureDarwin)
1236f8833479SBarry Smith    self.executeTest(self.configureWin32)
1237b10d012aSSatish Balay    self.executeTest(self.configureCygwinBrokenPipe)
1238569865ddSSatish Balay    self.executeTest(self.configureDefaultArch)
1239f8833479SBarry Smith    self.executeTest(self.configureScript)
1240f8833479SBarry Smith    self.executeTest(self.configureInstall)
124109bc878fSSatish Balay    self.executeTest(self.configureAtoll)
12427ce81a4bSJacob Faibussowitsch    for LANG in ['C', 'Cxx', 'CUDA', 'HIP', 'SYCL', 'FC']:
12437ce81a4bSJacob Faibussowitsch      compilerName = LANG.upper() if LANG in {'Cxx', 'FC'} else LANG+'C'
12447ce81a4bSJacob Faibussowitsch      if hasattr(self.setCompilers, compilerName):
12457ce81a4bSJacob Faibussowitsch        kwargs = {}
12467ce81a4bSJacob Faibussowitsch        if LANG in {'CUDA'}:
12477ce81a4bSJacob Faibussowitsch          # nvcc preprocesses the base file into a bunch of intermediate files, which are
12487ce81a4bSJacob Faibussowitsch          # then compiled by the host compiler. Why is this a problem?  Because the
12497ce81a4bSJacob Faibussowitsch          # generated coverage data is based on these preprocessed source files! So gcov
12507ce81a4bSJacob Faibussowitsch          # tries to read it later, but since its in the tmp directory it cannot. Thus we
12517ce81a4bSJacob Faibussowitsch          # need to keep them around (in a place we know about).
12527ce81a4bSJacob Faibussowitsch          nvcc_tmp_dir = os.path.join(self.petscdir.dir, self.arch.arch, 'nvcc_tmp')
12537ce81a4bSJacob Faibussowitsch          try:
12547ce81a4bSJacob Faibussowitsch            os.mkdir(nvcc_tmp_dir)
12557ce81a4bSJacob Faibussowitsch          except FileExistsError:
12567ce81a4bSJacob Faibussowitsch            pass
12577ce81a4bSJacob Faibussowitsch          kwargs['extra_coverage_flags'] = [
12587ce81a4bSJacob Faibussowitsch            '-Xcompiler --coverage -Xcompiler -fPIC --keep --keep-dir={}'.format(nvcc_tmp_dir)
12597ce81a4bSJacob Faibussowitsch          ]
12607ce81a4bSJacob Faibussowitsch          if self.kokkos.found:
12617ce81a4bSJacob Faibussowitsch            # yet again the kokkos nvcc_wrapper goes out of its way to be as useless as
12627ce81a4bSJacob Faibussowitsch            # possible. Its default arch (sm_35) is actually too low to compile kokkos,
12637ce81a4bSJacob Faibussowitsch            # for whatever reason this works if you dont use the --keep and --keep-dir
12647ce81a4bSJacob Faibussowitsch            # flags above.
12657ce81a4bSJacob Faibussowitsch            kwargs['extra_coverage_flags'].append('-arch=native')
12667ce81a4bSJacob Faibussowitsch          kwargs['extra_debug_flags'] = ['-Xcompiler -Og']
12677ce81a4bSJacob Faibussowitsch        self.executeTest(self.configureCoverage, args=[LANG], kargs=kwargs)
12687ce81a4bSJacob Faibussowitsch    self.executeTest(self.configureCoverageExecutable)
1269689a5dfaSJacob Faibussowitsch    self.executeTest(self.configureStrictPetscErrorCode)
1270f8833479SBarry Smith
1271f8833479SBarry Smith    self.Dump()
1272f8833479SBarry Smith    self.dumpConfigInfo()
12732a4161d9SMatthew G Knepley    self.dumpMachineInfo()
127449fe22e6SSatish Balay    self.delGenFiles()
127540277576SBarry Smith    # need to save the current state of BuildSystem so that postProcess() packages can read it in and perhaps run make install
127640277576SBarry Smith    self.framework.storeSubstitutions(self.framework.argDB)
1277492432c8SJed Brown    self.framework.argDB['configureCache'] = pickle.dumps(self.framework)
127840277576SBarry Smith    self.framework.argDB.save(force = True)
12798244ab14SJed Brown    self.DumpPkgconfig('PETSc.pc')
12808244ab14SJed Brown    self.DumpPkgconfig('petsc.pc')
1281351d3a41SMatthew G Knepley    self.DumpModule()
1282f7ad81e1SBarry Smith    self.postProcessPackages()
1283f8833479SBarry Smith    self.framework.log.write('================================================================================\n')
1284f8833479SBarry Smith    self.logClear()
1285f8833479SBarry Smith    return
1286