xref: /petsc/config/PETSc/Configure.py (revision 571416bbfcf5347cb6bed6e5b4b630e04e024673)
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):
18e4d7ee71SJacob Faibussowitsch    import logger
19e4d7ee71SJacob Faibussowitsch
205b4fc442SVaclav Hapla    desc = ['  Using GNU make: ' + self.make.make]
217ce81a4bSJacob Faibussowitsch    if self.defines.get('USE_COVERAGE'):
227ce81a4bSJacob Faibussowitsch      desc.extend([
237ce81a4bSJacob Faibussowitsch        '  Code coverage: yes',
247ce81a4bSJacob Faibussowitsch        '  Using code coverage executable: {}'.format(self.getMakeMacro('PETSC_COVERAGE_EXEC'))
257ce81a4bSJacob Faibussowitsch      ])
26e4d7ee71SJacob Faibussowitsch    banner_ends   = 'xxx'
27e4d7ee71SJacob Faibussowitsch    banner_middle = '=' * (logger.get_global_divider_length() - 2 * len(banner_ends))
28e4d7ee71SJacob Faibussowitsch    banner_line   = banner_middle.join((banner_ends, banner_ends))
29e4d7ee71SJacob Faibussowitsch    desc.append(banner_line)
30aa5c8b8eSBarry Smith    if not self.installed:
31dc0529c6SBarry Smith      desc.append(' Configure stage complete. Now build PETSc libraries with:')
325b4fc442SVaclav Hapla      desc.append('   %s PETSC_DIR=%s PETSC_ARCH=%s all' % (self.make.make_user, self.petscdir.dir, self.arch.arch))
33aa5c8b8eSBarry Smith    else:
34aa5c8b8eSBarry Smith      desc.append(' Installation complete. You do not need to run make to compile or install the software')
35e4d7ee71SJacob Faibussowitsch    desc.extend([banner_line, ''])
367ce81a4bSJacob Faibussowitsch    return '\n'.join(desc)
37f8833479SBarry Smith
38f8833479SBarry Smith  def setupHelp(self, help):
39f8833479SBarry Smith    import nargs
40ce0b2093SBarry Smith    help.addArgument('PETSc',  '-prefix=<dir>',                              nargs.Arg(None, '', 'Specifiy location to install PETSc (eg. /usr/local)'))
417deb5ab3SBarry Smith    help.addArgument('PETSc',  '-with-prefetch=<bool>',                      nargs.ArgBool(None, 1,'Enable checking for prefetch instructions'))
42eed94e11SSatish Balay    help.addArgument('Windows','-with-windows-graphics=<bool>',              nargs.ArgBool(None, 1,'Enable check for Windows Graphics'))
43569865ddSSatish Balay    help.addArgument('PETSc', '-with-default-arch=<bool>',                   nargs.ArgBool(None, 1, 'Allow using the last configured arch without setting PETSC_ARCH'))
4457cb31baSSatish Balay    help.addArgument('PETSc','-with-single-library=<bool>',                  nargs.ArgBool(None, 1,'Put all PETSc code into the single -lpetsc library'))
45cb297985SSatish Balay    help.addArgument('PETSc','-with-fortran-bindings=<bool>',                nargs.ArgBool(None, 1,'Build PETSc fortran bindings in the library and corresponding module files'))
46525d6f2eSBarry Smith    help.addArgument('PETSc', '-with-ios=<bool>',                            nargs.ArgBool(None, 0, 'Build an iPhone/iPad version of PETSc library'))
4721afe8ebSBarry Smith    help.addArgument('PETSc', '-with-display=<x11display>',                  nargs.Arg(None, '', 'Specifiy DISPLAY environmental variable for use with MATLAB test)'))
482c30b4dfSSatish Balay    help.addArgument('PETSc', '-with-package-scripts=<pyscripts>',           nargs.ArgFileList(None,None,'Specify configure package scripts for user provided packages'))
497ce81a4bSJacob Faibussowitsch    help.addArgument('PETSc', '-with-coverage=<bool>',                       nargs.ArgFuzzyBool(None, value=0, help='Enable or disable code-coverage collection'))
507ce81a4bSJacob 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'))
51a0c7f9aaSSamuel Khuvis    help.addArgument('PETSc', '-with-tau-perfstubs=<bool>',                  nargs.ArgBool(None, 1,'Enable TAU profiler stubs'))
52689a5dfaSJacob 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'))
53f8833479SBarry Smith    return
54f8833479SBarry Smith
556dd73af6SBarry Smith  def registerPythonFile(self,filename,directory):
566dd73af6SBarry Smith    ''' Add a python file to the framework and registers its headerprefix, ... externalpackagedir
576dd73af6SBarry Smith        directory is the directory where the file relative to the BuildSystem or config path in python notation with . '''
586dd73af6SBarry Smith    (utilityName, ext) = os.path.splitext(filename)
596dd73af6SBarry Smith    if not utilityName.startswith('.') and not utilityName.startswith('#') and ext == '.py' and not utilityName == '__init__':
606dd73af6SBarry Smith      if directory: directory = directory+'.'
616dd73af6SBarry Smith      utilityObj                             = self.framework.require(directory+utilityName, self)
626dd73af6SBarry Smith      utilityObj.headerPrefix                = self.headerPrefix
636dd73af6SBarry Smith      utilityObj.archProvider                = self.arch
646dd73af6SBarry Smith      utilityObj.languageProvider            = self.languages
656dd73af6SBarry Smith      utilityObj.installDirProvider          = self.installdir
666dd73af6SBarry Smith      utilityObj.externalPackagesDirProvider = self.externalpackagesdir
676dd73af6SBarry Smith      utilityObj.precisionProvider           = self.scalartypes
686dd73af6SBarry Smith      utilityObj.indexProvider               = self.indexTypes
696dd73af6SBarry Smith      setattr(self, utilityName.lower(), utilityObj)
7051294b80SMatthew G. Knepley      return utilityObj
7151294b80SMatthew G. Knepley    return None
726dd73af6SBarry Smith
73f8833479SBarry Smith  def setupDependencies(self, framework):
74f8833479SBarry Smith    config.base.Configure.setupDependencies(self, framework)
75dca78d2bSSatish Balay    self.programs      = framework.require('config.programs',           self)
76f8833479SBarry Smith    self.setCompilers  = framework.require('config.setCompilers',       self)
777ce81a4bSJacob Faibussowitsch    self.compilerFlags = framework.require('config.compilerFlags',      self)
7830b8aa07SMatthew G. Knepley    self.compilers     = framework.require('config.compilers',          self)
799d310bb7SBarry Smith    self.arch          = framework.require('PETSc.options.arch',        self.setCompilers)
809d310bb7SBarry Smith    self.petscdir      = framework.require('PETSc.options.petscdir',    self.arch)
819d310bb7SBarry Smith    self.installdir    = framework.require('PETSc.options.installDir',  self)
824e00a515SSatish Balay    self.dataFilesPath = framework.require('PETSc.options.dataFilesPath',self)
836dd73af6SBarry Smith    self.scalartypes   = framework.require('PETSc.options.scalarTypes', self)
846dd73af6SBarry Smith    self.indexTypes    = framework.require('PETSc.options.indexTypes',  self)
859d310bb7SBarry Smith    self.languages     = framework.require('PETSc.options.languages',   self.setCompilers)
8630b8aa07SMatthew G. Knepley    self.indexTypes    = framework.require('PETSc.options.indexTypes',  self.compilers)
87f8833479SBarry Smith    self.types         = framework.require('config.types',              self)
88f8833479SBarry Smith    self.headers       = framework.require('config.headers',            self)
89f8833479SBarry Smith    self.functions     = framework.require('config.functions',          self)
90f8833479SBarry Smith    self.libraries     = framework.require('config.libraries',          self)
91cd37d877SShri Abhyankar    self.atomics       = framework.require('config.atomics',            self)
929481793eSSatish Balay    self.make          = framework.require('config.packages.make',      self)
939552296fSBarry Smith    self.blasLapack    = framework.require('config.packages.BlasLapack',self)
94e6b0c433SBarry Smith    self.mpi           = framework.require('config.packages.MPI',       self)
950542e31aSBarry Smith    self.fortran       = framework.require('config.compilersFortran',   self)
960542e31aSBarry Smith    self.externalpackagesdir = framework.require('PETSc.options.externalpackagesdir',self)
9749d43ecaSSatish Balay
9809a6cbfcSBernhard M. Wiedemann    for utility in sorted(os.listdir(os.path.join('config','PETSc','options'))):
996dd73af6SBarry Smith      self.registerPythonFile(utility,'PETSc.options')
1009d310bb7SBarry Smith
10109a6cbfcSBernhard M. Wiedemann    for utility in sorted(os.listdir(os.path.join('config','BuildSystem','config','utilities'))):
1026dd73af6SBarry Smith      self.registerPythonFile(utility,'config.utilities')
10306e08bc7SBarry Smith
10409a6cbfcSBernhard M. Wiedemann    for package in sorted(os.listdir(os.path.join('config', 'BuildSystem', 'config', 'packages'))):
10551294b80SMatthew G. Knepley      obj = self.registerPythonFile(package,'config.packages')
10651294b80SMatthew G. Knepley      if obj:
10751294b80SMatthew G. Knepley        obj.archProvider                = self.framework.requireModule(obj.archProvider, obj)
10851294b80SMatthew G. Knepley        obj.languageProvider            = self.framework.requireModule(obj.languageProvider, obj)
10951294b80SMatthew G. Knepley        obj.installDirProvider          = self.framework.requireModule(obj.installDirProvider, obj)
11051294b80SMatthew G. Knepley        obj.externalPackagesDirProvider = self.framework.requireModule(obj.externalPackagesDirProvider, obj)
11151294b80SMatthew G. Knepley        obj.precisionProvider           = self.framework.requireModule(obj.precisionProvider, obj)
11251294b80SMatthew G. Knepley        obj.indexProvider               = self.framework.requireModule(obj.indexProvider, obj)
1136dd73af6SBarry Smith
1145faf1eacSMatthew G. Knepley    # Force blaslapack and opencl to depend on scalarType so precision is set before BlasLapack is built
1159d310bb7SBarry Smith    framework.require('PETSc.options.scalarTypes', self.f2cblaslapack)
1169d310bb7SBarry Smith    framework.require('PETSc.options.scalarTypes', self.fblaslapack)
1179d310bb7SBarry Smith    framework.require('PETSc.options.scalarTypes', self.blaslapack)
1185faf1eacSMatthew G. Knepley    framework.require('PETSc.options.scalarTypes', self.opencl)
119f8833479SBarry Smith
120dca78d2bSSatish Balay    self.programs.headerPrefix     = self.headerPrefix
1215b3958d3SJacob Faibussowitsch    self.setCompilers.headerPrefix = self.headerPrefix
122f8833479SBarry Smith    self.compilers.headerPrefix    = self.headerPrefix
1230542e31aSBarry Smith    self.fortran.headerPrefix      = self.headerPrefix
124f8833479SBarry Smith    self.types.headerPrefix        = self.headerPrefix
125f8833479SBarry Smith    self.headers.headerPrefix      = self.headerPrefix
126f8833479SBarry Smith    self.functions.headerPrefix    = self.headerPrefix
127f8833479SBarry Smith    self.libraries.headerPrefix    = self.headerPrefix
1286dd73af6SBarry Smith
1292c30b4dfSSatish Balay    # Register user provided package scripts
1302c30b4dfSSatish Balay    if 'with-package-scripts' in self.framework.argDB:
1312c30b4dfSSatish Balay      for script in self.framework.argDB['with-package-scripts']:
1322c30b4dfSSatish Balay        if os.path.splitext(script)[1] != '.py':
1332c30b4dfSSatish Balay          raise RuntimeError('Only python scripts compatible with configure package script format should be specified! Invalid option -with-package-scripts='+script)
1342c30b4dfSSatish Balay        self.framework.logPrint('User is registering a new package script: '+script)
1352c30b4dfSSatish Balay        dname,fname = os.path.split(script)
1362c30b4dfSSatish Balay        if dname: sys.path.append(dname)
1372c30b4dfSSatish Balay        self.registerPythonFile(fname,'')
1386dd73af6SBarry Smith
1396dd73af6SBarry Smith    # test for a variety of basic headers and functions
1404211eb48SBarry Smith    headersC = map(lambda name: name+'.h',['setjmp','dos','fcntl','float','io','malloc','pwd','strings',
141ace159c0SJed Brown                                            'unistd','machine/endian','sys/param','sys/procfs','sys/resource',
1422475b7caSBarry Smith                                            'sys/systeminfo','sys/times','sys/utsname',
1437e4f0192SMosè Giordano                                            'sys/socket','sys/wait','netinet/in','netdb','direct','time','Ws2tcpip','sys/types',
14418da0197SPierre Jolivet                                            'WindowsX','float','ieeefp','stdint','inttypes','immintrin'])
14545082d64SJed Brown    functions = ['access','_access','clock','drand48','getcwd','_getcwd','getdomainname','gethostname',
1464e04ae5aSSatish Balay                 'posix_memalign','popen','PXFGETARG','rand','getpagesize',
1474211eb48SBarry Smith                 'readlink','realpath','usleep','sleep','_sleep',
1482475b7caSBarry Smith                 'uname','snprintf','_snprintf','lseek','_lseek','time','fork','stricmp',
1492475b7caSBarry Smith                 'strcasecmp','bzero','dlopen','dlsym','dlclose','dlerror',
15067f4e542SPierre Jolivet                 '_set_output_format','_mkdir','socket','gethostbyname','fpresetsticky',
15118da0197SPierre Jolivet                 'fpsetsticky','__gcov_dump']
152b0651e32SBarry Smith    libraries = [(['fpe'],'handle_sigfpes')]
153b0651e32SBarry Smith    librariessock = [(['socket','nsl'],'socket')]
154f8833479SBarry Smith    self.headers.headers.extend(headersC)
155f8833479SBarry Smith    self.functions.functions.extend(functions)
156b0651e32SBarry Smith    self.libraries.libraries.extend(libraries)
157b0651e32SBarry Smith    if not hasattr(self,'socket'):
158b0651e32SBarry Smith      self.libraries.libraries.extend(librariessock)
159f8833479SBarry Smith    return
160f8833479SBarry Smith
1618244ab14SJed Brown  def DumpPkgconfig(self, petsc_pc):
162262119f8SBarry Smith    ''' Create a pkg-config file '''
163262119f8SBarry Smith    if not os.path.exists(os.path.join(self.petscdir.dir,self.arch.arch,'lib','pkgconfig')):
164262119f8SBarry Smith      os.makedirs(os.path.join(self.petscdir.dir,self.arch.arch,'lib','pkgconfig'))
1652eefe1c6SJed Brown    with open(os.path.join(self.petscdir.dir,self.arch.arch,'lib','pkgconfig',petsc_pc),'w') as fd:
1665e3311eeSJed Brown      cflags_inc = ['-I${includedir}']
167262119f8SBarry Smith      if self.framework.argDB['prefix']:
1685bb5b263SMatthew G. Knepley        fd.write('prefix='+self.installdir.dir+'\n')
169262119f8SBarry Smith      else:
170e1e675deSJed Brown        fd.write('prefix='+os.path.join(self.petscdir.dir, self.arch.arch)+'\n')
171e1e675deSJed Brown        cflags_inc.append('-I' + os.path.join(self.petscdir.dir, 'include'))
172262119f8SBarry Smith      fd.write('exec_prefix=${prefix}\n')
173262119f8SBarry Smith      fd.write('includedir=${prefix}/include\n')
1745e3311eeSJed Brown      fd.write('libdir=${prefix}/lib\n')
175262119f8SBarry Smith
1762eefe1c6SJed Brown      with self.setCompilers.Language('C'):
177262119f8SBarry Smith        fd.write('ccompiler='+self.setCompilers.getCompiler()+'\n')
178756c7f9fSJed Brown        fd.write('cflags_extra='+self.setCompilers.getCompilerFlags().strip()+'\n')
179756c7f9fSJed Brown        fd.write('cflags_dep='+self.compilers.dependenciesGenerationFlag.get('C','')+'\n')
180756c7f9fSJed Brown        fd.write('ldflag_rpath='+self.setCompilers.CSharedLinkerFlag+'\n')
18103e383c8SJed Brown      if hasattr(self.compilers, 'CXX'):
1822eefe1c6SJed Brown        with self.setCompilers.Language('C++'):
183262119f8SBarry Smith          fd.write('cxxcompiler='+self.setCompilers.getCompiler()+'\n')
184756c7f9fSJed Brown          fd.write('cxxflags_extra='+self.setCompilers.getCompilerFlags().strip()+'\n')
185262119f8SBarry Smith      if hasattr(self.compilers, 'FC'):
1862eefe1c6SJed Brown        with self.setCompilers.Language('FC'):
187262119f8SBarry Smith          fd.write('fcompiler='+self.setCompilers.getCompiler()+'\n')
188756c7f9fSJed Brown          fd.write('fflags_extra='+self.setCompilers.getCompilerFlags().strip()+'\n')
18950520af6SPatrick Sanan      if hasattr(self.compilers, 'CUDAC'):
19050520af6SPatrick Sanan        with self.setCompilers.Language('CUDA'):
19150520af6SPatrick Sanan          fd.write('cudacompiler='+self.setCompilers.getCompiler()+'\n')
19250520af6SPatrick Sanan          fd.write('cudaflags_extra='+self.setCompilers.getCompilerFlags().strip()+'\n')
1938f561fa3SPatrick Sanan          p = self.framework.require('config.packages.cuda')
19450520af6SPatrick Sanan          fd.write('cudalib='+self.libraries.toStringNoDupes(p.lib)+'\n')
19550520af6SPatrick Sanan          fd.write('cudainclude='+self.headers.toStringNoDupes(p.include)+'\n')
1967ba7a817SBarry Smith          if hasattr(self.setCompilers,'CUDA_CXX'):
1977ba7a817SBarry Smith            fd.write('cuda_cxx='+self.setCompilers.CUDA_CXX+'\n')
1987ba7a817SBarry Smith            fd.write('cuda_cxxflags='+self.setCompilers.CUDA_CXXFLAGS+'\n')
199262119f8SBarry Smith
200262119f8SBarry Smith      fd.write('\n')
201262119f8SBarry Smith      fd.write('Name: PETSc\n')
202262119f8SBarry Smith      fd.write('Description: Library to solve ODEs and algebraic equations\n')
203351d3a41SMatthew G Knepley      fd.write('Version: %s\n' % self.petscdir.version)
2045e3311eeSJed Brown      fd.write('Cflags: ' + ' '.join([self.setCompilers.CPPFLAGS] + cflags_inc) + '\n')
20537371b91SJed Brown      fd.write('Libs: '+self.libraries.toStringNoDupes(['-L${libdir}', self.petsclib], with_rpath=False)+'\n')
2068ebf8858SJed Brown      # Remove RPATH flags from library list.  User can add them using
2078ebf8858SJed Brown      # pkg-config --variable=ldflag_rpath and pkg-config --libs-only-L
208de8f682fSSatish 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')
209262119f8SBarry Smith    return
210262119f8SBarry Smith
211351d3a41SMatthew G Knepley  def DumpModule(self):
212351d3a41SMatthew G Knepley    ''' Create a module file '''
213af0996ceSBarry Smith    if not os.path.exists(os.path.join(self.petscdir.dir,self.arch.arch,'lib','petsc','conf','modules')):
214af0996ceSBarry Smith      os.makedirs(os.path.join(self.petscdir.dir,self.arch.arch,'lib','petsc','conf','modules'))
215af0996ceSBarry Smith    if not os.path.exists(os.path.join(self.petscdir.dir,self.arch.arch,'lib','petsc','conf','modules','petsc')):
216af0996ceSBarry Smith      os.makedirs(os.path.join(self.petscdir.dir,self.arch.arch,'lib','petsc','conf','modules','petsc'))
217351d3a41SMatthew G Knepley    if self.framework.argDB['prefix']:
2185bb5b263SMatthew G. Knepley      installdir  = self.installdir.dir
21955d606a3SSatish Balay      installarch = ''
22055d606a3SSatish Balay      installpath = os.path.join(installdir,'bin')
221351d3a41SMatthew G Knepley    else:
222351d3a41SMatthew G Knepley      installdir  = self.petscdir.dir
22355d606a3SSatish Balay      installarch = self.arch.arch
22455d606a3SSatish Balay      installpath = os.path.join(installdir,installarch,'bin')+':'+os.path.join(installdir,'bin')
225af0996ceSBarry Smith    fd = open(os.path.join(self.petscdir.dir,self.arch.arch,'lib','petsc','conf','modules','petsc',self.petscdir.version),'w')
226351d3a41SMatthew G Knepley    fd.write('''\
227351d3a41SMatthew G Knepley#%%Module
228351d3a41SMatthew G Knepley
229351d3a41SMatthew G Knepleyproc ModulesHelp { } {
230351d3a41SMatthew G Knepley    puts stderr "This module sets the path and environment variables for petsc-%s"
231a17b96a8SKyle Gerard Felker    puts stderr "     see https://petsc.org/ for more information      "
232351d3a41SMatthew G Knepley    puts stderr ""
233351d3a41SMatthew G Knepley}
234351d3a41SMatthew G Knepleymodule-whatis "PETSc - Portable, Extensible Toolkit for Scientific Computation"
235351d3a41SMatthew G Knepley
236dd486775SJed Brownset petsc_dir   "%s"
237dd486775SJed Brownset petsc_arch  "%s"
238351d3a41SMatthew G Knepley
239dd486775SJed Brownsetenv PETSC_ARCH "$petsc_arch"
240dd486775SJed Brownsetenv PETSC_DIR "$petsc_dir"
241dd486775SJed Brownprepend-path PATH "%s"
24255d606a3SSatish Balay''' % (self.petscdir.version, installdir, installarch, installpath))
243351d3a41SMatthew G Knepley    fd.close()
244351d3a41SMatthew G Knepley    return
245351d3a41SMatthew G Knepley
246f8833479SBarry Smith  def Dump(self):
247f8833479SBarry Smith    ''' Actually put the values into the configuration files '''
248f8833479SBarry Smith    # eventually everything between -- should be gone
24917f368bcSBarry Smith    if self.mpi.usingMPIUni:
25017f368bcSBarry Smith      #
25117f368bcSBarry Smith      # Remove any MPI/MPICH include files that may have been put here by previous runs of ./configure
2527908f030SMatthew 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)
25317f368bcSBarry Smith
254b5f71184SBarry Smith    self.logPrintDivider()
255b5f71184SBarry Smith    # Test for compiler-specific macros that need to be defined.
256b5f71184SBarry Smith    if self.setCompilers.isCrayVector('CC', self.log):
257b5f71184SBarry Smith      self.addDefine('HAVE_CRAY_VECTOR','1')
258b5f71184SBarry Smith
259b5f71184SBarry Smith    if self.functions.haveFunction('gethostbyname') and self.functions.haveFunction('socket') and self.headers.haveHeader('netinet/in.h'):
260b5f71184SBarry Smith      self.addDefine('USE_SOCKET_VIEWER','1')
261b5f71184SBarry Smith      if self.checkCompile('#include <sys/socket.h>','setsockopt(0,SOL_SOCKET,SO_REUSEADDR,0,0)'):
262b5f71184SBarry Smith        self.addDefine('HAVE_SO_REUSEADDR','1')
263b5f71184SBarry Smith
264b5f71184SBarry Smith    self.logPrintDivider()
2655f27b2e0SBarry Smith    self.setCompilers.pushLanguage('C')
2665f27b2e0SBarry Smith    compiler = self.setCompilers.getCompiler()
267217fe27eSSatish Balay    if [s for s in ['mpicc','mpiicc'] if os.path.basename(compiler).find(s)>=0]:
2685f27b2e0SBarry Smith      try:
2695f27b2e0SBarry Smith        output   = self.executeShellCommand(compiler + ' -show', log = self.log)[0]
2705f27b2e0SBarry Smith        compiler = output.split(' ')[0]
271f424265bSStefano Zampini        self.addDefine('MPICC_SHOW','"'+output.strip().replace('\n','\\\\n').replace('"','')+'"')
2725f27b2e0SBarry Smith      except:
273c9872b61SBarry Smith        self.addDefine('MPICC_SHOW','"Unavailable"')
274c9872b61SBarry Smith    else:
275c9872b61SBarry Smith      self.addDefine('MPICC_SHOW','"Unavailable"')
2765f27b2e0SBarry Smith    self.setCompilers.popLanguage()
277f8833479SBarry Smith#-----------------------------------------------------------------------------------------------------
278f8833479SBarry Smith
279f8833479SBarry Smith    # Sometimes we need C compiler, even if built with C++
280f8833479SBarry Smith    self.setCompilers.pushLanguage('C')
281e4c30378SBarry Smith    # do not use getCompilerFlags() because that automatically includes the CPPFLAGS so one ends up with duplication flags in makefile usage
282e4c30378SBarry Smith    self.addMakeMacro('CC_FLAGS',self.setCompilers.CFLAGS)
283f8833479SBarry Smith    self.setCompilers.popLanguage()
284f8833479SBarry Smith
28534f774f6SJed Brown    # And sometimes we need a C++ compiler even when PETSc is built with C
28634f774f6SJed Brown    if hasattr(self.compilers, 'CXX'):
28734f774f6SJed Brown      self.setCompilers.pushLanguage('Cxx')
28829921a8fSScott Kruger      self.addDefine('HAVE_CXX','1')
2890b119762SSatish Balay      self.addMakeMacro('CXXPP_FLAGS',self.setCompilers.CXXPPFLAGS)
290e4c30378SBarry Smith      # do not use getCompilerFlags() because that automatically includes the CXXPPFLAGS so one ends up with duplication flags in makefile usage
291e4c30378SBarry Smith      self.addMakeMacro('CXX_FLAGS',self.setCompilers.CXXFLAGS+' '+self.setCompilers.CXX_CXXFLAGS)
2922f4326f3SSatish Balay      cxx_linker = self.setCompilers.getLinker()
2932f4326f3SSatish Balay      self.addMakeMacro('CXX_LINKER',cxx_linker)
2942f4326f3SSatish Balay      self.addMakeMacro('CXX_LINKER_FLAGS',self.setCompilers.getLinkerFlags())
29534f774f6SJed Brown      self.setCompilers.popLanguage()
29605a757cfSSatish Balay    else:
29705a757cfSSatish Balay      self.addMakeMacro('CXX','')
29834f774f6SJed Brown
299f8833479SBarry Smith    # C preprocessor values
3001315f054SBarry Smith    self.addMakeMacro('CPP_FLAGS',self.setCompilers.CPPFLAGS)
301f8833479SBarry Smith
302f8833479SBarry Smith    # compiler values
303f8833479SBarry Smith    self.setCompilers.pushLanguage(self.languages.clanguage)
304f8833479SBarry Smith    self.addMakeMacro('PCC',self.setCompilers.getCompiler())
305e4c30378SBarry Smith    # do not use getCompilerFlags() because that automatically includes the preprocessor flags so one ends up with duplication flags in makefile usage
306e4c30378SBarry Smith    if self.languages.clanguage == 'C':
307e4c30378SBarry Smith      self.addMakeMacro('PCC_FLAGS','$(CC_FLAGS)')
308e4c30378SBarry Smith    else:
309e4c30378SBarry Smith      self.addMakeMacro('PCC_FLAGS','$(CXX_FLAGS)')
310f8833479SBarry Smith    self.setCompilers.popLanguage()
311f8833479SBarry Smith    # .o or .obj
312f8833479SBarry Smith    self.addMakeMacro('CC_SUFFIX','o')
313f8833479SBarry Smith
314f8833479SBarry Smith    # executable linker values
315f8833479SBarry Smith    self.setCompilers.pushLanguage(self.languages.clanguage)
316f8833479SBarry Smith    pcc_linker = self.setCompilers.getLinker()
317f8833479SBarry Smith    self.addMakeMacro('PCC_LINKER',pcc_linker)
318eacb1f64SJunchao Zhang    # We need to add sycl flags when linking petsc. See more in sycl.py.
319eacb1f64SJunchao Zhang    if hasattr(self.compilers, 'SYCLC'):
320eacb1f64SJunchao Zhang      self.addMakeMacro('PCC_LINKER_FLAGS',self.setCompilers.getLinkerFlags()+' '+self.setCompilers.SYCLFLAGS+' '+self.setCompilers.SYCLC_LINKER_FLAGS)
321eacb1f64SJunchao Zhang    else:
322c84a332bSSatish Balay      self.addMakeMacro('PCC_LINKER_FLAGS',self.setCompilers.getLinkerFlags())
323f8833479SBarry Smith    self.setCompilers.popLanguage()
324f8833479SBarry Smith    # '' for Unix, .exe for Windows
325f8833479SBarry Smith    self.addMakeMacro('CC_LINKER_SUFFIX','')
326f8833479SBarry Smith
327f8833479SBarry Smith    if hasattr(self.compilers, 'FC'):
328cb297985SSatish Balay      if self.framework.argDB['with-fortran-bindings']:
329257f4e5aSSatish Balay        if not self.fortran.fortranIsF90:
330257f4e5aSSatish Balay          raise RuntimeError('Error! Fortran compiler "'+self.compilers.FC+'" does not support F90! PETSc fortran bindings require a F90 compiler')
331fbf9dbe5SBarry Smith        self.addDefine('USE_FORTRAN_BINDINGS','1')
332f8833479SBarry Smith      self.setCompilers.pushLanguage('FC')
333f8833479SBarry Smith      # need FPPFLAGS in config/setCompilers
3340b119762SSatish Balay      self.addMakeMacro('FPP_FLAGS',self.setCompilers.FPPFLAGS)
335f8833479SBarry Smith
336f8833479SBarry Smith      # compiler values
337f8833479SBarry Smith      self.addMakeMacro('FC_FLAGS',self.setCompilers.getCompilerFlags())
338f8833479SBarry Smith      self.setCompilers.popLanguage()
339f8833479SBarry Smith      # .o or .obj
340f8833479SBarry Smith      self.addMakeMacro('FC_SUFFIX','o')
341f8833479SBarry Smith
342f8833479SBarry Smith      # executable linker values
343f8833479SBarry Smith      self.setCompilers.pushLanguage('FC')
344a9acdec7SBarry Smith      self.addMakeMacro('FC_LINKER',self.setCompilers.getLinker())
3456d53d35eSSatish Balay      self.addMakeMacro('FC_LINKER_FLAGS',self.setCompilers.getLinkerFlags())
346bb82cf9cSSatish Balay      self.setCompilers.popLanguage()
3475d631499SMatthew Knepley
3485d631499SMatthew Knepley      # F90 Modules
3495d631499SMatthew Knepley      if self.setCompilers.fortranModuleIncludeFlag:
3505d631499SMatthew Knepley        self.addMakeMacro('FC_MODULE_FLAG', self.setCompilers.fortranModuleIncludeFlag)
3516ddd6694SSatish Balay      else: # for non-f90 compilers like g77
3526ddd6694SSatish Balay        self.addMakeMacro('FC_MODULE_FLAG', '-I')
353a324c51cSMatthew G Knepley      if self.setCompilers.fortranModuleIncludeFlag:
354a324c51cSMatthew G Knepley        self.addMakeMacro('FC_MODULE_OUTPUT_FLAG', self.setCompilers.fortranModuleOutputFlag)
355f8833479SBarry Smith    else:
356f8833479SBarry Smith      self.addMakeMacro('FC','')
357f8833479SBarry Smith
35846a3958fSBarry Smith    if hasattr(self.compilers, 'CUDAC'):
3597ff2890cSSatish Balay      self.setCompilers.pushLanguage('CUDA')
360d93a25ecSSatish Balay      self.addMakeMacro('CUDAC_FLAGS',self.setCompilers.getCompilerFlags())
36150dcbc5aSJunchao Zhang      self.addMakeMacro('CUDAPP_FLAGS',self.setCompilers.CUDAPPFLAGS)
3627ff2890cSSatish Balay      self.setCompilers.popLanguage()
3637ff2890cSSatish Balay
364694a2f0eSJunchao Zhang    if hasattr(self.compilers, 'HIPC'):
36528f796eaSScott Kruger      self.setCompilers.pushLanguage('HIP')
366694a2f0eSJunchao Zhang      self.addMakeMacro('HIPC_FLAGS',self.setCompilers.getCompilerFlags())
3677fb1458fSStefano Zampini      self.addMakeMacro('HIPPP_FLAGS',self.setCompilers.HIPPPFLAGS)
36828f796eaSScott Kruger      self.setCompilers.popLanguage()
36928f796eaSScott Kruger
37050dcbc5aSJunchao Zhang    if hasattr(self.compilers, 'SYCLC'):
37128f796eaSScott Kruger      self.setCompilers.pushLanguage('SYCL')
37250dcbc5aSJunchao Zhang      self.addMakeMacro('SYCLC_FLAGS',self.setCompilers.getCompilerFlags())
373eacb1f64SJunchao Zhang      self.addMakeMacro('SYCLC_LINKER_FLAGS',self.setCompilers.getLinkerFlags())
37450dcbc5aSJunchao Zhang      self.addMakeMacro('SYCLPP_FLAGS',self.setCompilers.SYCLPPFLAGS)
37528f796eaSScott Kruger      self.setCompilers.popLanguage()
37628f796eaSScott Kruger
377f8833479SBarry Smith    # shared library linker values
378f8833479SBarry Smith    self.setCompilers.pushLanguage(self.languages.clanguage)
379f8833479SBarry Smith    # need to fix BuildSystem to collect these separately
380f8833479SBarry Smith    self.addMakeMacro('SL_LINKER',self.setCompilers.getLinker())
38170db8aa6SSatish Balay    self.addMakeMacro('SL_LINKER_FLAGS','${PCC_LINKER_FLAGS}')
382f8833479SBarry Smith    self.setCompilers.popLanguage()
383f8833479SBarry Smith    # One of 'a', 'so', 'lib', 'dll', 'dylib' (perhaps others also?) depending on the library generator and architecture
384f8833479SBarry Smith    # Note: . is not included in this macro, consistent with AR_LIB_SUFFIX
385f8833479SBarry Smith    if self.setCompilers.sharedLibraryExt == self.setCompilers.AR_LIB_SUFFIX:
386f8833479SBarry Smith      self.addMakeMacro('SL_LINKER_SUFFIX', '')
38746bc77b6SBarry Smith      self.addDefine('SLSUFFIX','""')
388f8833479SBarry Smith    else:
389f8833479SBarry Smith      self.addMakeMacro('SL_LINKER_SUFFIX', self.setCompilers.sharedLibraryExt)
39046bc77b6SBarry Smith      self.addDefine('SLSUFFIX','"'+self.setCompilers.sharedLibraryExt+'"')
391bb82cf9cSSatish Balay
39223e93537SBarry Smith    self.addMakeMacro('SL_LINKER_LIBS','${PETSC_EXTERNAL_LIB_BASIC}')
393bb82cf9cSSatish Balay
394f8833479SBarry Smith#-----------------------------------------------------------------------------------------------------
395f8833479SBarry Smith
396f8833479SBarry Smith    # CONLY or CPP. We should change the PETSc makefiles to do this better
397f8833479SBarry Smith    if self.languages.clanguage == 'C': lang = 'CONLY'
398f8833479SBarry Smith    else: lang = 'CXXONLY'
399f8833479SBarry Smith    self.addMakeMacro('PETSC_LANGUAGE',lang)
400f8833479SBarry Smith
401f8833479SBarry Smith    # real or complex
402f8833479SBarry Smith    self.addMakeMacro('PETSC_SCALAR',self.scalartypes.scalartype)
403f8833479SBarry Smith    # double or float
404f8833479SBarry Smith    self.addMakeMacro('PETSC_PRECISION',self.scalartypes.precision)
405f8833479SBarry Smith
406f8833479SBarry Smith    if self.framework.argDB['with-batch']:
407f8833479SBarry Smith      self.addMakeMacro('PETSC_WITH_BATCH','1')
408f8833479SBarry Smith
409f8833479SBarry Smith#-----------------------------------------------------------------------------------------------------
410a6cc6bb1SBarry Smith    # print include and lib for makefiles
411b5f71184SBarry Smith    self.logPrintDivider()
412f8833479SBarry Smith    self.framework.packages.reverse()
4135a21677cSJed Brown    petscincludes = [os.path.join(self.petscdir.dir,'include'),os.path.join(self.petscdir.dir,self.arch.arch,'include')]
4148749e224SJunchao Zhang    petscincludes_install = [os.path.join(self.installdir.dir, 'include')] if self.framework.argDB['prefix'] else petscincludes
4155a21677cSJed Brown    includes = []
416de8f682fSSatish Balay    self.packagelibs = []
417f8833479SBarry Smith    for i in self.framework.packages:
4187f0ff1afSBarry Smith      if not i.required:
4193972cb20SJacob Faibussowitsch        if i.devicePackage:
4203972cb20SJacob Faibussowitsch          self.addDefine('HAVE_DEVICE',1)
421eeb16384SBarry Smith        self.addDefine('HAVE_'+i.PACKAGE.replace('-','_'), 1)  # ONLY list package if it is used directly by PETSc (and not only by another package)
422f8833479SBarry Smith      if not isinstance(i.lib, list):
423f8833479SBarry Smith        i.lib = [i.lib]
424de8f682fSSatish Balay      if i.linkedbypetsc: self.packagelibs.extend(i.lib)
425eeb16384SBarry Smith      self.addMakeMacro(i.PACKAGE.replace('-','_')+'_LIB', self.libraries.toStringNoDupes(i.lib))
426f8833479SBarry Smith      if hasattr(i,'include'):
427f8833479SBarry Smith        if not isinstance(i.include,list):
428f8833479SBarry Smith          i.include = [i.include]
429ac9e4c42SSatish Balay        includes.extend(i.include)
430eeb16384SBarry Smith        self.addMakeMacro(i.PACKAGE.replace('-','_')+'_INCLUDE',self.headers.toStringNoDupes(i.include))
431*571416bbSBarry Smith
432de8f682fSSatish Balay    self.complibs = self.compilers.flibs+self.compilers.cxxlibs+self.compilers.LIBS.split()
433de8f682fSSatish Balay    self.PETSC_EXTERNAL_LIB_BASIC = self.libraries.toStringNoDupes(self.packagelibs+self.complibs)
434de8f682fSSatish Balay
435de8f682fSSatish Balay    self.addMakeMacro('PETSC_EXTERNAL_LIB_BASIC',self.PETSC_EXTERNAL_LIB_BASIC)
4365a21677cSJed Brown    allincludes = petscincludes + includes
4375a21677cSJed Brown    allincludes_install = petscincludes_install + includes
4385a21677cSJed Brown    self.PETSC_CC_INCLUDES = self.headers.toStringNoDupes(allincludes)
4395a21677cSJed Brown    self.PETSC_CC_INCLUDES_INSTALL = self.headers.toStringNoDupes(allincludes_install)
4405a21677cSJed Brown    self.addMakeMacro('PETSC_CC_INCLUDES',self.PETSC_CC_INCLUDES)
4415a21677cSJed Brown    self.addMakeMacro('PETSC_CC_INCLUDES_INSTALL', self.PETSC_CC_INCLUDES_INSTALL)
442cbd5cc15SBarry Smith    if hasattr(self.compilers, 'FC'):
4435a21677cSJed Brown      def modinc(includes):
4440542e31aSBarry Smith        return includes if self.fortran.fortranIsF90 else []
4455a21677cSJed Brown      self.addMakeMacro('PETSC_FC_INCLUDES',self.headers.toStringNoDupes(allincludes,modinc(allincludes)))
4465a21677cSJed Brown      self.addMakeMacro('PETSC_FC_INCLUDES_INSTALL',self.headers.toStringNoDupes(allincludes_install,modinc(allincludes_install)))
447f8833479SBarry Smith
448*571416bbSBarry Smith    LIB_DIR = os.path.join(self.installdir.dir,'lib')
449*571416bbSBarry Smith    self.addDefine('LIB_DIR','"'+LIB_DIR+'"')
450f8833479SBarry Smith
4510f3b21c2SBarry Smith    if self.framework.argDB['with-single-library']:
452*571416bbSBarry Smith      self.petsclib = '-lpetsc'
453*571416bbSBarry Smith      self.addDefine('USE_SINGLE_LIBRARY', '1')
4540f3b21c2SBarry Smith      self.addMakeMacro('LIBNAME','${INSTALL_LIB_DIR}/libpetsc.${AR_LIB_SUFFIX}')
45557cb31baSSatish Balay      self.addMakeMacro('SHLIBS','libpetsc')
456*571416bbSBarry Smith      self.addMakeMacro('PETSC_WITH_EXTERNAL_LIB',self.libraries.toStringNoDupes(['-L'+LIB_DIR, '-lpetsc']+self.packagelibs+self.complibs))
457ea820d49SSatish Balay      self.addMakeMacro('PETSC_SYS_LIB','${PETSC_WITH_EXTERNAL_LIB}')
458ea820d49SSatish Balay      self.addMakeMacro('PETSC_VEC_LIB','${PETSC_WITH_EXTERNAL_LIB}')
459ea820d49SSatish Balay      self.addMakeMacro('PETSC_MAT_LIB','${PETSC_WITH_EXTERNAL_LIB}')
460ea820d49SSatish Balay      self.addMakeMacro('PETSC_DM_LIB','${PETSC_WITH_EXTERNAL_LIB}')
461ea820d49SSatish Balay      self.addMakeMacro('PETSC_KSP_LIB','${PETSC_WITH_EXTERNAL_LIB}')
462ea820d49SSatish Balay      self.addMakeMacro('PETSC_SNES_LIB','${PETSC_WITH_EXTERNAL_LIB}')
463ea820d49SSatish Balay      self.addMakeMacro('PETSC_TS_LIB','${PETSC_WITH_EXTERNAL_LIB}')
464b0a7d7e7SSatish Balay      self.addMakeMacro('PETSC_TAO_LIB','${PETSC_WITH_EXTERNAL_LIB}')
465*571416bbSBarry Smith    else:
466*571416bbSBarry Smith      self.petsclib = '-lpetsctao -lpetscts -lpetscsnes -lpetscksp -lpetscdm -lpetscmat -lpetscvec -lpetscsys'
467*571416bbSBarry Smith      self.addMakeMacro('PETSC_SYS_LIB', self.libraries.toStringNoDupes(['-L'+LIB_DIR,'-lpetscsys']+self.packagelibs+self.complibs))
468*571416bbSBarry Smith      self.addMakeMacro('PETSC_VEC_LIB', self.libraries.toStringNoDupes(['-L'+LIB_DIR,'-lpetscvec','-lpetscsys']+self.packagelibs+self.complibs))
469*571416bbSBarry Smith      self.addMakeMacro('PETSC_MAT_LIB', self.libraries.toStringNoDupes(['-L'+LIB_DIR,'-lpetscmat','-lpetscvec','-lpetscsys']+self.packagelibs+self.complibs))
470*571416bbSBarry Smith      self.addMakeMacro('PETSC_DM_LIB',  self.libraries.toStringNoDupes(['-L'+LIB_DIR,'-lpetscdm','-lpetscmat','-lpetscvec','-lpetscsys']+self.packagelibs+self.complibs))
471*571416bbSBarry Smith      self.addMakeMacro('PETSC_KSP_LIB', self.libraries.toStringNoDupes(['-L'+LIB_DIR,'-lpetscksp','-lpetscdm','-lpetscmat','-lpetscvec','-lpetscsys']+self.packagelibs+self.complibs))
472*571416bbSBarry Smith      self.addMakeMacro('PETSC_SNES_LIB',self.libraries.toStringNoDupes(['-L'+LIB_DIR,'-lpetscsnes','-lpetscksp','-lpetscdm','-lpetscmat','-lpetscvec','-lpetscsys']+self.packagelibs+self.complibs))
473*571416bbSBarry Smith      self.addMakeMacro('PETSC_TS_LIB',  self.libraries.toStringNoDupes(['-L'+LIB_DIR,'-lpetscts','-lpetscsnes','-lpetscksp','-lpetscdm','-lpetscmat','-lpetscvec','-lpetscsys']+self.packagelibs+self.complibs))
474*571416bbSBarry Smith      self.addMakeMacro('PETSC_TAO_LIB', self.libraries.toStringNoDupes(['-L'+LIB_DIR,'-lpetsctao','-lpetscts','-lpetscsnes','-lpetscksp','-lpetscdm','-lpetscmat','-lpetscvec','-lpetscsys']+self.packagelibs+self.complibs))
475*571416bbSBarry Smith    self.addMakeMacro('PETSC_LIB','${PETSC_TAO_LIB}')
476*571416bbSBarry Smith    self.addMakeMacro('PETSC_LIB_BASIC',self.petsclib)
4770f3b21c2SBarry Smith
478f8833479SBarry Smith    if not os.path.exists(os.path.join(self.petscdir.dir,self.arch.arch,'lib')):
479f8833479SBarry Smith      os.makedirs(os.path.join(self.petscdir.dir,self.arch.arch,'lib'))
480f8833479SBarry Smith
48113f9d092SSatish Balay# add a makefile endtry for display
48213f9d092SSatish Balay    if self.framework.argDB['with-display']:
48313f9d092SSatish Balay      self.addMakeMacro('DISPLAY',self.framework.argDB['with-display'])
48413f9d092SSatish Balay
485f8833479SBarry Smith    # add a makefile entry for configure options
486f8833479SBarry Smith    self.addMakeMacro('CONFIGURE_OPTIONS', self.framework.getOptionsString(['configModules', 'optionsModule']).replace('\"','\\"'))
487a0c7f9aaSSamuel Khuvis
488a0c7f9aaSSamuel Khuvis    if self.framework.argDB['with-tau-perfstubs']:
489a0c7f9aaSSamuel Khuvis      self.addDefine('HAVE_TAU_PERFSTUBS',1)
490f8833479SBarry Smith    return
491f8833479SBarry Smith
492f8833479SBarry Smith  def dumpConfigInfo(self):
493f8833479SBarry Smith    import time
494c6ef1b5bSJed Brown    fd = open(os.path.join(self.arch.arch,'include','petscconfiginfo.h'),'w')
495dc25a686SPierre Jolivet    fd.write('static const char *petscconfigureoptions = "'+self.framework.getOptionsString(['configModules', 'optionsModule']).replace('\"','\\"').replace('\\ ','\\\\ ')+'";\n')
496f8833479SBarry Smith    fd.close()
497f8833479SBarry Smith    return
498f8833479SBarry Smith
4992a4161d9SMatthew G Knepley  def dumpMachineInfo(self):
5002a4161d9SMatthew G Knepley    import platform
501a970bd74SBernhard M. Wiedemann    import datetime
5022a4161d9SMatthew G Knepley    import time
50340373944SSatish Balay    import script
504ca77dbeeSGeoffrey Irving    def escape(s):
505e08ecd42SSatish Balay      return s.replace('"',r'\"').replace(r'\ ',r'\\ ') # novermin
506c6ef1b5bSJed Brown    fd = open(os.path.join(self.arch.arch,'include','petscmachineinfo.h'),'w')
5072a4161d9SMatthew G Knepley    fd.write('static const char *petscmachineinfo = \"\\n\"\n')
5082a4161d9SMatthew G Knepley    fd.write('\"-----------------------------------------\\n\"\n')
509a970bd74SBernhard M. Wiedemann    buildhost = platform.node()
510a970bd74SBernhard M. Wiedemann    if os.environ.get('SOURCE_DATE_EPOCH'):
511a970bd74SBernhard M. Wiedemann      buildhost = "reproducible"
512a970bd74SBernhard M. Wiedemann    buildtime = datetime.datetime.utcfromtimestamp(int(os.environ.get('SOURCE_DATE_EPOCH', time.time())))
513a970bd74SBernhard M. Wiedemann    fd.write('\"Libraries compiled on %s on %s \\n\"\n' % (buildtime, buildhost))
51460acdfe7SSatish Balay    fd.write('\"Machine characteristics: %s\\n\"\n' % (platform.platform()))
5155188cb68SSatish Balay    fd.write('\"Using PETSc directory: %s\\n\"\n' % (escape(self.installdir.petscDir)))
5165188cb68SSatish Balay    fd.write('\"Using PETSc arch: %s\\n\"\n' % (escape(self.installdir.petscArch)))
517cdec380aSBarry Smith    fd.write('\"-----------------------------------------\\n\";\n')
5182a4161d9SMatthew G Knepley    fd.write('static const char *petsccompilerinfo = \"\\n\"\n')
5192a4161d9SMatthew G Knepley    self.setCompilers.pushLanguage(self.languages.clanguage)
5205f27b2e0SBarry Smith    fd.write('\"Using C compiler: %s %s \\n\"\n' % (escape(self.setCompilers.getCompiler()), escape(self.setCompilers.getCompilerFlags())))
5212a4161d9SMatthew G Knepley    self.setCompilers.popLanguage()
5228782282cSMatthew G Knepley    if hasattr(self.compilers, 'FC'):
5232a4161d9SMatthew G Knepley      self.setCompilers.pushLanguage('FC')
5245f27b2e0SBarry Smith      fd.write('\"Using Fortran compiler: %s %s  %s\\n\"\n' % (escape(self.setCompilers.getCompiler()), escape(self.setCompilers.getCompilerFlags()), escape(self.setCompilers.CPPFLAGS)))
5252a4161d9SMatthew G Knepley      self.setCompilers.popLanguage()
526cdec380aSBarry Smith    fd.write('\"-----------------------------------------\\n\";\n')
5272a4161d9SMatthew G Knepley    fd.write('static const char *petsccompilerflagsinfo = \"\\n\"\n')
5285a21677cSJed Brown    fd.write('\"Using include paths: %s\\n\"\n' % (escape(self.PETSC_CC_INCLUDES_INSTALL.replace('${PETSC_DIR}', self.installdir.petscDir))))
529cdec380aSBarry Smith    fd.write('\"-----------------------------------------\\n\";\n')
5302a4161d9SMatthew G Knepley    fd.write('static const char *petsclinkerinfo = \"\\n\"\n')
5312a4161d9SMatthew G Knepley    self.setCompilers.pushLanguage(self.languages.clanguage)
532ca77dbeeSGeoffrey Irving    fd.write('\"Using C linker: %s\\n\"\n' % (escape(self.setCompilers.getLinker())))
5332a4161d9SMatthew G Knepley    self.setCompilers.popLanguage()
5348782282cSMatthew G Knepley    if hasattr(self.compilers, 'FC'):
5352a4161d9SMatthew G Knepley      self.setCompilers.pushLanguage('FC')
536ca77dbeeSGeoffrey Irving      fd.write('\"Using Fortran linker: %s\\n\"\n' % (escape(self.setCompilers.getLinker())))
5372a4161d9SMatthew G Knepley      self.setCompilers.popLanguage()
5385188cb68SSatish Balay    fd.write('\"Using libraries: %s%s -L%s %s %s\\n\"\n' % (escape(self.setCompilers.CSharedLinkerFlag), escape(os.path.join(self.installdir.petscDir, self.installdir.petscArch, 'lib')), escape(os.path.join(self.installdir.petscDir, self.installdir.petscArch, 'lib')), escape(self.petsclib), escape(self.PETSC_EXTERNAL_LIB_BASIC)))
539cdec380aSBarry Smith    fd.write('\"-----------------------------------------\\n\";\n')
5402a4161d9SMatthew G Knepley    fd.close()
5412a4161d9SMatthew G Knepley    return
542b2843cf1SBarry Smith
543b2843cf1SBarry Smith  def configurePrefetch(self):
544b2843cf1SBarry Smith    '''Sees if there are any prefetch functions supported'''
5457fca349cSMatthew G. Knepley    if config.setCompilers.Configure.isSolaris(self.log) or self.framework.argDB['with-ios'] or not self.framework.argDB['with-prefetch']:
54693f78423SSatish Balay      self.addDefine('Prefetch(a,b,c)', ' ')
54793f78423SSatish Balay      return
548ec284106SBarry Smith    self.pushLanguage(self.languages.clanguage)
54910699583SJed Brown    if self.checkLink('#include <xmmintrin.h>', 'void *v = 0;_mm_prefetch((const char*)v,_MM_HINT_NTA);\n'):
55050d8bf02SJed Brown      # The Intel Intrinsics manual [1] specifies the prototype
55150d8bf02SJed Brown      #
55250d8bf02SJed Brown      #   void _mm_prefetch(char const *a, int sel);
55350d8bf02SJed Brown      #
55450d8bf02SJed Brown      # but other vendors seem to insist on using subtly different
55550d8bf02SJed Brown      # prototypes, including void* for the pointer, and an enum for
55650d8bf02SJed Brown      # sel.  These are both reasonable changes, but negatively impact
55750d8bf02SJed Brown      # portability.
55850d8bf02SJed Brown      #
559a8d69d7bSBarry Smith      # [1] https://software.intel.com/file/6373
56050d8bf02SJed Brown      self.addDefine('HAVE_XMMINTRIN_H', 1)
56150d8bf02SJed Brown      self.addDefine('Prefetch(a,b,c)', '_mm_prefetch((const char*)(a),(c))')
56250d8bf02SJed Brown      self.addDefine('PREFETCH_HINT_NTA', '_MM_HINT_NTA')
56350d8bf02SJed Brown      self.addDefine('PREFETCH_HINT_T0',  '_MM_HINT_T0')
56450d8bf02SJed Brown      self.addDefine('PREFETCH_HINT_T1',  '_MM_HINT_T1')
56550d8bf02SJed Brown      self.addDefine('PREFETCH_HINT_T2',  '_MM_HINT_T2')
56650d8bf02SJed Brown    elif self.checkLink('#include <xmmintrin.h>', 'void *v = 0;_mm_prefetch(v,_MM_HINT_NTA);\n'):
56750d8bf02SJed Brown      self.addDefine('HAVE_XMMINTRIN_H', 1)
56850d8bf02SJed Brown      self.addDefine('Prefetch(a,b,c)', '_mm_prefetch((const void*)(a),(c))')
56950d8bf02SJed Brown      self.addDefine('PREFETCH_HINT_NTA', '_MM_HINT_NTA')
57050d8bf02SJed Brown      self.addDefine('PREFETCH_HINT_T0',  '_MM_HINT_T0')
57150d8bf02SJed Brown      self.addDefine('PREFETCH_HINT_T1',  '_MM_HINT_T1')
57250d8bf02SJed Brown      self.addDefine('PREFETCH_HINT_T2',  '_MM_HINT_T2')
57310699583SJed Brown    elif self.checkLink('', 'void *v = 0;__builtin_prefetch(v,0,0);\n'):
57410699583SJed Brown      # From GCC docs: void __builtin_prefetch(const void *addr,int rw,int locality)
57510699583SJed Brown      #
57610699583SJed Brown      #   The value of rw is a compile-time constant one or zero; one
57710699583SJed Brown      #   means that the prefetch is preparing for a write to the memory
57810699583SJed Brown      #   address and zero, the default, means that the prefetch is
57910699583SJed Brown      #   preparing for a read. The value locality must be a compile-time
58010699583SJed Brown      #   constant integer between zero and three. A value of zero means
58110699583SJed Brown      #   that the data has no temporal locality, so it need not be left
58210699583SJed Brown      #   in the cache after the access. A value of three means that the
58310699583SJed Brown      #   data has a high degree of temporal locality and should be left
58410699583SJed Brown      #   in all levels of cache possible. Values of one and two mean,
58510699583SJed Brown      #   respectively, a low or moderate degree of temporal locality.
58610699583SJed Brown      #
58710699583SJed Brown      # Here we adopt Intel's x86/x86-64 naming scheme for the locality
58810699583SJed Brown      # hints.  Using macros for these values in necessary since some
58910699583SJed Brown      # compilers require an enum.
59010699583SJed Brown      self.addDefine('Prefetch(a,b,c)', '__builtin_prefetch((a),(b),(c))')
59110699583SJed Brown      self.addDefine('PREFETCH_HINT_NTA', '0')
59210699583SJed Brown      self.addDefine('PREFETCH_HINT_T0',  '3')
59310699583SJed Brown      self.addDefine('PREFETCH_HINT_T1',  '2')
59410699583SJed Brown      self.addDefine('PREFETCH_HINT_T2',  '1')
595b2843cf1SBarry Smith    else:
596b2843cf1SBarry Smith      self.addDefine('Prefetch(a,b,c)', ' ')
5977d490b44SBarry Smith    self.popLanguage()
598b2843cf1SBarry Smith
59949fe22e6SSatish Balay  def delGenFiles(self):
60049fe22e6SSatish Balay    '''Delete generated files'''
60149fe22e6SSatish Balay    delfile = os.path.join(self.arch.arch,'lib','petsc','conf','files')
60249fe22e6SSatish Balay    try:
60349fe22e6SSatish Balay      os.unlink(delfile)
60449fe22e6SSatish Balay    except: pass
60549fe22e6SSatish Balay
60609bc878fSSatish Balay  def configureAtoll(self):
60709bc878fSSatish Balay    '''Checks if atoll exists'''
60822164b4cSPierre 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'):
60909bc878fSSatish Balay       self.addDefine('HAVE_ATOLL', '1')
61009bc878fSSatish Balay
6112400fdedSBarry Smith  def configureUnused(self):
6122400fdedSBarry Smith    '''Sees if __attribute((unused)) is supported'''
6131adaff47SSean Farley    if self.framework.argDB['with-ios']:
6142400fdedSBarry Smith      self.addDefine('UNUSED', ' ')
6152400fdedSBarry Smith      return
6162400fdedSBarry Smith    self.pushLanguage(self.languages.clanguage)
61722164b4cSPierre 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'):
6182400fdedSBarry Smith      self.addDefine('UNUSED', '__attribute((unused))')
6192400fdedSBarry Smith    else:
6202400fdedSBarry Smith      self.addDefine('UNUSED', ' ')
6212400fdedSBarry Smith    self.popLanguage()
6222400fdedSBarry Smith
62398ed35c3SBarry Smith  def configureIsatty(self):
62498ed35c3SBarry Smith    '''Check if the Unix C function isatty() works correctly
62598ed35c3SBarry Smith       Actually just assumes it does not work correctly on batch systems'''
62698ed35c3SBarry Smith    if not self.framework.argDB['with-batch']:
62798ed35c3SBarry Smith      self.addDefine('USE_ISATTY',1)
62898ed35c3SBarry Smith
6291ef8df7fSJed Brown  def configureDeprecated(self):
6301ef8df7fSJed Brown    '''Check if __attribute((deprecated)) is supported'''
6317efe37a1SJacob Faibussowitsch    def checkDeprecated(macro_base, src, is_intel):
6327efe37a1SJacob Faibussowitsch      '''
6337efe37a1SJacob Faibussowitsch      run through the various attribute deprecated combinations and define MACRO_BAS(why) to the result
6347efe37a1SJacob Faibussowitsch      it if it compiles.
6357efe37a1SJacob Faibussowitsch
6367efe37a1SJacob Faibussowitsch      If none of the combos work, defines MACRO_BASE(why) as empty
6377efe37a1SJacob Faibussowitsch      '''
6385029be03SJacob Faibussowitsch      full_macro_name = macro_base + '(string_literal_why)'
6397efe37a1SJacob Faibussowitsch      for prefix in ('__attribute__', '__attribute','__declspec'):
6407efe37a1SJacob Faibussowitsch        if prefix == '__declspec':
6417efe37a1SJacob Faibussowitsch          # declspec does not have an extra set of brackets around the arguments
6425029be03SJacob Faibussowitsch          attr_bodies = ('deprecated(string_literal_why)', 'deprecated')
6431ef8df7fSJed Brown        else:
6445029be03SJacob Faibussowitsch          attr_bodies = ('(deprecated(string_literal_why))', '(deprecated)')
6457efe37a1SJacob Faibussowitsch
6467efe37a1SJacob Faibussowitsch        for attr_body in attr_bodies:
6477efe37a1SJacob Faibussowitsch          attr_def = '{}({})'.format(prefix, attr_body)
6487efe37a1SJacob Faibussowitsch          test_src = '\n'.join((
6497efe37a1SJacob Faibussowitsch            '#define {} {}'.format(full_macro_name, attr_def),
6507efe37a1SJacob Faibussowitsch            src.format(macro_base + '("asdasdadsasd")')
6517efe37a1SJacob Faibussowitsch          ))
6527efe37a1SJacob Faibussowitsch          if self.checkCompile(test_src, ''):
6537efe37a1SJacob Faibussowitsch            self.logPrint('configureDeprecated: \'{}\' appears to work'.format(attr_def))
6547efe37a1SJacob Faibussowitsch            if is_intel and '(why)' in attr_body:
6557efe37a1SJacob Faibussowitsch              self.logPrint('configureDeprecated: Intel has conspired to make a supremely environment-sensitive compiler. The Intel compiler looks at the gcc executable in the environment to determine the language compatibility that it should attempt to emulate. Some important Cray installations have built PETSc using the Intel compiler, but with a newer gcc module loaded (e.g. 4.7). Thus at PETSc configure time, the Intel compiler decides to support the string argument, but the gcc found in the default user environment is older and does not support the argument.\n'.format(attr_def))
6567efe37a1SJacob Faibussowitsch              self.logPrint('*** WE WILL THEREFORE REJECT \'{}\' AND CONTINUE TESTING ***'.format(attr_def))
6577efe37a1SJacob Faibussowitsch              continue
6587efe37a1SJacob Faibussowitsch            self.addDefine(full_macro_name, attr_def)
6597efe37a1SJacob Faibussowitsch            return
6607efe37a1SJacob Faibussowitsch
6617efe37a1SJacob Faibussowitsch      self.addDefine(full_macro_name, ' ')
6627efe37a1SJacob Faibussowitsch      return
6637efe37a1SJacob Faibussowitsch
6647efe37a1SJacob Faibussowitsch    lang = self.languages.clanguage
6657efe37a1SJacob Faibussowitsch    with self.Language(lang):
6667efe37a1SJacob Faibussowitsch      is_intel = self.setCompilers.isIntel(self.getCompiler(lang=lang), self.log)
6675029be03SJacob Faibussowitsch      checkDeprecated('DEPRECATED_FUNCTION_BASE', '{} int myfunc(void) {{ return 1; }}', is_intel)
6685029be03SJacob Faibussowitsch      checkDeprecated('DEPRECATED_TYPEDEF_BASE', 'typedef int my_int {};', is_intel)
6695029be03SJacob Faibussowitsch      checkDeprecated('DEPRECATED_ENUM_BASE', 'enum E {{ oldval {}, newval }};', is_intel)
670a14d4ff0SToby Isaac      checkDeprecated('DEPRECATED_OBJECT_BASE', '{} int x;', is_intel)
6717efe37a1SJacob Faibussowitsch      # I was unable to make a CPP macro that takes the old and new values as separate
6727efe37a1SJacob Faibussowitsch      # arguments and builds the message needed by _Pragma hence the deprecation message is
6737efe37a1SJacob Faibussowitsch      # handled as it is
67405de396fSBarry Smith      if self.checkCompile('#define TEST _Pragma("GCC warning \"Testing _Pragma\"") value'):
6755029be03SJacob Faibussowitsch        self.addDefine('DEPRECATED_MACRO_BASE_(why)', '_Pragma(#why)')
6765029be03SJacob Faibussowitsch        self.addDefine('DEPRECATED_MACRO_BASE(string_literal_why)', self.substPrefix + '_DEPRECATED_MACRO_BASE_(GCC warning string_literal_why)')
67705de396fSBarry Smith      else:
6785029be03SJacob Faibussowitsch        self.addDefine('DEPRECATED_MACRO_BASE(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
83722164b4cSPierre Jolivet    if not self.checkCompile('#include <sys/types.h>\n','uid_t u;\n(void)u'):
838f8833479SBarry Smith      self.addTypedef('int', 'uid_t')
839f8833479SBarry Smith      self.addTypedef('int', 'gid_t')
84022164b4cSPierre 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'):
841f8833479SBarry Smith      self.framework.addDefine('R_OK', '04')
842f8833479SBarry Smith      self.framework.addDefine('W_OK', '02')
843f8833479SBarry Smith      self.framework.addDefine('X_OK', '01')
844f8833479SBarry Smith    if not self.checkLink('#include <sys/stat.h>\n','int a=0;\nif (S_ISDIR(a)){}\n'):
845f8833479SBarry Smith      self.framework.addDefine('S_ISREG(a)', '(((a)&_S_IFMT) == _S_IFREG)')
846f8833479SBarry Smith      self.framework.addDefine('S_ISDIR(a)', '(((a)&_S_IFMT) == _S_IFDIR)')
8474e8afd12SMosè Giordano    if self.checkCompile('#include <windows.h>\n','LARGE_INTEGER a;\nDWORD b=a.u.HighPart;\n'):
848f8833479SBarry Smith      self.addDefine('HAVE_LARGE_INTEGER_U',1)
849f8833479SBarry Smith
850f8833479SBarry Smith    # Windows requires a Binary file creation flag when creating/opening binary files.  Is a better test in order?
8514e8afd12SMosè Giordano    if self.checkCompile('#include <windows.h>\n#include <fcntl.h>\n', 'int flags = O_BINARY;'):
852f8833479SBarry Smith      self.addDefine('HAVE_O_BINARY',1)
853f8833479SBarry Smith
854f8833479SBarry Smith    if self.compilers.CC.find('win32fe') >= 0:
855ad4212abSSatish Balay      self.addDefine('HAVE_WINDOWS_COMPILERS',1)
856f8833479SBarry Smith      self.addDefine('DIR_SEPARATOR','\'\\\\\'')
857f8833479SBarry Smith      self.addDefine('REPLACE_DIR_SEPARATOR','\'/\'')
858f8833479SBarry Smith      self.addDefine('CANNOT_START_DEBUGGER',1)
8595188cb68SSatish Balay      (petscdir,error,status) = self.executeShellCommand('cygpath -w '+self.installdir.petscDir, log = self.log)
86034531a4dSSatish Balay      self.addDefine('DIR','"'+petscdir.replace('\\','\\\\')+'"')
8615188cb68SSatish Balay      (petscdir,error,status) = self.executeShellCommand('cygpath -m '+self.installdir.petscDir, log = self.log)
862e433681fSSatish Balay      self.addMakeMacro('wPETSC_DIR',petscdir)
8634e00a515SSatish Balay      if self.dataFilesPath.datafilespath:
8644e00a515SSatish Balay        (datafilespath,error,status) = self.executeShellCommand('cygpath -m '+self.dataFilesPath.datafilespath, log = self.log)
8654e00a515SSatish Balay        self.addMakeMacro('DATAFILESPATH',datafilespath)
8664e00a515SSatish Balay
867f8833479SBarry Smith    else:
868f8833479SBarry Smith      self.addDefine('REPLACE_DIR_SEPARATOR','\'\\\\\'')
869f8833479SBarry Smith      self.addDefine('DIR_SEPARATOR','\'/\'')
8705188cb68SSatish Balay      self.addDefine('DIR','"'+self.installdir.petscDir+'"')
8715188cb68SSatish Balay      self.addMakeMacro('wPETSC_DIR',self.installdir.petscDir)
8724e00a515SSatish Balay      if self.dataFilesPath.datafilespath:
8734e00a515SSatish Balay        self.addMakeMacro('DATAFILESPATH',self.dataFilesPath.datafilespath)
8745188cb68SSatish Balay    self.addDefine('ARCH','"'+self.installdir.petscArch+'"')
875f8833479SBarry Smith    return
876f8833479SBarry Smith
877d3d5cfdcSJacob Faibussowitsch  def configureCoverageForLang(self, log_printer_cls, lang, extra_coverage_flags=None, extra_debug_flags=None):
8787ce81a4bSJacob Faibussowitsch    """
8797ce81a4bSJacob Faibussowitsch    Check that a compiler accepts code-coverage flags. If the compiler does accept code-coverage flags
8807ce81a4bSJacob Faibussowitsch    try to set debugging flags equivalent to -Og.
8817ce81a4bSJacob Faibussowitsch
8827ce81a4bSJacob Faibussowitsch    Arguments:
8837ce81a4bSJacob Faibussowitsch    - lang: the language to check the coverage flag for
8847ce81a4bSJacob Faibussowitsch    - extra_coverage_flags: a list of extra flags to use when checking the coverage flags
8857ce81a4bSJacob Faibussowitsch    - extra_debug_flags: a list of extra flags to try when setting debug flags
8867ce81a4bSJacob Faibussowitsch
8877ce81a4bSJacob Faibussowitsch    On success:
8887ce81a4bSJacob Faibussowitsch    - defines PETSC_USE_COVERAGE to 1
8897ce81a4bSJacob Faibussowitsch    """
890d3d5cfdcSJacob Faibussowitsch    log_print = log_printer_cls(self)
8917ce81a4bSJacob Faibussowitsch
8927ce81a4bSJacob Faibussowitsch    def quoted(string):
8937ce81a4bSJacob Faibussowitsch      return string.join(("'", "'"))
8947ce81a4bSJacob Faibussowitsch
8957ce81a4bSJacob Faibussowitsch    def make_flag_list(default, extra):
8967ce81a4bSJacob Faibussowitsch      ret = [default]
8977ce81a4bSJacob Faibussowitsch      if extra is not None:
8987ce81a4bSJacob Faibussowitsch        assert isinstance(extra, list)
8997ce81a4bSJacob Faibussowitsch        ret.extend(extra)
9007ce81a4bSJacob Faibussowitsch      return ret
9017ce81a4bSJacob Faibussowitsch
9027ce81a4bSJacob Faibussowitsch    log_print('Checking coverage flag for language {}'.format(lang))
9039b81490aSJacob Faibussowitsch
9049b81490aSJacob Faibussowitsch    compiler = self.getCompiler(lang=lang)
9059b81490aSJacob Faibussowitsch    if self.setCompilers.isGNU(compiler, self.log):
906d3d5cfdcSJacob Faibussowitsch      is_gnuish = True
9079b81490aSJacob Faibussowitsch    elif self.setCompilers.isClang(compiler, self.log):
908d3d5cfdcSJacob Faibussowitsch      is_gnuish = True
9099b81490aSJacob Faibussowitsch    else:
910d3d5cfdcSJacob Faibussowitsch      is_gnuish = False
9117ce81a4bSJacob Faibussowitsch
9127ce81a4bSJacob Faibussowitsch    # if not gnuish and we don't have a set of extra flags, bail
913d3d5cfdcSJacob Faibussowitsch    if not is_gnuish and extra_coverage_flags is None:
9147ce81a4bSJacob 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)))
9157ce81a4bSJacob Faibussowitsch      return
9167ce81a4bSJacob Faibussowitsch
9177ce81a4bSJacob Faibussowitsch    coverage_flags = make_flag_list('--coverage', extra_coverage_flags)
9187ce81a4bSJacob Faibussowitsch    log_print('Checking set of coverage flags: {}'.format(coverage_flags))
9197ce81a4bSJacob Faibussowitsch
9209b81490aSJacob Faibussowitsch    found = None
9217ce81a4bSJacob Faibussowitsch    with self.Language(lang):
9227ce81a4bSJacob Faibussowitsch      with self.setCompilers.Language(lang):
9237ce81a4bSJacob Faibussowitsch        for flag in coverage_flags:
9249b81490aSJacob Faibussowitsch          # the linker also needs to see the coverage flag
9259b81490aSJacob Faibussowitsch          with self.setCompilers.extraCompilerFlags([flag], compilerOnly=False) as skip_flags:
9269b81490aSJacob Faibussowitsch            if not skip_flags and self.checkRun():
9279b81490aSJacob Faibussowitsch              # flag was accepted
9289b81490aSJacob Faibussowitsch              found = flag
9297ce81a4bSJacob Faibussowitsch              break
9309b81490aSJacob Faibussowitsch
9317ce81a4bSJacob Faibussowitsch          log_print(
9327ce81a4bSJacob Faibussowitsch            'Compiler {} did not accept coverage flag {}'.format(quoted(compiler), quoted(flag))
9337ce81a4bSJacob Faibussowitsch          )
9347ce81a4bSJacob Faibussowitsch
9359b81490aSJacob Faibussowitsch        if found is None:
9369b81490aSJacob Faibussowitsch          log_print(
9379b81490aSJacob Faibussowitsch            'Compiler {} did not accept ANY coverage flags: {}, bailing!'.format(
9389b81490aSJacob Faibussowitsch              quoted(compiler), coverage_flags
9399b81490aSJacob Faibussowitsch            )
9409b81490aSJacob Faibussowitsch          )
9417ce81a4bSJacob Faibussowitsch          return
9427ce81a4bSJacob Faibussowitsch
9439b81490aSJacob Faibussowitsch        # must do this exactly here since:
9449b81490aSJacob Faibussowitsch        #
9459b81490aSJacob Faibussowitsch        # 1. setCompilers.extraCompilerFlags() will reset the compiler flags on __exit__()
9469b81490aSJacob Faibussowitsch        #    (so cannot do it in the loop)
9479b81490aSJacob Faibussowitsch        # 2. we need to set the compiler flag while setCompilers.Language() is still in
9489b81490aSJacob Faibussowitsch        #    effect (so cannot do it outside the with statements)
9499b81490aSJacob Faibussowitsch        self.setCompilers.insertCompilerFlag(flag, False)
9509b81490aSJacob Faibussowitsch
9517ce81a4bSJacob Faibussowitsch    if not self.functions.haveFunction('__gcov_dump'):
9527ce81a4bSJacob Faibussowitsch      self.functions.checkClassify(['__gcov_dump'])
9537ce81a4bSJacob Faibussowitsch
9547ce81a4bSJacob Faibussowitsch    # now check if we can override the optimization level. It is only kosher to do so if
9557ce81a4bSJacob Faibussowitsch    # the user did not explicitly set the optimization flags (via CFLAGS, CXXFLAGS,
9567ce81a4bSJacob Faibussowitsch    # CXXOPTFLAGS, etc). If they have done so, we sternly warn them about their lapse in
9577ce81a4bSJacob Faibussowitsch    # judgement
9587ce81a4bSJacob Faibussowitsch    with self.Language(lang):
9597ce81a4bSJacob Faibussowitsch      compiler_flags = self.getCompilerFlags()
9607ce81a4bSJacob Faibussowitsch
9617ce81a4bSJacob Faibussowitsch    user_set          = 0
9627ce81a4bSJacob Faibussowitsch    allowed_opt_flags = re.compile(r'|'.join((r'-O[01g]', r'-g[1-9]*')))
9637ce81a4bSJacob Faibussowitsch    for flagsname in [self.getCompilerFlagsName(lang), self.compilerFlags.getOptionalFlagsName(lang)]:
9647ce81a4bSJacob Faibussowitsch      if flagsname in self.argDB:
9657ce81a4bSJacob Faibussowitsch        opt_flags = [
9667ce81a4bSJacob Faibussowitsch          f for f in self.compilerFlags.findOptFlags(compiler_flags) if not allowed_opt_flags.match(f)
9677ce81a4bSJacob Faibussowitsch        ]
9687ce81a4bSJacob Faibussowitsch        if opt_flags:
9697ce81a4bSJacob 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)))
9707ce81a4bSJacob Faibussowitsch          user_set = 1
9717ce81a4bSJacob Faibussowitsch          break
9727ce81a4bSJacob Faibussowitsch
97310b490f0SJacob Faibussowitsch    # disable this for now, the warning should be sufficient. If the user still chooses to
97410b490f0SJacob Faibussowitsch    # ignore it, then that's on them
97510b490f0SJacob Faibussowitsch    if 0 and not user_set:
9767ce81a4bSJacob Faibussowitsch      debug_flags = make_flag_list('-Og', extra_debug_flags)
9777ce81a4bSJacob Faibussowitsch      with self.setCompilers.Language(lang):
9787ce81a4bSJacob Faibussowitsch        for flag in debug_flags:
9797ce81a4bSJacob Faibussowitsch          try:
9807ce81a4bSJacob Faibussowitsch            self.setCompilers.addCompilerFlag(flag)
9817ce81a4bSJacob Faibussowitsch          except RuntimeError:
9827ce81a4bSJacob Faibussowitsch            continue
9837ce81a4bSJacob Faibussowitsch          break
9847ce81a4bSJacob Faibussowitsch
9857ce81a4bSJacob Faibussowitsch    self.addDefine('USE_COVERAGE', 1)
9867ce81a4bSJacob Faibussowitsch    return
9877ce81a4bSJacob Faibussowitsch
9889b81490aSJacob Faibussowitsch  def configureCoverage(self):
989d3d5cfdcSJacob Faibussowitsch    """
990d3d5cfdcSJacob Faibussowitsch    Configure coverage for all available languages.
991d3d5cfdcSJacob Faibussowitsch
992d3d5cfdcSJacob Faibussowitsch    If user did not request coverage, this function does nothing and returns immediatel.
993d3d5cfdcSJacob Faibussowitsch    Therefore the following only apply to the case where the user requested coverage.
994d3d5cfdcSJacob Faibussowitsch
995d3d5cfdcSJacob Faibussowitsch    On success:
996d3d5cfdcSJacob Faibussowitsch    - defines PETSC_USE_COVERAGE to 1
997d3d5cfdcSJacob Faibussowitsch
998d3d5cfdcSJacob Faibussowitsch    On failure:
999d3d5cfdcSJacob Faibussowitsch    - If no compilers supported the coverage flag, throws RuntimeError
1000d3d5cfdcSJacob Faibussowitsch    -
1001d3d5cfdcSJacob Faibussowitsch    """
1002d3d5cfdcSJacob Faibussowitsch    class LogPrinter:
1003d3d5cfdcSJacob Faibussowitsch      def __init__(self, cfg):
1004d3d5cfdcSJacob Faibussowitsch        self.cfg = cfg
1005d3d5cfdcSJacob Faibussowitsch        try:
1006d3d5cfdcSJacob Faibussowitsch          import inspect
1007d3d5cfdcSJacob Faibussowitsch
1008d3d5cfdcSJacob Faibussowitsch          calling_func_stack = inspect.stack()[1]
1009d3d5cfdcSJacob Faibussowitsch          if sys.version_info >= (3, 5):
1010d3d5cfdcSJacob Faibussowitsch            func_name = calling_func_stack.function
1011d3d5cfdcSJacob Faibussowitsch          else:
1012d3d5cfdcSJacob Faibussowitsch            func_name = calling_func_stack[3]
1013d3d5cfdcSJacob Faibussowitsch        except:
1014d3d5cfdcSJacob Faibussowitsch          func_name = 'Unknown'
1015d3d5cfdcSJacob Faibussowitsch        self.fmt_str = func_name + '(): {}'
1016d3d5cfdcSJacob Faibussowitsch
1017d3d5cfdcSJacob Faibussowitsch      def __call__(self, msg, *args, **kwargs):
1018d3d5cfdcSJacob Faibussowitsch        return self.cfg.logPrint(self.fmt_str.format(msg), *args, **kwargs)
1019d3d5cfdcSJacob Faibussowitsch
1020d3d5cfdcSJacob Faibussowitsch    argdb_flag = 'with-coverage'
1021d3d5cfdcSJacob Faibussowitsch    log_print  = LogPrinter(self)
1022d3d5cfdcSJacob Faibussowitsch    if not self.argDB[argdb_flag]:
1023d3d5cfdcSJacob Faibussowitsch      log_print('coverage was disabled from command line or default')
1024d3d5cfdcSJacob Faibussowitsch      return
1025d3d5cfdcSJacob Faibussowitsch
1026d3d5cfdcSJacob Faibussowitsch    tested_langs = []
10279b81490aSJacob Faibussowitsch    for LANG in ['C', 'Cxx', 'CUDA', 'HIP', 'SYCL', 'FC']:
10289b81490aSJacob Faibussowitsch      compilerName = LANG.upper() if LANG in {'Cxx', 'FC'} else LANG + 'C'
10299b81490aSJacob Faibussowitsch      if hasattr(self.setCompilers, compilerName):
10309b81490aSJacob Faibussowitsch        kwargs = {}
10319b81490aSJacob Faibussowitsch        if LANG in {'CUDA'}:
10329b81490aSJacob Faibussowitsch          # nvcc preprocesses the base file into a bunch of intermediate files, which are
10339b81490aSJacob Faibussowitsch          # then compiled by the host compiler. Why is this a problem?  Because the
10349b81490aSJacob Faibussowitsch          # generated coverage data is based on these preprocessed source files! So gcov
10359b81490aSJacob Faibussowitsch          # tries to read it later, but since its in the tmp directory it cannot. Thus we
10369b81490aSJacob Faibussowitsch          # need to keep them around (in a place we know about).
10379b81490aSJacob Faibussowitsch          nvcc_tmp_dir = os.path.join(self.petscdir.dir, self.arch.arch, 'nvcc_tmp')
10389b81490aSJacob Faibussowitsch          try:
10399b81490aSJacob Faibussowitsch            os.mkdir(nvcc_tmp_dir)
10409b81490aSJacob Faibussowitsch          except FileExistsError:
10419b81490aSJacob Faibussowitsch            pass
10429b81490aSJacob Faibussowitsch          kwargs['extra_coverage_flags'] = [
10439b81490aSJacob Faibussowitsch            '-Xcompiler --coverage -Xcompiler -fPIC --keep --keep-dir={}'.format(nvcc_tmp_dir)
10449b81490aSJacob Faibussowitsch          ]
10459b81490aSJacob Faibussowitsch          if self.kokkos.found:
10469b81490aSJacob Faibussowitsch            # yet again the kokkos nvcc_wrapper goes out of its way to be as useless as
10479b81490aSJacob Faibussowitsch            # possible. Its default arch (sm_35) is actually too low to compile kokkos,
10489b81490aSJacob Faibussowitsch            # for whatever reason this works if you dont use the --keep and --keep-dir
10499b81490aSJacob Faibussowitsch            # flags above.
10509b81490aSJacob Faibussowitsch            kwargs['extra_coverage_flags'].append('-arch=native')
10519b81490aSJacob Faibussowitsch            kwargs['extra_debug_flags'] = ['-Xcompiler -Og']
1052d3d5cfdcSJacob Faibussowitsch        tested_langs.append(LANG)
1053d3d5cfdcSJacob Faibussowitsch        self.executeTest(self.configureCoverageForLang, args=[LogPrinter, LANG], kargs=kwargs)
1054d3d5cfdcSJacob Faibussowitsch
1055d3d5cfdcSJacob Faibussowitsch    if not self.defines.get('USE_COVERAGE'):
1056d3d5cfdcSJacob Faibussowitsch      # coverage was requested but no compilers accepted it, this is an error
1057d3d5cfdcSJacob Faibussowitsch      raise RuntimeError(
1058d3d5cfdcSJacob Faibussowitsch        'Coverage was requested (--{}={}) but none of the compilers supported it:\n{}\n'.format(
1059d3d5cfdcSJacob Faibussowitsch          argdb_flag, self.argDB[argdb_flag],
1060d3d5cfdcSJacob Faibussowitsch          '\n'.join(['  - {} ({})'.format(self.getCompiler(lang=lang), lang) for lang in tested_langs])
1061d3d5cfdcSJacob Faibussowitsch        )
1062d3d5cfdcSJacob Faibussowitsch      )
1063d3d5cfdcSJacob Faibussowitsch
1064d3d5cfdcSJacob Faibussowitsch    return
1065d3d5cfdcSJacob Faibussowitsch    # Disabled for now, since this does not really work. It solves the problem of
1066d3d5cfdcSJacob Faibussowitsch    # "undefined reference to __gcov_flush()" but if we add -lgcov we get:
1067d3d5cfdcSJacob Faibussowitsch    #
1068d3d5cfdcSJacob Faibussowitsch    # duplicate symbol '___gcov_reset' in:
1069d3d5cfdcSJacob Faibussowitsch    #     /Library/.../libclang_rt.profile_osx.a(GCDAProfiling.c.o)
1070d3d5cfdcSJacob Faibussowitsch    #     /opt/.../libgcov.a(_gcov_reset.o)
1071d3d5cfdcSJacob Faibussowitsch    # duplicate symbol '___gcov_dump' in:
1072d3d5cfdcSJacob Faibussowitsch    #     /opt/.../libgcov.a(_gcov_dump.o)
1073d3d5cfdcSJacob Faibussowitsch    #     /Library/.../libclang_rt.profile_osx.a(GCDAProfiling.c.o)
1074d3d5cfdcSJacob Faibussowitsch    # duplicate symbol '___gcov_fork' in:
1075d3d5cfdcSJacob Faibussowitsch    #     /opt/.../libgcov.a(_gcov_fork.o)
1076d3d5cfdcSJacob Faibussowitsch    #     /Library/.../libclang_rt.profile_osx.a(GCDAProfiling.c.o)
1077d3d5cfdcSJacob Faibussowitsch    #
1078d3d5cfdcSJacob Faibussowitsch    # I don't know how to solve this.
1079d3d5cfdcSJacob Faibussowitsch
1080d3d5cfdcSJacob Faibussowitsch    log_print('Checking if compilers can cross-link disparate coverage libraries')
1081d3d5cfdcSJacob Faibussowitsch    # At least one of the compilers has coverage enabled. Now need to make sure multiple
1082d3d5cfdcSJacob Faibussowitsch    # code coverage impls work together, specifically when using clang C/C++ compiler with
1083d3d5cfdcSJacob Faibussowitsch    # gfortran.
1084d3d5cfdcSJacob Faibussowitsch    if not hasattr(self.setCompilers, 'FC'):
1085d3d5cfdcSJacob Faibussowitsch      log_print('No fortran compiler detected. No need to check cross-linking!')
1086d3d5cfdcSJacob Faibussowitsch      return
1087d3d5cfdcSJacob Faibussowitsch
1088d3d5cfdcSJacob Faibussowitsch    c_lang = self.languages.clanguage
1089d3d5cfdcSJacob Faibussowitsch    if not self.setCompilers.isClang(self.getCompiler(lang=c_lang), self.log):
1090d3d5cfdcSJacob Faibussowitsch      # must be GCC
1091d3d5cfdcSJacob Faibussowitsch      log_print('C-language ({}) compiler is not clang, assuming it is GCC, so cross-linking with FC ({}) assumed to be OK'.format(c_lang, self.getCompiler(lang='FC')))
1092d3d5cfdcSJacob Faibussowitsch      return
1093d3d5cfdcSJacob Faibussowitsch
1094d3d5cfdcSJacob Faibussowitsch    # If we are here we:
1095d3d5cfdcSJacob Faibussowitsch    #   1. Have both C/C++ compiler and fortran compiler
1096d3d5cfdcSJacob Faibussowitsch    #   2. The C/C++ compiler is *not* the same as the fortran compiler (unless we start
1097d3d5cfdcSJacob Faibussowitsch    #      using flang)
1098d3d5cfdcSJacob Faibussowitsch    #
1099d3d5cfdcSJacob Faibussowitsch    # Now we check if we can cross-link
1100d3d5cfdcSJacob Faibussowitsch    def can_cross_link(**kwargs):
1101d3d5cfdcSJacob Faibussowitsch      f_body = "      subroutine foo()\n      print*,'testing'\n      return\n      end\n"
1102d3d5cfdcSJacob Faibussowitsch      c_body = "int main() { }"
1103d3d5cfdcSJacob Faibussowitsch
1104d3d5cfdcSJacob Faibussowitsch      return self.compilers.checkCrossLink(
1105d3d5cfdcSJacob Faibussowitsch        f_body, c_body, language1='FC', language2=c_lang, extralibs=self.compilers.flibs, **kwargs
1106d3d5cfdcSJacob Faibussowitsch      )
1107d3d5cfdcSJacob Faibussowitsch
1108d3d5cfdcSJacob Faibussowitsch    log_print('Trying to cross-link WITHOUT extra libs')
1109d3d5cfdcSJacob Faibussowitsch    if can_cross_link():
1110d3d5cfdcSJacob Faibussowitsch      log_print('Successfully cross-linked WITHOUT extra libs')
1111d3d5cfdcSJacob Faibussowitsch      # success, we already can cross-link
1112d3d5cfdcSJacob Faibussowitsch      return
1113d3d5cfdcSJacob Faibussowitsch
1114d3d5cfdcSJacob Faibussowitsch    extra_libs = ['-lgcov']
1115d3d5cfdcSJacob Faibussowitsch    log_print('Trying to cross-link with extra libs: {}'.format(extra_libs))
1116d3d5cfdcSJacob Faibussowitsch    if can_cross_link(extraObjs=extra_libs):
1117d3d5cfdcSJacob Faibussowitsch      log_print(
1118d3d5cfdcSJacob Faibussowitsch        'Successfully cross-linked using extra libs: {}, adding them to LIBS'.format(extra_libs)
1119d3d5cfdcSJacob Faibussowitsch      )
1120d3d5cfdcSJacob Faibussowitsch      self.setCompilers.LIBS += ' ' + ' '.join(extra_libs)
1121d3d5cfdcSJacob Faibussowitsch    else:
1122d3d5cfdcSJacob Faibussowitsch      # maybe should be an error?
1123d3d5cfdcSJacob Faibussowitsch      self.logPrintWarning("Could not successfully cross-link covered code between {} and FC. Sometimes this is a false positive. Assuming this does eventually end up working when the full link-line is assembled when building PETSc. If you later encounter linker errors about missing __gcov_exit(), __gcov_init(), __llvm_cov_flush() etc. this is why!".format(c_lang))
11249b81490aSJacob Faibussowitsch    return
11259b81490aSJacob Faibussowitsch
11267ce81a4bSJacob Faibussowitsch  def configureCoverageExecutable(self):
11277ce81a4bSJacob Faibussowitsch    """
11287ce81a4bSJacob Faibussowitsch    Check that a code-coverage collecting tool exists and is on PATH.
11297ce81a4bSJacob Faibussowitsch
11307ce81a4bSJacob Faibussowitsch    On success:
11317ce81a4bSJacob Faibussowitsch    - Adds PETSC_COVERAGE_EXEC make macro containing the full path to the coverage tool executable.
11327ce81a4bSJacob Faibussowitsch
11337ce81a4bSJacob Faibussowitsch    Raises RuntimeError if:
11347ce81a4bSJacob Faibussowitsch    - User explicitly requests auto-detection of the coverage tool from command line, and this
11357ce81a4bSJacob Faibussowitsch      routine fails to guess the suitable tool name.
11367ce81a4bSJacob Faibussowitsch    - The routine fails to find the tool, and --with-coverage is true
11377ce81a4bSJacob Faibussowitsch    """
11387ce81a4bSJacob Faibussowitsch    def log_print(msg, *args, **kwargs):
11397ce81a4bSJacob Faibussowitsch      self.logPrint('checkCoverage: '+str(msg), *args, **kwargs)
11407ce81a4bSJacob Faibussowitsch      return
11417ce81a4bSJacob Faibussowitsch
11427ce81a4bSJacob Faibussowitsch    def quoted(string):
11437ce81a4bSJacob Faibussowitsch      return string.join(("'", "'"))
11447ce81a4bSJacob Faibussowitsch
11457ce81a4bSJacob Faibussowitsch    required         = bool(self.argDB['with-coverage'])
11467ce81a4bSJacob Faibussowitsch    arg_opt          = self.argDB['with-coverage-exec']
11477ce81a4bSJacob Faibussowitsch    use_default_path = True
11487ce81a4bSJacob Faibussowitsch    search_path      = ''
11497ce81a4bSJacob Faibussowitsch
11507ce81a4bSJacob Faibussowitsch    log_print('{} to find an executable'.format('REQUIRED' if required else 'NOT required'))
11517ce81a4bSJacob Faibussowitsch    if arg_opt in {'auto', 'default-auto', '1'}:
11527ce81a4bSJacob Faibussowitsch      # detect it based on the C language compiler, hopefully this does not clash!
1153095fb05fSJacob Faibussowitsch      lang     = self.setCompilers.languages.clanguage
1154095fb05fSJacob Faibussowitsch      compiler = self.getCompiler(lang=lang)
11557ce81a4bSJacob Faibussowitsch      log_print('User did not explicitly set coverage exec (got {}), trying to auto-detect based on compiler {}'.format(quoted(arg_opt), quoted(compiler)))
11567ce81a4bSJacob Faibussowitsch      if self.setCompilers.isGNU(compiler, self.log):
1157095fb05fSJacob Faibussowitsch        compiler_version_re = re.compile(r'[gG][cC\+\-]+[0-9]* \(.+\) (\d+)\.(\d+)\.(\d+)')
11587ce81a4bSJacob Faibussowitsch        exec_names          = ['gcov']
11597ce81a4bSJacob Faibussowitsch      elif self.setCompilers.isClang(compiler, self.log):
1160095fb05fSJacob Faibussowitsch        compiler_version_re = re.compile(r'clang version (\d+)\.(\d+)\.(\d+)')
11617ce81a4bSJacob Faibussowitsch        exec_names          = ['llvm-cov']
11627ce81a4bSJacob Faibussowitsch        if self.setCompilers.isDarwin(self.log):
11637ce81a4bSJacob Faibussowitsch          # macOS masquerades llvm-cov as just 'gcov', so we add this to the list in case
11647ce81a4bSJacob Faibussowitsch          # bare llvm-cov does not work
11657ce81a4bSJacob Faibussowitsch          exec_names.append('gcov')
11667ce81a4bSJacob Faibussowitsch      elif arg_opt == 'default-auto' and not required:
11677ce81a4bSJacob Faibussowitsch        # default-auto implies the user did not set it via command line!
11687ce81a4bSJacob 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)))
11697ce81a4bSJacob Faibussowitsch        return
11707ce81a4bSJacob Faibussowitsch      else:
11717ce81a4bSJacob Faibussowitsch        # implies 'auto' explicitly set by user, or we were required to find
11727ce81a4bSJacob Faibussowitsch        # something. either way we should error
11737ce81a4bSJacob Faibussowitsch        raise RuntimeError('Could not auto-detect coverage tool for {}, please set coverage tool name explicitly'.format(quoted(compiler)))
1174095fb05fSJacob Faibussowitsch
1175095fb05fSJacob Faibussowitsch      try:
1176095fb05fSJacob Faibussowitsch        compiler_version_str = self.compilerFlags.version[lang]
1177095fb05fSJacob Faibussowitsch      except KeyError:
1178095fb05fSJacob Faibussowitsch        compiler_version_str = 'Unknown'
1179095fb05fSJacob Faibussowitsch
1180095fb05fSJacob Faibussowitsch      log_print('Searching version string {} (for compiler {}) using pattern {}'.format(quoted(compiler_version_str), quoted(compiler), quoted(compiler_version_re.pattern)))
1181095fb05fSJacob Faibussowitsch      compiler_version = compiler_version_re.search(compiler_version_str)
1182095fb05fSJacob Faibussowitsch      if compiler_version is not None:
1183095fb05fSJacob Faibussowitsch        log_print('Found major = {}, minor = {}, patch = {}'.format(compiler_version.group(1), compiler_version.group(2), compiler_version.group(3)))
1184095fb05fSJacob Faibussowitsch        # form [llvm-cov-14, llvm-cov-14.0, llvm-cov, etc.]
1185095fb05fSJacob Faibussowitsch        cov_exec_name = exec_names[0]
1186095fb05fSJacob Faibussowitsch        exec_names    = [
1187095fb05fSJacob Faibussowitsch          # llvm-cov-14
1188095fb05fSJacob Faibussowitsch          '{}-{}'.format(cov_exec_name, compiler_version.group(1)),
1189095fb05fSJacob Faibussowitsch           # llvm-cov-14.0
1190095fb05fSJacob Faibussowitsch          '{}-{}.{}'.format(cov_exec_name, compiler_version.group(1), compiler_version.group(2))
1191095fb05fSJacob Faibussowitsch        ] + exec_names
11927ce81a4bSJacob Faibussowitsch    else:
11937ce81a4bSJacob Faibussowitsch      log_print('User explicitly set coverage exec as {}'.format(quoted(arg_opt)))
11947ce81a4bSJacob Faibussowitsch      par_dir = os.path.dirname(arg_opt)
11957ce81a4bSJacob Faibussowitsch      if os.path.exists(par_dir):
11967ce81a4bSJacob Faibussowitsch        # arg_opt is path-like, we should only search the provided directory when we go
11977ce81a4bSJacob Faibussowitsch        # looking for the tool
11987ce81a4bSJacob Faibussowitsch        use_default_path = False
11997ce81a4bSJacob Faibussowitsch        search_path      = par_dir
12007ce81a4bSJacob Faibussowitsch      exec_names = [arg_opt]
12017ce81a4bSJacob Faibussowitsch
12027ce81a4bSJacob Faibussowitsch    make_macro_name = 'PETSC_COVERAGE_EXEC'
12037ce81a4bSJacob Faibussowitsch    log_print('Checking for coverage tool(s):\n{}'.format('\n'.join('- '+t for t in exec_names)))
12047ce81a4bSJacob Faibussowitsch    found_exec = self.getExecutables(
12057ce81a4bSJacob Faibussowitsch      exec_names,
12067ce81a4bSJacob Faibussowitsch      path=search_path, getFullPath=True, useDefaultPath=use_default_path, resultName=make_macro_name
12077ce81a4bSJacob Faibussowitsch    )
12087ce81a4bSJacob Faibussowitsch
12097ce81a4bSJacob Faibussowitsch    if found_exec is None:
12107ce81a4bSJacob Faibussowitsch      # didn't find the coverage tool
12117ce81a4bSJacob Faibussowitsch      if required:
12127ce81a4bSJacob Faibussowitsch        raise RuntimeError('Coverage tool(s) {} could not be found. Please provide explicit path to coverage tool'.format(exec_names))
12137ce81a4bSJacob Faibussowitsch      return
12147ce81a4bSJacob Faibussowitsch
12157ce81a4bSJacob Faibussowitsch    found_exec_name = os.path.basename(found_exec)
12167ce81a4bSJacob Faibussowitsch    if 'llvm-cov' in found_exec_name and 'gcov' not in found_exec_name:
12177ce81a4bSJacob Faibussowitsch      # llvm-cov needs to be called as 'llvm-cov gcov' to work
12187ce81a4bSJacob Faibussowitsch      self.addMakeMacro(make_macro_name, found_exec + ' gcov')
12197ce81a4bSJacob Faibussowitsch    return
12207ce81a4bSJacob Faibussowitsch
1221689a5dfaSJacob Faibussowitsch  def configureStrictPetscErrorCode(self):
1222689a5dfaSJacob Faibussowitsch    """
1223689a5dfaSJacob Faibussowitsch    Enables or disables strict PetscErrorCode checking.
1224689a5dfaSJacob Faibussowitsch
1225689a5dfaSJacob Faibussowitsch    If --with-strict-petscerrorcode = 1:
1226689a5dfaSJacob Faibussowitsch    - defines PETSC_USE_STRICT_PETSCERRORCODE to 1
1227689a5dfaSJacob Faibussowitsch
1228689a5dfaSJacob Faibussowitsch    Else:
1229689a5dfaSJacob Faibussowitsch    - deletes any prior PETSC_USE_STRICT_PETSCERRORCODE definitions (if they exist)
1230689a5dfaSJacob Faibussowitsch    """
1231689a5dfaSJacob Faibussowitsch    define_name = 'USE_STRICT_PETSCERRORCODE'
1232689a5dfaSJacob Faibussowitsch    if self.argDB['with-strict-petscerrorcode']:
1233689a5dfaSJacob Faibussowitsch      self.addDefine(define_name, 1)
1234689a5dfaSJacob Faibussowitsch    else:
1235689a5dfaSJacob Faibussowitsch      # in case it was somehow added previously
1236689a5dfaSJacob Faibussowitsch      self.delDefine(define_name)
1237689a5dfaSJacob Faibussowitsch    return
1238689a5dfaSJacob Faibussowitsch
1239f8833479SBarry Smith#-----------------------------------------------------------------------------------------------------
1240b10d012aSSatish Balay  def configureCygwinBrokenPipe(self):
1241b10d012aSSatish Balay    '''Cygwin version <= 1.7.18 had issues with pipes and long commands invoked from gnu-make
1242b10d012aSSatish Balay    http://cygwin.com/ml/cygwin/2013-05/msg00340.html '''
12437fca349cSMatthew G. Knepley    if config.setCompilers.Configure.isCygwin(self.log):
1244b10d012aSSatish Balay      import platform
1245b10d012aSSatish Balay      import re
1246b10d012aSSatish Balay      r=re.compile("([0-9]+).([0-9]+).([0-9]+)")
1247b10d012aSSatish Balay      m=r.match(platform.release())
1248b10d012aSSatish Balay      major=int(m.group(1))
1249b10d012aSSatish Balay      minor=int(m.group(2))
1250b10d012aSSatish Balay      subminor=int(m.group(3))
1251b10d012aSSatish Balay      if ((major < 1) or (major == 1 and minor < 7) or (major == 1 and minor == 7 and subminor <= 18)):
1252b10d012aSSatish Balay        self.addMakeMacro('PETSC_CYGWIN_BROKEN_PIPE','1')
1253b10d012aSSatish Balay    return
1254b10d012aSSatish Balay
1255b10d012aSSatish Balay#-----------------------------------------------------------------------------------------------------
1256569865ddSSatish Balay  def configureDefaultArch(self):
1257af0996ceSBarry Smith    conffile = os.path.join('lib','petsc','conf', 'petscvariables')
1258569865ddSSatish Balay    if self.framework.argDB['with-default-arch']:
1259c6ef1b5bSJed Brown      fd = open(conffile, 'w')
1260569865ddSSatish Balay      fd.write('PETSC_ARCH='+self.arch.arch+'\n')
1261da93591fSBarry Smith      fd.write('PETSC_DIR='+self.petscdir.dir+'\n')
1262b9b902edSJed Brown      fd.write('include '+os.path.join('$(PETSC_DIR)','$(PETSC_ARCH)','lib','petsc','conf','petscvariables')+'\n')
1263569865ddSSatish Balay      fd.close()
1264569865ddSSatish Balay      self.framework.actions.addArgument('PETSc', 'Build', 'Set default architecture to '+self.arch.arch+' in '+conffile)
1265569865ddSSatish Balay    elif os.path.isfile(conffile):
1266569865ddSSatish Balay      try:
1267569865ddSSatish Balay        os.unlink(conffile)
1268569865ddSSatish Balay      except:
1269569865ddSSatish Balay        raise RuntimeError('Unable to remove file '+conffile+'. Did a different user create it?')
1270569865ddSSatish Balay    return
1271569865ddSSatish Balay
1272569865ddSSatish Balay#-----------------------------------------------------------------------------------------------------
1273f8833479SBarry Smith  def configureScript(self):
1274f8833479SBarry Smith    '''Output a script in the conf directory which will reproduce the configuration'''
1275f8833479SBarry Smith    import nargs
1276495bf1a9SSatish Balay    import sys
1277af0996ceSBarry Smith    scriptName = os.path.join(self.arch.arch,'lib','petsc','conf', 'reconfigure-'+self.arch.arch+'.py')
1278f8833479SBarry Smith    args = dict([(nargs.Arg.parseArgument(arg)[0], arg) for arg in self.framework.clArgs])
1279e97fc2efSSatish Balay    if 'with-clean' in args:
1280e97fc2efSSatish Balay      del args['with-clean']
1281d418e2d7SSatish Balay    if 'force' in args:
1282d418e2d7SSatish Balay      del args['force']
1283f8833479SBarry Smith    if 'configModules' in args:
12841063a081SSatish Balay      if nargs.Arg.parseArgument(args['configModules'])[1] == 'PETSc.Configure':
1285f8833479SBarry Smith        del args['configModules']
1286f8833479SBarry Smith    if 'optionsModule' in args:
128723a19ef1SSatish Balay      if nargs.Arg.parseArgument(args['optionsModule'])[1] == 'config.compilerOptions':
1288f8833479SBarry Smith        del args['optionsModule']
1289f8833479SBarry Smith    if not 'PETSC_ARCH' in args:
12901063a081SSatish Balay      args['PETSC_ARCH'] = 'PETSC_ARCH='+str(self.arch.arch)
1291c6ef1b5bSJed Brown    f = open(scriptName, 'w')
1292495bf1a9SSatish Balay    f.write('#!'+sys.executable+'\n')
1293f8833479SBarry Smith    f.write('if __name__ == \'__main__\':\n')
1294f8833479SBarry Smith    f.write('  import sys\n')
12957561c02cSSatish Balay    f.write('  import os\n')
12967561c02cSSatish Balay    f.write('  sys.path.insert(0, os.path.abspath(\'config\'))\n')
1297f8833479SBarry Smith    f.write('  import configure\n')
12981063a081SSatish Balay    # pretty print repr(args.values())
12991063a081SSatish Balay    f.write('  configure_options = [\n')
13008bec23c5SJed Brown    for itm in sorted(args.values()):
13011063a081SSatish Balay      f.write('    \''+str(itm)+'\',\n')
13021063a081SSatish Balay    f.write('  ]\n')
1303f8833479SBarry Smith    f.write('  configure.petsc_configure(configure_options)\n')
1304f8833479SBarry Smith    f.close()
1305f8833479SBarry Smith    try:
13065b6bfdb9SJed Brown      os.chmod(scriptName, 0o775)
13075b6bfdb9SJed Brown    except OSError as e:
1308f8833479SBarry Smith      self.framework.logPrint('Unable to make reconfigure script executable:\n'+str(e))
1309f8833479SBarry Smith    self.framework.actions.addArgument('PETSc', 'File creation', 'Created '+scriptName+' for automatic reconfiguration')
1310f8833479SBarry Smith    return
1311f8833479SBarry Smith
1312f8833479SBarry Smith  def configureInstall(self):
1313f8833479SBarry Smith    '''Setup the directories for installation'''
1314f8833479SBarry Smith    if self.framework.argDB['prefix']:
13155b4fc442SVaclav Hapla      self.addMakeRule('print_mesg_after_build','',
13165b4fc442SVaclav Hapla       ['-@echo "========================================="',
13175b4fc442SVaclav Hapla        '-@echo "Now to install the libraries do:"',
13185b4fc442SVaclav Hapla        '-@echo "%s${MAKE_USER} PETSC_DIR=${PETSC_DIR} PETSC_ARCH=${PETSC_ARCH} install"' % self.installdir.installSudo,
1319315b77e6SSatish Balay        '-@echo "========================================="'])
1320f8833479SBarry Smith    else:
13215b4fc442SVaclav Hapla      self.addMakeRule('print_mesg_after_build','',
13225b4fc442SVaclav Hapla       ['-@echo "========================================="',
13235b4fc442SVaclav Hapla        '-@echo "Now to check if the libraries are working do:"',
13245b4fc442SVaclav Hapla        '-@echo "${MAKE_USER} PETSC_DIR=${PETSC_DIR} PETSC_ARCH=${PETSC_ARCH} check"',
1325315b77e6SSatish Balay        '-@echo "========================================="'])
1326f8833479SBarry Smith      return
1327f8833479SBarry Smith
132828bb2e72SSatish Balay  def postProcessPackages(self):
132928bb2e72SSatish Balay    postPackages=[]
133028bb2e72SSatish Balay    for i in self.framework.packages:
133128bb2e72SSatish Balay      if hasattr(i,'postProcess'): postPackages.append(i)
133228bb2e72SSatish Balay    if postPackages:
1333e64d19dfSSatish Balay      # ctetgen needs petsc conf files. so attempt to create them early
1334a77eb93bSSatish Balay      self.framework.dumpConfFiles()
1335d9293e7bSBarry Smith      # tacky fix for dependency of Aluimia on Pflotran; requested via petsc-dev Matt provide a correct fix
1336d9293e7bSBarry Smith      for i in postPackages:
1337d9293e7bSBarry Smith        if i.name.upper() in ['PFLOTRAN']:
1338d9293e7bSBarry Smith          i.postProcess()
1339d9293e7bSBarry Smith          postPackages.remove(i)
134028bb2e72SSatish Balay      for i in postPackages: i.postProcess()
1341aa5c8b8eSBarry Smith      for i in postPackages:
1342aa5c8b8eSBarry Smith        if i.installedpetsc:
1343aa5c8b8eSBarry Smith          self.installed = 1
1344aa5c8b8eSBarry Smith          break
134528bb2e72SSatish Balay    return
1346f8833479SBarry Smith
1347f8833479SBarry Smith  def configure(self):
1348bf3e94a3SBarry Smith    if 'package-prefix-hash' in self.argDB:
1349bf3e94a3SBarry Smith      # turn off prefix if it was only used to for installing external packages.
1350bf3e94a3SBarry Smith      self.framework.argDB['prefix'] = ''
1351bf3e94a3SBarry Smith      self.dir = os.path.abspath(os.path.join(self.petscdir.dir, self.arch.arch))
1352bf3e94a3SBarry Smith      self.installdir.dir = self.dir
1353bf3e94a3SBarry Smith      self.installdir.petscDir = self.petscdir.dir
1354bf3e94a3SBarry Smith      self.petscDir = self.petscdir.dir
1355bf3e94a3SBarry Smith      self.petscArch = self.arch.arch
1356bf3e94a3SBarry Smith      self.addMakeMacro('PREFIXDIR',self.dir)
1357bf3e94a3SBarry Smith      self.confDir = os.path.abspath(os.path.join(self.petscdir.dir, self.arch.arch))
1358bf3e94a3SBarry Smith
1359f8833479SBarry Smith    if not os.path.samefile(self.petscdir.dir, os.getcwd()):
1360f8833479SBarry Smith      raise RuntimeError('Wrong PETSC_DIR option specified: '+str(self.petscdir.dir) + '\n  Configure invoked in: '+os.path.realpath(os.getcwd()))
1361550489e3SMatthew 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):
13623552d8fbSSatish Balay      raise RuntimeError('Incorrect option --prefix='+self.framework.argDB['prefix']+' specified. It cannot be same as PETSC_DIR!')
13638fd0dbdbSBarry Smith    if self.framework.argDB['prefix'] and self.framework.argDB['prefix'].find(' ') > -1:
13648fd0dbdbSBarry 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')
1365c16c35a9SSatish 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)):
1366c16c35a9SSatish Balay      raise RuntimeError('Incorrect option --prefix='+self.framework.argDB['prefix']+' specified. It cannot be same as PETSC_DIR/PETSC_ARCH!')
1367f16c1317SJed Brown    self.framework.header          = os.path.join(self.arch.arch,'include','petscconf.h')
1368f16c1317SJed Brown    self.framework.cHeader         = os.path.join(self.arch.arch,'include','petscfix.h')
1369bf113f49SJacob Faibussowitsch    self.framework.poisonheader    = os.path.join(self.arch.arch,'include','petscconf_poison.h')
13709c735a01SStefano Zampini    self.framework.pkgheader       = os.path.join(self.arch.arch,'include','petscpkg_version.h')
1371af0996ceSBarry Smith    self.framework.makeMacroHeader = os.path.join(self.arch.arch,'lib','petsc','conf','petscvariables')
1372af0996ceSBarry Smith    self.framework.makeRuleHeader  = os.path.join(self.arch.arch,'lib','petsc','conf','petscrules')
1373f8833479SBarry Smith    if self.libraries.math is None:
1374f8833479SBarry Smith      raise RuntimeError('PETSc requires a functional math library. Please send configure.log to petsc-maint@mcs.anl.gov.')
1375f8833479SBarry Smith    if self.languages.clanguage == 'Cxx' and not hasattr(self.compilers, 'CXX'):
1376f8833479SBarry Smith      raise RuntimeError('Cannot set C language to C++ without a functional C++ compiler.')
1377ed938b00SJed Brown    self.executeTest(self.configureRTLDDefault)
1378b2843cf1SBarry Smith    self.executeTest(self.configurePrefetch)
13792400fdedSBarry Smith    self.executeTest(self.configureUnused)
13801ef8df7fSJed Brown    self.executeTest(self.configureDeprecated)
138198ed35c3SBarry Smith    self.executeTest(self.configureIsatty)
1382e8e972b2SVaclav Hapla    self.executeTest(self.configureExpect)
1383e8e972b2SVaclav Hapla    self.executeTest(self.configureAlign)
1384e8e972b2SVaclav Hapla    self.executeTest(self.configureFunctionName)
1385e8e972b2SVaclav Hapla    self.executeTest(self.configureIntptrt)
1386f8833479SBarry Smith    self.executeTest(self.configureSolaris)
1387f8833479SBarry Smith    self.executeTest(self.configureLinux)
13880f64ec89SBarry Smith    self.executeTest(self.configureDarwin)
1389f8833479SBarry Smith    self.executeTest(self.configureWin32)
1390b10d012aSSatish Balay    self.executeTest(self.configureCygwinBrokenPipe)
1391569865ddSSatish Balay    self.executeTest(self.configureDefaultArch)
1392f8833479SBarry Smith    self.executeTest(self.configureScript)
1393f8833479SBarry Smith    self.executeTest(self.configureInstall)
139409bc878fSSatish Balay    self.executeTest(self.configureAtoll)
13959b81490aSJacob Faibussowitsch    self.executeTest(self.configureCoverage)
13967ce81a4bSJacob Faibussowitsch    self.executeTest(self.configureCoverageExecutable)
1397689a5dfaSJacob Faibussowitsch    self.executeTest(self.configureStrictPetscErrorCode)
1398f8833479SBarry Smith
1399f8833479SBarry Smith    self.Dump()
1400f8833479SBarry Smith    self.dumpConfigInfo()
14012a4161d9SMatthew G Knepley    self.dumpMachineInfo()
140249fe22e6SSatish Balay    self.delGenFiles()
140340277576SBarry Smith    # need to save the current state of BuildSystem so that postProcess() packages can read it in and perhaps run make install
140440277576SBarry Smith    self.framework.storeSubstitutions(self.framework.argDB)
1405492432c8SJed Brown    self.framework.argDB['configureCache'] = pickle.dumps(self.framework)
140640277576SBarry Smith    self.framework.argDB.save(force = True)
14078244ab14SJed Brown    self.DumpPkgconfig('PETSc.pc')
14088244ab14SJed Brown    self.DumpPkgconfig('petsc.pc')
1409351d3a41SMatthew G Knepley    self.DumpModule()
1410f7ad81e1SBarry Smith    self.postProcessPackages()
1411f8833479SBarry Smith    self.framework.log.write('================================================================================\n')
1412f8833479SBarry Smith    self.logClear()
1413f8833479SBarry Smith    return
1414