xref: /petsc/config/PETSc/Configure.py (revision a0c7f9aa3b59be96b422b2e54790cd20c111f46e)
1f8833479SBarry Smithimport config.base
2f8833479SBarry Smith
3f8833479SBarry Smithimport os
46dd73af6SBarry Smithimport sys
5f8833479SBarry Smithimport re
6492432c8SJed Brownimport pickle
7f8833479SBarry Smith
8f8833479SBarry Smithclass Configure(config.base.Configure):
9f8833479SBarry Smith  def __init__(self, framework):
10f8833479SBarry Smith    config.base.Configure.__init__(self, framework)
11f8833479SBarry Smith    self.headerPrefix = 'PETSC'
12f8833479SBarry Smith    self.substPrefix  = 'PETSC'
13aa5c8b8eSBarry Smith    self.installed    = 0 # 1 indicates that Configure itself has already compiled and installed PETSc
144161761dSBarry Smith    self.found        = 1
15f8833479SBarry Smith    return
16f8833479SBarry Smith
177c939e48SSatish Balay  def __str2__(self):
185b4fc442SVaclav Hapla    desc = ['  Using GNU make: ' + self.make.make]
197ce81a4bSJacob Faibussowitsch    if self.defines.get('USE_COVERAGE'):
207ce81a4bSJacob Faibussowitsch      desc.extend([
217ce81a4bSJacob Faibussowitsch        '  Code coverage: yes',
227ce81a4bSJacob Faibussowitsch        '  Using code coverage executable: {}'.format(self.getMakeMacro('PETSC_COVERAGE_EXEC'))
237ce81a4bSJacob Faibussowitsch      ])
24aa5c8b8eSBarry Smith    if not self.installed:
25a0022257SSatish Balay      desc.append('xxx=========================================================================xxx')
26dc0529c6SBarry Smith      desc.append(' Configure stage complete. Now build PETSc libraries with:')
275b4fc442SVaclav Hapla      desc.append('   %s PETSC_DIR=%s PETSC_ARCH=%s all' % (self.make.make_user, self.petscdir.dir, self.arch.arch))
28a0022257SSatish Balay      desc.append('xxx=========================================================================xxx')
29aa5c8b8eSBarry Smith    else:
30aa5c8b8eSBarry Smith      desc.append('xxx=========================================================================xxx')
31aa5c8b8eSBarry Smith      desc.append(' Installation complete. You do not need to run make to compile or install the software')
32aa5c8b8eSBarry Smith      desc.append('xxx=========================================================================xxx')
337ce81a4bSJacob Faibussowitsch    desc.append('')
347ce81a4bSJacob Faibussowitsch    return '\n'.join(desc)
35f8833479SBarry Smith
36f8833479SBarry Smith  def setupHelp(self, help):
37f8833479SBarry Smith    import nargs
38ce0b2093SBarry Smith    help.addArgument('PETSc',  '-prefix=<dir>',                              nargs.Arg(None, '', 'Specifiy location to install PETSc (eg. /usr/local)'))
397deb5ab3SBarry Smith    help.addArgument('PETSc',  '-with-prefetch=<bool>',                      nargs.ArgBool(None, 1,'Enable checking for prefetch instructions'))
40eed94e11SSatish Balay    help.addArgument('Windows','-with-windows-graphics=<bool>',              nargs.ArgBool(None, 1,'Enable check for Windows Graphics'))
41569865ddSSatish Balay    help.addArgument('PETSc', '-with-default-arch=<bool>',                   nargs.ArgBool(None, 1, 'Allow using the last configured arch without setting PETSC_ARCH'))
4257cb31baSSatish Balay    help.addArgument('PETSc','-with-single-library=<bool>',                  nargs.ArgBool(None, 1,'Put all PETSc code into the single -lpetsc library'))
43cb297985SSatish Balay    help.addArgument('PETSc','-with-fortran-bindings=<bool>',                nargs.ArgBool(None, 1,'Build PETSc fortran bindings in the library and corresponding module files'))
44525d6f2eSBarry Smith    help.addArgument('PETSc', '-with-ios=<bool>',                            nargs.ArgBool(None, 0, 'Build an iPhone/iPad version of PETSc library'))
4513f9d092SSatish Balay    help.addArgument('PETSc', '-with-display=<x11display>',                  nargs.Arg(None, '', 'Specifiy DISPLAY env variable for use with matlab test)'))
462c30b4dfSSatish Balay    help.addArgument('PETSc', '-with-package-scripts=<pyscripts>',           nargs.ArgFileList(None,None,'Specify configure package scripts for user provided packages'))
477ce81a4bSJacob Faibussowitsch    help.addArgument('PETSc', '-with-coverage=<bool>',                       nargs.ArgFuzzyBool(None, value=0, help='Enable or disable code-coverage collection'))
487ce81a4bSJacob Faibussowitsch    help.addArgument('PETSc', '-with-coverage-exec=<executable>',            nargs.ArgExecutable(None, value='default-auto', mustExist=0, help='Name of executable to use for post-processing coverage data, e.g. \'gcov\' or \'llvm-cov\'. Pass \'auto\' to let configure infer from compiler'))
49*a0c7f9aaSSamuel Khuvis    help.addArgument('PETSc', '-with-tau-perfstubs=<bool>',                  nargs.ArgBool(None, 1,'Enable TAU profiler stubs'))
50f8833479SBarry Smith    return
51f8833479SBarry Smith
526dd73af6SBarry Smith  def registerPythonFile(self,filename,directory):
536dd73af6SBarry Smith    ''' Add a python file to the framework and registers its headerprefix, ... externalpackagedir
546dd73af6SBarry Smith        directory is the directory where the file relative to the BuildSystem or config path in python notation with . '''
556dd73af6SBarry Smith    (utilityName, ext) = os.path.splitext(filename)
566dd73af6SBarry Smith    if not utilityName.startswith('.') and not utilityName.startswith('#') and ext == '.py' and not utilityName == '__init__':
576dd73af6SBarry Smith      if directory: directory = directory+'.'
586dd73af6SBarry Smith      utilityObj                             = self.framework.require(directory+utilityName, self)
596dd73af6SBarry Smith      utilityObj.headerPrefix                = self.headerPrefix
606dd73af6SBarry Smith      utilityObj.archProvider                = self.arch
616dd73af6SBarry Smith      utilityObj.languageProvider            = self.languages
626dd73af6SBarry Smith      utilityObj.installDirProvider          = self.installdir
636dd73af6SBarry Smith      utilityObj.externalPackagesDirProvider = self.externalpackagesdir
646dd73af6SBarry Smith      utilityObj.precisionProvider           = self.scalartypes
656dd73af6SBarry Smith      utilityObj.indexProvider               = self.indexTypes
666dd73af6SBarry Smith      setattr(self, utilityName.lower(), utilityObj)
6751294b80SMatthew G. Knepley      return utilityObj
6851294b80SMatthew G. Knepley    return None
696dd73af6SBarry Smith
70f8833479SBarry Smith  def setupDependencies(self, framework):
71f8833479SBarry Smith    config.base.Configure.setupDependencies(self, framework)
72dca78d2bSSatish Balay    self.programs      = framework.require('config.programs',           self)
73f8833479SBarry Smith    self.setCompilers  = framework.require('config.setCompilers',       self)
747ce81a4bSJacob Faibussowitsch    self.compilerFlags = framework.require('config.compilerFlags',      self)
7530b8aa07SMatthew G. Knepley    self.compilers     = framework.require('config.compilers',          self)
769d310bb7SBarry Smith    self.arch          = framework.require('PETSc.options.arch',        self.setCompilers)
779d310bb7SBarry Smith    self.petscdir      = framework.require('PETSc.options.petscdir',    self.arch)
789d310bb7SBarry Smith    self.installdir    = framework.require('PETSc.options.installDir',  self)
794e00a515SSatish Balay    self.dataFilesPath = framework.require('PETSc.options.dataFilesPath',self)
806dd73af6SBarry Smith    self.scalartypes   = framework.require('PETSc.options.scalarTypes', self)
816dd73af6SBarry Smith    self.indexTypes    = framework.require('PETSc.options.indexTypes',  self)
829d310bb7SBarry Smith    self.languages     = framework.require('PETSc.options.languages',   self.setCompilers)
8330b8aa07SMatthew G. Knepley    self.indexTypes    = framework.require('PETSc.options.indexTypes',  self.compilers)
84f8833479SBarry Smith    self.types         = framework.require('config.types',              self)
85f8833479SBarry Smith    self.headers       = framework.require('config.headers',            self)
86f8833479SBarry Smith    self.functions     = framework.require('config.functions',          self)
87f8833479SBarry Smith    self.libraries     = framework.require('config.libraries',          self)
88cd37d877SShri Abhyankar    self.atomics       = framework.require('config.atomics',            self)
899481793eSSatish Balay    self.make          = framework.require('config.packages.make',      self)
909552296fSBarry Smith    self.blasLapack    = framework.require('config.packages.BlasLapack',self)
91e6b0c433SBarry Smith    self.mpi           = framework.require('config.packages.MPI',       self)
920542e31aSBarry Smith    self.fortran       = framework.require('config.compilersFortran',   self)
930542e31aSBarry Smith    self.externalpackagesdir = framework.require('PETSc.options.externalpackagesdir',self)
9449d43ecaSSatish Balay
9509a6cbfcSBernhard M. Wiedemann    for utility in sorted(os.listdir(os.path.join('config','PETSc','options'))):
966dd73af6SBarry Smith      self.registerPythonFile(utility,'PETSc.options')
979d310bb7SBarry Smith
9809a6cbfcSBernhard M. Wiedemann    for utility in sorted(os.listdir(os.path.join('config','BuildSystem','config','utilities'))):
996dd73af6SBarry Smith      self.registerPythonFile(utility,'config.utilities')
10006e08bc7SBarry Smith
10109a6cbfcSBernhard M. Wiedemann    for package in sorted(os.listdir(os.path.join('config', 'BuildSystem', 'config', 'packages'))):
10251294b80SMatthew G. Knepley      obj = self.registerPythonFile(package,'config.packages')
10351294b80SMatthew G. Knepley      if obj:
10451294b80SMatthew G. Knepley        obj.archProvider                = self.framework.requireModule(obj.archProvider, obj)
10551294b80SMatthew G. Knepley        obj.languageProvider            = self.framework.requireModule(obj.languageProvider, obj)
10651294b80SMatthew G. Knepley        obj.installDirProvider          = self.framework.requireModule(obj.installDirProvider, obj)
10751294b80SMatthew G. Knepley        obj.externalPackagesDirProvider = self.framework.requireModule(obj.externalPackagesDirProvider, obj)
10851294b80SMatthew G. Knepley        obj.precisionProvider           = self.framework.requireModule(obj.precisionProvider, obj)
10951294b80SMatthew G. Knepley        obj.indexProvider               = self.framework.requireModule(obj.indexProvider, obj)
1106dd73af6SBarry Smith
1115faf1eacSMatthew G. Knepley    # Force blaslapack and opencl to depend on scalarType so precision is set before BlasLapack is built
1129d310bb7SBarry Smith    framework.require('PETSc.options.scalarTypes', self.f2cblaslapack)
1139d310bb7SBarry Smith    framework.require('PETSc.options.scalarTypes', self.fblaslapack)
1149d310bb7SBarry Smith    framework.require('PETSc.options.scalarTypes', self.blaslapack)
1155faf1eacSMatthew G. Knepley    framework.require('PETSc.options.scalarTypes', self.opencl)
116f8833479SBarry Smith
117dca78d2bSSatish Balay    self.programs.headerPrefix     = self.headerPrefix
1185b3958d3SJacob Faibussowitsch    self.setCompilers.headerPrefix = self.headerPrefix
119f8833479SBarry Smith    self.compilers.headerPrefix    = self.headerPrefix
1200542e31aSBarry Smith    self.fortran.headerPrefix      = self.headerPrefix
121f8833479SBarry Smith    self.types.headerPrefix        = self.headerPrefix
122f8833479SBarry Smith    self.headers.headerPrefix      = self.headerPrefix
123f8833479SBarry Smith    self.functions.headerPrefix    = self.headerPrefix
124f8833479SBarry Smith    self.libraries.headerPrefix    = self.headerPrefix
1256dd73af6SBarry Smith
1262c30b4dfSSatish Balay    # Register user provided package scripts
1272c30b4dfSSatish Balay    if 'with-package-scripts' in self.framework.argDB:
1282c30b4dfSSatish Balay      for script in self.framework.argDB['with-package-scripts']:
1292c30b4dfSSatish Balay        if os.path.splitext(script)[1] != '.py':
1302c30b4dfSSatish Balay          raise RuntimeError('Only python scripts compatible with configure package script format should be specified! Invalid option -with-package-scripts='+script)
1312c30b4dfSSatish Balay        self.framework.logPrint('User is registering a new package script: '+script)
1322c30b4dfSSatish Balay        dname,fname = os.path.split(script)
1332c30b4dfSSatish Balay        if dname: sys.path.append(dname)
1342c30b4dfSSatish Balay        self.registerPythonFile(fname,'')
1356dd73af6SBarry Smith
1366dd73af6SBarry Smith    # test for a variety of basic headers and functions
1374211eb48SBarry Smith    headersC = map(lambda name: name+'.h',['setjmp','dos','fcntl','float','io','malloc','pwd','strings',
138ba61063dSBarry Smith                                            'unistd','sys/sysinfo','machine/endian','sys/param','sys/procfs','sys/resource',
1392475b7caSBarry Smith                                            'sys/systeminfo','sys/times','sys/utsname',
1407e4f0192SMosè Giordano                                            'sys/socket','sys/wait','netinet/in','netdb','direct','time','Ws2tcpip','sys/types',
1412475b7caSBarry Smith                                            'WindowsX','float','ieeefp','stdint','pthread','inttypes','immintrin','zmmintrin'])
14245082d64SJed Brown    functions = ['access','_access','clock','drand48','getcwd','_getcwd','getdomainname','gethostname',
143d7976ebaSJed Brown                 'getwd','posix_memalign','popen','PXFGETARG','rand','getpagesize',
1444211eb48SBarry Smith                 'readlink','realpath','usleep','sleep','_sleep',
1452475b7caSBarry Smith                 'uname','snprintf','_snprintf','lseek','_lseek','time','fork','stricmp',
1462475b7caSBarry Smith                 'strcasecmp','bzero','dlopen','dlsym','dlclose','dlerror',
147aaf3846cSSatish Balay                 '_set_output_format','_mkdir','socket','gethostbyname','_pipe','fpresetsticky',
1480866c665SSatish Balay                 'fpsetsticky','__gcov_dump','fstatat']
149b0651e32SBarry Smith    libraries = [(['fpe'],'handle_sigfpes')]
150b0651e32SBarry Smith    librariessock = [(['socket','nsl'],'socket')]
151f8833479SBarry Smith    self.headers.headers.extend(headersC)
152f8833479SBarry Smith    self.functions.functions.extend(functions)
153b0651e32SBarry Smith    self.libraries.libraries.extend(libraries)
154b0651e32SBarry Smith    if not hasattr(self,'socket'):
155b0651e32SBarry Smith      self.libraries.libraries.extend(librariessock)
156f8833479SBarry Smith    return
157f8833479SBarry Smith
1588244ab14SJed Brown  def DumpPkgconfig(self, petsc_pc):
159262119f8SBarry Smith    ''' Create a pkg-config file '''
160262119f8SBarry Smith    if not os.path.exists(os.path.join(self.petscdir.dir,self.arch.arch,'lib','pkgconfig')):
161262119f8SBarry Smith      os.makedirs(os.path.join(self.petscdir.dir,self.arch.arch,'lib','pkgconfig'))
1622eefe1c6SJed Brown    with open(os.path.join(self.petscdir.dir,self.arch.arch,'lib','pkgconfig',petsc_pc),'w') as fd:
1635e3311eeSJed Brown      cflags_inc = ['-I${includedir}']
164262119f8SBarry Smith      if self.framework.argDB['prefix']:
1655bb5b263SMatthew G. Knepley        fd.write('prefix='+self.installdir.dir+'\n')
166262119f8SBarry Smith      else:
167e1e675deSJed Brown        fd.write('prefix='+os.path.join(self.petscdir.dir, self.arch.arch)+'\n')
168e1e675deSJed Brown        cflags_inc.append('-I' + os.path.join(self.petscdir.dir, 'include'))
169262119f8SBarry Smith      fd.write('exec_prefix=${prefix}\n')
170262119f8SBarry Smith      fd.write('includedir=${prefix}/include\n')
1715e3311eeSJed Brown      fd.write('libdir=${prefix}/lib\n')
172262119f8SBarry Smith
1732eefe1c6SJed Brown      with self.setCompilers.Language('C'):
174262119f8SBarry Smith        fd.write('ccompiler='+self.setCompilers.getCompiler()+'\n')
175756c7f9fSJed Brown        fd.write('cflags_extra='+self.setCompilers.getCompilerFlags().strip()+'\n')
176756c7f9fSJed Brown        fd.write('cflags_dep='+self.compilers.dependenciesGenerationFlag.get('C','')+'\n')
177756c7f9fSJed Brown        fd.write('ldflag_rpath='+self.setCompilers.CSharedLinkerFlag+'\n')
17803e383c8SJed Brown      if hasattr(self.compilers, 'CXX'):
1792eefe1c6SJed Brown        with self.setCompilers.Language('C++'):
180262119f8SBarry Smith          fd.write('cxxcompiler='+self.setCompilers.getCompiler()+'\n')
181756c7f9fSJed Brown          fd.write('cxxflags_extra='+self.setCompilers.getCompilerFlags().strip()+'\n')
182262119f8SBarry Smith      if hasattr(self.compilers, 'FC'):
1832eefe1c6SJed Brown        with self.setCompilers.Language('FC'):
184262119f8SBarry Smith          fd.write('fcompiler='+self.setCompilers.getCompiler()+'\n')
185756c7f9fSJed Brown          fd.write('fflags_extra='+self.setCompilers.getCompilerFlags().strip()+'\n')
18650520af6SPatrick Sanan      if hasattr(self.compilers, 'CUDAC'):
18750520af6SPatrick Sanan        with self.setCompilers.Language('CUDA'):
18850520af6SPatrick Sanan          fd.write('cudacompiler='+self.setCompilers.getCompiler()+'\n')
18950520af6SPatrick Sanan          fd.write('cudaflags_extra='+self.setCompilers.getCompilerFlags().strip()+'\n')
1908f561fa3SPatrick Sanan          p = self.framework.require('config.packages.cuda')
19150520af6SPatrick Sanan          fd.write('cudalib='+self.libraries.toStringNoDupes(p.lib)+'\n')
19250520af6SPatrick Sanan          fd.write('cudainclude='+self.headers.toStringNoDupes(p.include)+'\n')
1937ba7a817SBarry Smith          if hasattr(self.setCompilers,'CUDA_CXX'):
1947ba7a817SBarry Smith            fd.write('cuda_cxx='+self.setCompilers.CUDA_CXX+'\n')
1957ba7a817SBarry Smith            fd.write('cuda_cxxflags='+self.setCompilers.CUDA_CXXFLAGS+'\n')
196262119f8SBarry Smith
197262119f8SBarry Smith      fd.write('\n')
198262119f8SBarry Smith      fd.write('Name: PETSc\n')
199262119f8SBarry Smith      fd.write('Description: Library to solve ODEs and algebraic equations\n')
200351d3a41SMatthew G Knepley      fd.write('Version: %s\n' % self.petscdir.version)
2015e3311eeSJed Brown      fd.write('Cflags: ' + ' '.join([self.setCompilers.CPPFLAGS] + cflags_inc) + '\n')
20237371b91SJed Brown      fd.write('Libs: '+self.libraries.toStringNoDupes(['-L${libdir}', self.petsclib], with_rpath=False)+'\n')
2038ebf8858SJed Brown      # Remove RPATH flags from library list.  User can add them using
2048ebf8858SJed Brown      # pkg-config --variable=ldflag_rpath and pkg-config --libs-only-L
205de8f682fSSatish 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')
206262119f8SBarry Smith    return
207262119f8SBarry Smith
208351d3a41SMatthew G Knepley  def DumpModule(self):
209351d3a41SMatthew G Knepley    ''' Create a module file '''
210af0996ceSBarry Smith    if not os.path.exists(os.path.join(self.petscdir.dir,self.arch.arch,'lib','petsc','conf','modules')):
211af0996ceSBarry Smith      os.makedirs(os.path.join(self.petscdir.dir,self.arch.arch,'lib','petsc','conf','modules'))
212af0996ceSBarry Smith    if not os.path.exists(os.path.join(self.petscdir.dir,self.arch.arch,'lib','petsc','conf','modules','petsc')):
213af0996ceSBarry Smith      os.makedirs(os.path.join(self.petscdir.dir,self.arch.arch,'lib','petsc','conf','modules','petsc'))
214351d3a41SMatthew G Knepley    if self.framework.argDB['prefix']:
2155bb5b263SMatthew G. Knepley      installdir  = self.installdir.dir
21655d606a3SSatish Balay      installarch = ''
21755d606a3SSatish Balay      installpath = os.path.join(installdir,'bin')
218351d3a41SMatthew G Knepley    else:
219351d3a41SMatthew G Knepley      installdir  = self.petscdir.dir
22055d606a3SSatish Balay      installarch = self.arch.arch
22155d606a3SSatish Balay      installpath = os.path.join(installdir,installarch,'bin')+':'+os.path.join(installdir,'bin')
222af0996ceSBarry Smith    fd = open(os.path.join(self.petscdir.dir,self.arch.arch,'lib','petsc','conf','modules','petsc',self.petscdir.version),'w')
223351d3a41SMatthew G Knepley    fd.write('''\
224351d3a41SMatthew G Knepley#%%Module
225351d3a41SMatthew G Knepley
226351d3a41SMatthew G Knepleyproc ModulesHelp { } {
227351d3a41SMatthew G Knepley    puts stderr "This module sets the path and environment variables for petsc-%s"
228a17b96a8SKyle Gerard Felker    puts stderr "     see https://petsc.org/ for more information      "
229351d3a41SMatthew G Knepley    puts stderr ""
230351d3a41SMatthew G Knepley}
231351d3a41SMatthew G Knepleymodule-whatis "PETSc - Portable, Extensible Toolkit for Scientific Computation"
232351d3a41SMatthew G Knepley
233dd486775SJed Brownset petsc_dir   "%s"
234dd486775SJed Brownset petsc_arch  "%s"
235351d3a41SMatthew G Knepley
236dd486775SJed Brownsetenv PETSC_ARCH "$petsc_arch"
237dd486775SJed Brownsetenv PETSC_DIR "$petsc_dir"
238dd486775SJed Brownprepend-path PATH "%s"
23955d606a3SSatish Balay''' % (self.petscdir.version, installdir, installarch, installpath))
240351d3a41SMatthew G Knepley    fd.close()
241351d3a41SMatthew G Knepley    return
242351d3a41SMatthew G Knepley
243f8833479SBarry Smith  def Dump(self):
244f8833479SBarry Smith    ''' Actually put the values into the configuration files '''
245f8833479SBarry Smith    # eventually everything between -- should be gone
24617f368bcSBarry Smith    if self.mpi.usingMPIUni:
24717f368bcSBarry Smith      #
24817f368bcSBarry Smith      # Remove any MPI/MPICH include files that may have been put here by previous runs of ./configure
2497908f030SMatthew 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)
25017f368bcSBarry Smith
251b5f71184SBarry Smith    self.logPrintDivider()
252b5f71184SBarry Smith    # Test for compiler-specific macros that need to be defined.
253b5f71184SBarry Smith    if self.setCompilers.isCrayVector('CC', self.log):
254b5f71184SBarry Smith      self.addDefine('HAVE_CRAY_VECTOR','1')
255b5f71184SBarry Smith
256b5f71184SBarry Smith    if self.functions.haveFunction('gethostbyname') and self.functions.haveFunction('socket') and self.headers.haveHeader('netinet/in.h'):
257b5f71184SBarry Smith      self.addDefine('USE_SOCKET_VIEWER','1')
258b5f71184SBarry Smith      if self.checkCompile('#include <sys/socket.h>','setsockopt(0,SOL_SOCKET,SO_REUSEADDR,0,0)'):
259b5f71184SBarry Smith        self.addDefine('HAVE_SO_REUSEADDR','1')
260b5f71184SBarry Smith
261b5f71184SBarry Smith    self.logPrintDivider()
2625f27b2e0SBarry Smith    self.setCompilers.pushLanguage('C')
2635f27b2e0SBarry Smith    compiler = self.setCompilers.getCompiler()
264217fe27eSSatish Balay    if [s for s in ['mpicc','mpiicc'] if os.path.basename(compiler).find(s)>=0]:
2655f27b2e0SBarry Smith      try:
2665f27b2e0SBarry Smith        output   = self.executeShellCommand(compiler + ' -show', log = self.log)[0]
2675f27b2e0SBarry Smith        compiler = output.split(' ')[0]
268f424265bSStefano Zampini        self.addDefine('MPICC_SHOW','"'+output.strip().replace('\n','\\\\n').replace('"','')+'"')
2695f27b2e0SBarry Smith      except:
270c9872b61SBarry Smith        self.addDefine('MPICC_SHOW','"Unavailable"')
271c9872b61SBarry Smith    else:
272c9872b61SBarry Smith      self.addDefine('MPICC_SHOW','"Unavailable"')
2735f27b2e0SBarry Smith    self.setCompilers.popLanguage()
274f8833479SBarry Smith#-----------------------------------------------------------------------------------------------------
275f8833479SBarry Smith
276f8833479SBarry Smith    # Sometimes we need C compiler, even if built with C++
277f8833479SBarry Smith    self.setCompilers.pushLanguage('C')
278e4c30378SBarry Smith    # do not use getCompilerFlags() because that automatically includes the CPPFLAGS so one ends up with duplication flags in makefile usage
279e4c30378SBarry Smith    self.addMakeMacro('CC_FLAGS',self.setCompilers.CFLAGS)
280f8833479SBarry Smith    self.setCompilers.popLanguage()
281f8833479SBarry Smith
28234f774f6SJed Brown    # And sometimes we need a C++ compiler even when PETSc is built with C
28334f774f6SJed Brown    if hasattr(self.compilers, 'CXX'):
28434f774f6SJed Brown      self.setCompilers.pushLanguage('Cxx')
28529921a8fSScott Kruger      self.addDefine('HAVE_CXX','1')
2860b119762SSatish Balay      self.addMakeMacro('CXXPP_FLAGS',self.setCompilers.CXXPPFLAGS)
287e4c30378SBarry Smith      # do not use getCompilerFlags() because that automatically includes the CXXPPFLAGS so one ends up with duplication flags in makefile usage
288e4c30378SBarry Smith      self.addMakeMacro('CXX_FLAGS',self.setCompilers.CXXFLAGS+' '+self.setCompilers.CXX_CXXFLAGS)
2892f4326f3SSatish Balay      cxx_linker = self.setCompilers.getLinker()
2902f4326f3SSatish Balay      self.addMakeMacro('CXX_LINKER',cxx_linker)
2912f4326f3SSatish Balay      self.addMakeMacro('CXX_LINKER_FLAGS',self.setCompilers.getLinkerFlags())
29234f774f6SJed Brown      self.setCompilers.popLanguage()
29334f774f6SJed Brown
294f8833479SBarry Smith    # C preprocessor values
2951315f054SBarry Smith    self.addMakeMacro('CPP_FLAGS',self.setCompilers.CPPFLAGS)
296f8833479SBarry Smith
297f8833479SBarry Smith    # compiler values
298f8833479SBarry Smith    self.setCompilers.pushLanguage(self.languages.clanguage)
299f8833479SBarry Smith    self.addMakeMacro('PCC',self.setCompilers.getCompiler())
300e4c30378SBarry Smith    # do not use getCompilerFlags() because that automatically includes the preprocessor flags so one ends up with duplication flags in makefile usage
301e4c30378SBarry Smith    if self.languages.clanguage == 'C':
302e4c30378SBarry Smith      self.addMakeMacro('PCC_FLAGS','$(CC_FLAGS)')
303e4c30378SBarry Smith    else:
304e4c30378SBarry Smith      self.addMakeMacro('PCC_FLAGS','$(CXX_FLAGS)')
305f8833479SBarry Smith    self.setCompilers.popLanguage()
306f8833479SBarry Smith    # .o or .obj
307f8833479SBarry Smith    self.addMakeMacro('CC_SUFFIX','o')
308f8833479SBarry Smith
309f8833479SBarry Smith    # executable linker values
310f8833479SBarry Smith    self.setCompilers.pushLanguage(self.languages.clanguage)
311f8833479SBarry Smith    pcc_linker = self.setCompilers.getLinker()
312f8833479SBarry Smith    self.addMakeMacro('PCC_LINKER',pcc_linker)
313c84a332bSSatish Balay    self.addMakeMacro('PCC_LINKER_FLAGS',self.setCompilers.getLinkerFlags())
314f8833479SBarry Smith    self.setCompilers.popLanguage()
315f8833479SBarry Smith    # '' for Unix, .exe for Windows
316f8833479SBarry Smith    self.addMakeMacro('CC_LINKER_SUFFIX','')
317f8833479SBarry Smith
318f8833479SBarry Smith    if hasattr(self.compilers, 'FC'):
319cb297985SSatish Balay      if self.framework.argDB['with-fortran-bindings']:
320257f4e5aSSatish Balay        if not self.fortran.fortranIsF90:
321257f4e5aSSatish Balay          raise RuntimeError('Error! Fortran compiler "'+self.compilers.FC+'" does not support F90! PETSc fortran bindings require a F90 compiler')
322cb297985SSatish Balay        self.addDefine('HAVE_FORTRAN','1')
323f8833479SBarry Smith      self.setCompilers.pushLanguage('FC')
324f8833479SBarry Smith      # need FPPFLAGS in config/setCompilers
3250b119762SSatish Balay      self.addMakeMacro('FPP_FLAGS',self.setCompilers.FPPFLAGS)
326f8833479SBarry Smith
327f8833479SBarry Smith      # compiler values
328f8833479SBarry Smith      self.addMakeMacro('FC_FLAGS',self.setCompilers.getCompilerFlags())
329f8833479SBarry Smith      self.setCompilers.popLanguage()
330f8833479SBarry Smith      # .o or .obj
331f8833479SBarry Smith      self.addMakeMacro('FC_SUFFIX','o')
332f8833479SBarry Smith
333f8833479SBarry Smith      # executable linker values
334f8833479SBarry Smith      self.setCompilers.pushLanguage('FC')
335a9acdec7SBarry Smith      self.addMakeMacro('FC_LINKER',self.setCompilers.getLinker())
3366d53d35eSSatish Balay      self.addMakeMacro('FC_LINKER_FLAGS',self.setCompilers.getLinkerFlags())
3373feacd00SBarry Smith      # apple requires this shared library linker flag on SOME versions of the os
3383feacd00SBarry Smith      if self.setCompilers.getLinkerFlags().find('-Wl,-commons,use_dylibs') > -1:
3393feacd00SBarry Smith        self.addMakeMacro('DARWIN_COMMONS_USE_DYLIBS',' -Wl,-commons,use_dylibs ')
340bb82cf9cSSatish Balay      self.setCompilers.popLanguage()
3415d631499SMatthew Knepley
3425d631499SMatthew Knepley      # F90 Modules
3435d631499SMatthew Knepley      if self.setCompilers.fortranModuleIncludeFlag:
3445d631499SMatthew Knepley        self.addMakeMacro('FC_MODULE_FLAG', self.setCompilers.fortranModuleIncludeFlag)
3456ddd6694SSatish Balay      else: # for non-f90 compilers like g77
3466ddd6694SSatish Balay        self.addMakeMacro('FC_MODULE_FLAG', '-I')
347a324c51cSMatthew G Knepley      if self.setCompilers.fortranModuleIncludeFlag:
348a324c51cSMatthew G Knepley        self.addMakeMacro('FC_MODULE_OUTPUT_FLAG', self.setCompilers.fortranModuleOutputFlag)
349f8833479SBarry Smith    else:
350f8833479SBarry Smith      self.addMakeMacro('FC','')
351f8833479SBarry Smith
35246a3958fSBarry Smith    if hasattr(self.compilers, 'CUDAC'):
3537ff2890cSSatish Balay      self.setCompilers.pushLanguage('CUDA')
354d93a25ecSSatish Balay      self.addMakeMacro('CUDAC_FLAGS',self.setCompilers.getCompilerFlags())
35550dcbc5aSJunchao Zhang      self.addMakeMacro('CUDAPP_FLAGS',self.setCompilers.CUDAPPFLAGS)
3567ff2890cSSatish Balay      self.setCompilers.popLanguage()
3577ff2890cSSatish Balay
358694a2f0eSJunchao Zhang    if hasattr(self.compilers, 'HIPC'):
35928f796eaSScott Kruger      self.setCompilers.pushLanguage('HIP')
360694a2f0eSJunchao Zhang      self.addMakeMacro('HIPC_FLAGS',self.setCompilers.getCompilerFlags())
3617fb1458fSStefano Zampini      self.addMakeMacro('HIPPP_FLAGS',self.setCompilers.HIPPPFLAGS)
36228f796eaSScott Kruger      self.setCompilers.popLanguage()
36328f796eaSScott Kruger
36450dcbc5aSJunchao Zhang    if hasattr(self.compilers, 'SYCLC'):
36528f796eaSScott Kruger      self.setCompilers.pushLanguage('SYCL')
36650dcbc5aSJunchao Zhang      self.addMakeMacro('SYCLC_FLAGS',self.setCompilers.getCompilerFlags())
36750dcbc5aSJunchao Zhang      self.addMakeMacro('SYCLPP_FLAGS',self.setCompilers.SYCLPPFLAGS)
36828f796eaSScott Kruger      self.setCompilers.popLanguage()
36928f796eaSScott Kruger
370f8833479SBarry Smith    # shared library linker values
371f8833479SBarry Smith    self.setCompilers.pushLanguage(self.languages.clanguage)
372f8833479SBarry Smith    # need to fix BuildSystem to collect these separately
373f8833479SBarry Smith    self.addMakeMacro('SL_LINKER',self.setCompilers.getLinker())
37470db8aa6SSatish Balay    self.addMakeMacro('SL_LINKER_FLAGS','${PCC_LINKER_FLAGS}')
375f8833479SBarry Smith    self.setCompilers.popLanguage()
376f8833479SBarry Smith    # One of 'a', 'so', 'lib', 'dll', 'dylib' (perhaps others also?) depending on the library generator and architecture
377f8833479SBarry Smith    # Note: . is not included in this macro, consistent with AR_LIB_SUFFIX
378f8833479SBarry Smith    if self.setCompilers.sharedLibraryExt == self.setCompilers.AR_LIB_SUFFIX:
379f8833479SBarry Smith      self.addMakeMacro('SL_LINKER_SUFFIX', '')
38046bc77b6SBarry Smith      self.addDefine('SLSUFFIX','""')
381f8833479SBarry Smith    else:
382f8833479SBarry Smith      self.addMakeMacro('SL_LINKER_SUFFIX', self.setCompilers.sharedLibraryExt)
38346bc77b6SBarry Smith      self.addDefine('SLSUFFIX','"'+self.setCompilers.sharedLibraryExt+'"')
384bb82cf9cSSatish Balay
38523e93537SBarry Smith    self.addMakeMacro('SL_LINKER_LIBS','${PETSC_EXTERNAL_LIB_BASIC}')
386bb82cf9cSSatish Balay
387f8833479SBarry Smith#-----------------------------------------------------------------------------------------------------
388f8833479SBarry Smith
389f8833479SBarry Smith    # CONLY or CPP. We should change the PETSc makefiles to do this better
390f8833479SBarry Smith    if self.languages.clanguage == 'C': lang = 'CONLY'
391f8833479SBarry Smith    else: lang = 'CXXONLY'
392f8833479SBarry Smith    self.addMakeMacro('PETSC_LANGUAGE',lang)
393f8833479SBarry Smith
394f8833479SBarry Smith    # real or complex
395f8833479SBarry Smith    self.addMakeMacro('PETSC_SCALAR',self.scalartypes.scalartype)
396f8833479SBarry Smith    # double or float
397f8833479SBarry Smith    self.addMakeMacro('PETSC_PRECISION',self.scalartypes.precision)
398f8833479SBarry Smith
399f8833479SBarry Smith    if self.framework.argDB['with-batch']:
400f8833479SBarry Smith      self.addMakeMacro('PETSC_WITH_BATCH','1')
401f8833479SBarry Smith
402f8833479SBarry Smith#-----------------------------------------------------------------------------------------------------
403a6cc6bb1SBarry Smith    # print include and lib for makefiles
404b5f71184SBarry Smith    self.logPrintDivider()
405f8833479SBarry Smith    self.framework.packages.reverse()
4065a21677cSJed Brown    petscincludes = [os.path.join(self.petscdir.dir,'include'),os.path.join(self.petscdir.dir,self.arch.arch,'include')]
4078749e224SJunchao Zhang    petscincludes_install = [os.path.join(self.installdir.dir, 'include')] if self.framework.argDB['prefix'] else petscincludes
4085a21677cSJed Brown    includes = []
409de8f682fSSatish Balay    self.packagelibs = []
410f8833479SBarry Smith    for i in self.framework.packages:
4117f0ff1afSBarry Smith      if not i.required:
4123972cb20SJacob Faibussowitsch        if i.devicePackage:
4133972cb20SJacob Faibussowitsch          self.addDefine('HAVE_DEVICE',1)
414eeb16384SBarry Smith        self.addDefine('HAVE_'+i.PACKAGE.replace('-','_'), 1)  # ONLY list package if it is used directly by PETSc (and not only by another package)
415f8833479SBarry Smith      if not isinstance(i.lib, list):
416f8833479SBarry Smith        i.lib = [i.lib]
417de8f682fSSatish Balay      if i.linkedbypetsc: self.packagelibs.extend(i.lib)
418eeb16384SBarry Smith      self.addMakeMacro(i.PACKAGE.replace('-','_')+'_LIB', self.libraries.toStringNoDupes(i.lib))
419f8833479SBarry Smith      if hasattr(i,'include'):
420f8833479SBarry Smith        if not isinstance(i.include,list):
421f8833479SBarry Smith          i.include = [i.include]
422ac9e4c42SSatish Balay        includes.extend(i.include)
423eeb16384SBarry Smith        self.addMakeMacro(i.PACKAGE.replace('-','_')+'_INCLUDE',self.headers.toStringNoDupes(i.include))
4242df986feSBarry Smith    if self.framework.argDB['with-single-library']:
425e282ce78SJed Brown      self.petsclib = '-lpetsc'
42691bb3077SSatish Balay    else:
427e282ce78SJed Brown      self.petsclib = '-lpetscts -lpetscsnes -lpetscksp -lpetscdm -lpetscmat -lpetscvec -lpetscsys'
428de8f682fSSatish Balay    self.complibs = self.compilers.flibs+self.compilers.cxxlibs+self.compilers.LIBS.split()
4295a21677cSJed Brown    self.PETSC_WITH_EXTERNAL_LIB = self.libraries.toStringNoDupes(['-L${PETSC_DIR}/${PETSC_ARCH}/lib', self.petsclib]+self.packagelibs+self.complibs)
430de8f682fSSatish Balay    self.PETSC_EXTERNAL_LIB_BASIC = self.libraries.toStringNoDupes(self.packagelibs+self.complibs)
431de8f682fSSatish Balay
432de8f682fSSatish Balay    self.addMakeMacro('PETSC_EXTERNAL_LIB_BASIC',self.PETSC_EXTERNAL_LIB_BASIC)
4335a21677cSJed Brown    allincludes = petscincludes + includes
4345a21677cSJed Brown    allincludes_install = petscincludes_install + includes
4355a21677cSJed Brown    self.PETSC_CC_INCLUDES = self.headers.toStringNoDupes(allincludes)
4365a21677cSJed Brown    self.PETSC_CC_INCLUDES_INSTALL = self.headers.toStringNoDupes(allincludes_install)
4375a21677cSJed Brown    self.addMakeMacro('PETSC_CC_INCLUDES',self.PETSC_CC_INCLUDES)
4385a21677cSJed Brown    self.addMakeMacro('PETSC_CC_INCLUDES_INSTALL', self.PETSC_CC_INCLUDES_INSTALL)
439cbd5cc15SBarry Smith    if hasattr(self.compilers, 'FC'):
4405a21677cSJed Brown      def modinc(includes):
4410542e31aSBarry Smith        return includes if self.fortran.fortranIsF90 else []
4425a21677cSJed Brown      self.addMakeMacro('PETSC_FC_INCLUDES',self.headers.toStringNoDupes(allincludes,modinc(allincludes)))
4435a21677cSJed Brown      self.addMakeMacro('PETSC_FC_INCLUDES_INSTALL',self.headers.toStringNoDupes(allincludes_install,modinc(allincludes_install)))
444f8833479SBarry Smith
4455bb5b263SMatthew G. Knepley    self.addDefine('LIB_DIR','"'+os.path.join(self.installdir.dir,'lib')+'"')
446f8833479SBarry Smith
4470f3b21c2SBarry Smith    if self.framework.argDB['with-single-library']:
4480f3b21c2SBarry Smith      # overrides the values set in conf/variables
4490f3b21c2SBarry Smith      self.addMakeMacro('LIBNAME','${INSTALL_LIB_DIR}/libpetsc.${AR_LIB_SUFFIX}')
45057cb31baSSatish Balay      self.addMakeMacro('SHLIBS','libpetsc')
451bccf1c12SBarry Smith      self.addMakeMacro('PETSC_LIB_BASIC','-lpetsc')
452797063a9SSatish Balay      self.addMakeMacro('PETSC_KSP_LIB_BASIC','-lpetsc')
453797063a9SSatish Balay      self.addMakeMacro('PETSC_TS_LIB_BASIC','-lpetsc')
454b0a7d7e7SSatish Balay      self.addMakeMacro('PETSC_TAO_LIB_BASIC','-lpetsc')
455de8f682fSSatish Balay      self.addMakeMacro('PETSC_WITH_EXTERNAL_LIB',self.PETSC_WITH_EXTERNAL_LIB)
456bb84e0fdSBarry Smith      self.addDefine('USE_SINGLE_LIBRARY', '1')
4572df986feSBarry Smith      if self.sharedlibraries.useShared:
458ea820d49SSatish Balay        self.addMakeMacro('PETSC_SYS_LIB','${C_SH_LIB_PATH} ${PETSC_WITH_EXTERNAL_LIB}')
459ea820d49SSatish Balay        self.addMakeMacro('PETSC_VEC_LIB','${C_SH_LIB_PATH} ${PETSC_WITH_EXTERNAL_LIB}')
460ea820d49SSatish Balay        self.addMakeMacro('PETSC_MAT_LIB','${C_SH_LIB_PATH} ${PETSC_WITH_EXTERNAL_LIB}')
461ea820d49SSatish Balay        self.addMakeMacro('PETSC_DM_LIB','${C_SH_LIB_PATH} ${PETSC_WITH_EXTERNAL_LIB}')
462ea820d49SSatish Balay        self.addMakeMacro('PETSC_KSP_LIB','${C_SH_LIB_PATH} ${PETSC_WITH_EXTERNAL_LIB}')
463ea820d49SSatish Balay        self.addMakeMacro('PETSC_SNES_LIB','${C_SH_LIB_PATH} ${PETSC_WITH_EXTERNAL_LIB}')
464ea820d49SSatish Balay        self.addMakeMacro('PETSC_TS_LIB','${C_SH_LIB_PATH} ${PETSC_WITH_EXTERNAL_LIB}')
465b0a7d7e7SSatish Balay        self.addMakeMacro('PETSC_TAO_LIB','${C_SH_LIB_PATH} ${PETSC_WITH_EXTERNAL_LIB}')
466fdb87e33SJed Brown        self.addMakeMacro('PETSC_CHARACTERISTIC_LIB','${C_SH_LIB_PATH} ${PETSC_WITH_EXTERNAL_LIB}')
467ea820d49SSatish Balay        self.addMakeMacro('PETSC_LIB','${C_SH_LIB_PATH} ${PETSC_WITH_EXTERNAL_LIB}')
468ea820d49SSatish Balay        self.addMakeMacro('PETSC_CONTRIB','${C_SH_LIB_PATH} ${PETSC_WITH_EXTERNAL_LIB}')
4692df986feSBarry Smith      else:
470ea820d49SSatish Balay        self.addMakeMacro('PETSC_SYS_LIB','${PETSC_WITH_EXTERNAL_LIB}')
471ea820d49SSatish Balay        self.addMakeMacro('PETSC_VEC_LIB','${PETSC_WITH_EXTERNAL_LIB}')
472ea820d49SSatish Balay        self.addMakeMacro('PETSC_MAT_LIB','${PETSC_WITH_EXTERNAL_LIB}')
473ea820d49SSatish Balay        self.addMakeMacro('PETSC_DM_LIB','${PETSC_WITH_EXTERNAL_LIB}')
474ea820d49SSatish Balay        self.addMakeMacro('PETSC_KSP_LIB','${PETSC_WITH_EXTERNAL_LIB}')
475ea820d49SSatish Balay        self.addMakeMacro('PETSC_SNES_LIB','${PETSC_WITH_EXTERNAL_LIB}')
476ea820d49SSatish Balay        self.addMakeMacro('PETSC_TS_LIB','${PETSC_WITH_EXTERNAL_LIB}')
477b0a7d7e7SSatish Balay        self.addMakeMacro('PETSC_TAO_LIB','${PETSC_WITH_EXTERNAL_LIB}')
478fdb87e33SJed Brown        self.addMakeMacro('PETSC_CHARACTERISTIC_LIB','${PETSC_WITH_EXTERNAL_LIB}')
479ea820d49SSatish Balay        self.addMakeMacro('PETSC_LIB','${PETSC_WITH_EXTERNAL_LIB}')
480ea820d49SSatish Balay        self.addMakeMacro('PETSC_CONTRIB','${PETSC_WITH_EXTERNAL_LIB}')
4810f3b21c2SBarry Smith
482f8833479SBarry Smith    if not os.path.exists(os.path.join(self.petscdir.dir,self.arch.arch,'lib')):
483f8833479SBarry Smith      os.makedirs(os.path.join(self.petscdir.dir,self.arch.arch,'lib'))
484f8833479SBarry Smith
48513f9d092SSatish Balay# add a makefile endtry for display
48613f9d092SSatish Balay    if self.framework.argDB['with-display']:
48713f9d092SSatish Balay      self.addMakeMacro('DISPLAY',self.framework.argDB['with-display'])
48813f9d092SSatish Balay
489f8833479SBarry Smith    # add a makefile entry for configure options
490f8833479SBarry Smith    self.addMakeMacro('CONFIGURE_OPTIONS', self.framework.getOptionsString(['configModules', 'optionsModule']).replace('\"','\\"'))
491*a0c7f9aaSSamuel Khuvis
492*a0c7f9aaSSamuel Khuvis    if self.framework.argDB['with-tau-perfstubs']:
493*a0c7f9aaSSamuel Khuvis      self.addDefine('HAVE_TAU_PERFSTUBS',1)
494f8833479SBarry Smith    return
495f8833479SBarry Smith
496f8833479SBarry Smith  def dumpConfigInfo(self):
497f8833479SBarry Smith    import time
498c6ef1b5bSJed Brown    fd = open(os.path.join(self.arch.arch,'include','petscconfiginfo.h'),'w')
499dc25a686SPierre Jolivet    fd.write('static const char *petscconfigureoptions = "'+self.framework.getOptionsString(['configModules', 'optionsModule']).replace('\"','\\"').replace('\\ ','\\\\ ')+'";\n')
500f8833479SBarry Smith    fd.close()
501f8833479SBarry Smith    return
502f8833479SBarry Smith
5032a4161d9SMatthew G Knepley  def dumpMachineInfo(self):
5042a4161d9SMatthew G Knepley    import platform
505a970bd74SBernhard M. Wiedemann    import datetime
5062a4161d9SMatthew G Knepley    import time
50740373944SSatish Balay    import script
508ca77dbeeSGeoffrey Irving    def escape(s):
509e08ecd42SSatish Balay      return s.replace('"',r'\"').replace(r'\ ',r'\\ ') # novermin
510c6ef1b5bSJed Brown    fd = open(os.path.join(self.arch.arch,'include','petscmachineinfo.h'),'w')
5112a4161d9SMatthew G Knepley    fd.write('static const char *petscmachineinfo = \"\\n\"\n')
5122a4161d9SMatthew G Knepley    fd.write('\"-----------------------------------------\\n\"\n')
513a970bd74SBernhard M. Wiedemann    buildhost = platform.node()
514a970bd74SBernhard M. Wiedemann    if os.environ.get('SOURCE_DATE_EPOCH'):
515a970bd74SBernhard M. Wiedemann      buildhost = "reproducible"
516a970bd74SBernhard M. Wiedemann    buildtime = datetime.datetime.utcfromtimestamp(int(os.environ.get('SOURCE_DATE_EPOCH', time.time())))
517a970bd74SBernhard M. Wiedemann    fd.write('\"Libraries compiled on %s on %s \\n\"\n' % (buildtime, buildhost))
51860acdfe7SSatish Balay    fd.write('\"Machine characteristics: %s\\n\"\n' % (platform.platform()))
5195188cb68SSatish Balay    fd.write('\"Using PETSc directory: %s\\n\"\n' % (escape(self.installdir.petscDir)))
5205188cb68SSatish Balay    fd.write('\"Using PETSc arch: %s\\n\"\n' % (escape(self.installdir.petscArch)))
521cdec380aSBarry Smith    fd.write('\"-----------------------------------------\\n\";\n')
5222a4161d9SMatthew G Knepley    fd.write('static const char *petsccompilerinfo = \"\\n\"\n')
5232a4161d9SMatthew G Knepley    self.setCompilers.pushLanguage(self.languages.clanguage)
5245f27b2e0SBarry Smith    fd.write('\"Using C compiler: %s %s \\n\"\n' % (escape(self.setCompilers.getCompiler()), escape(self.setCompilers.getCompilerFlags())))
5252a4161d9SMatthew G Knepley    self.setCompilers.popLanguage()
5268782282cSMatthew G Knepley    if hasattr(self.compilers, 'FC'):
5272a4161d9SMatthew G Knepley      self.setCompilers.pushLanguage('FC')
5285f27b2e0SBarry Smith      fd.write('\"Using Fortran compiler: %s %s  %s\\n\"\n' % (escape(self.setCompilers.getCompiler()), escape(self.setCompilers.getCompilerFlags()), escape(self.setCompilers.CPPFLAGS)))
5292a4161d9SMatthew G Knepley      self.setCompilers.popLanguage()
530cdec380aSBarry Smith    fd.write('\"-----------------------------------------\\n\";\n')
5312a4161d9SMatthew G Knepley    fd.write('static const char *petsccompilerflagsinfo = \"\\n\"\n')
5325a21677cSJed Brown    fd.write('\"Using include paths: %s\\n\"\n' % (escape(self.PETSC_CC_INCLUDES_INSTALL.replace('${PETSC_DIR}', self.installdir.petscDir))))
533cdec380aSBarry Smith    fd.write('\"-----------------------------------------\\n\";\n')
5342a4161d9SMatthew G Knepley    fd.write('static const char *petsclinkerinfo = \"\\n\"\n')
5352a4161d9SMatthew G Knepley    self.setCompilers.pushLanguage(self.languages.clanguage)
536ca77dbeeSGeoffrey Irving    fd.write('\"Using C linker: %s\\n\"\n' % (escape(self.setCompilers.getLinker())))
5372a4161d9SMatthew G Knepley    self.setCompilers.popLanguage()
5388782282cSMatthew G Knepley    if hasattr(self.compilers, 'FC'):
5392a4161d9SMatthew G Knepley      self.setCompilers.pushLanguage('FC')
540ca77dbeeSGeoffrey Irving      fd.write('\"Using Fortran linker: %s\\n\"\n' % (escape(self.setCompilers.getLinker())))
5412a4161d9SMatthew G Knepley      self.setCompilers.popLanguage()
5425188cb68SSatish 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)))
543cdec380aSBarry Smith    fd.write('\"-----------------------------------------\\n\";\n')
5442a4161d9SMatthew G Knepley    fd.close()
5452a4161d9SMatthew G Knepley    return
546b2843cf1SBarry Smith
547b2843cf1SBarry Smith  def configurePrefetch(self):
548b2843cf1SBarry Smith    '''Sees if there are any prefetch functions supported'''
5497fca349cSMatthew G. Knepley    if config.setCompilers.Configure.isSolaris(self.log) or self.framework.argDB['with-ios'] or not self.framework.argDB['with-prefetch']:
55093f78423SSatish Balay      self.addDefine('Prefetch(a,b,c)', ' ')
55193f78423SSatish Balay      return
552ec284106SBarry Smith    self.pushLanguage(self.languages.clanguage)
55310699583SJed Brown    if self.checkLink('#include <xmmintrin.h>', 'void *v = 0;_mm_prefetch((const char*)v,_MM_HINT_NTA);\n'):
55450d8bf02SJed Brown      # The Intel Intrinsics manual [1] specifies the prototype
55550d8bf02SJed Brown      #
55650d8bf02SJed Brown      #   void _mm_prefetch(char const *a, int sel);
55750d8bf02SJed Brown      #
55850d8bf02SJed Brown      # but other vendors seem to insist on using subtly different
55950d8bf02SJed Brown      # prototypes, including void* for the pointer, and an enum for
56050d8bf02SJed Brown      # sel.  These are both reasonable changes, but negatively impact
56150d8bf02SJed Brown      # portability.
56250d8bf02SJed Brown      #
563a8d69d7bSBarry Smith      # [1] https://software.intel.com/file/6373
56450d8bf02SJed Brown      self.addDefine('HAVE_XMMINTRIN_H', 1)
56550d8bf02SJed Brown      self.addDefine('Prefetch(a,b,c)', '_mm_prefetch((const char*)(a),(c))')
56650d8bf02SJed Brown      self.addDefine('PREFETCH_HINT_NTA', '_MM_HINT_NTA')
56750d8bf02SJed Brown      self.addDefine('PREFETCH_HINT_T0',  '_MM_HINT_T0')
56850d8bf02SJed Brown      self.addDefine('PREFETCH_HINT_T1',  '_MM_HINT_T1')
56950d8bf02SJed Brown      self.addDefine('PREFETCH_HINT_T2',  '_MM_HINT_T2')
57050d8bf02SJed Brown    elif self.checkLink('#include <xmmintrin.h>', 'void *v = 0;_mm_prefetch(v,_MM_HINT_NTA);\n'):
57150d8bf02SJed Brown      self.addDefine('HAVE_XMMINTRIN_H', 1)
57250d8bf02SJed Brown      self.addDefine('Prefetch(a,b,c)', '_mm_prefetch((const void*)(a),(c))')
57350d8bf02SJed Brown      self.addDefine('PREFETCH_HINT_NTA', '_MM_HINT_NTA')
57450d8bf02SJed Brown      self.addDefine('PREFETCH_HINT_T0',  '_MM_HINT_T0')
57550d8bf02SJed Brown      self.addDefine('PREFETCH_HINT_T1',  '_MM_HINT_T1')
57650d8bf02SJed Brown      self.addDefine('PREFETCH_HINT_T2',  '_MM_HINT_T2')
57710699583SJed Brown    elif self.checkLink('', 'void *v = 0;__builtin_prefetch(v,0,0);\n'):
57810699583SJed Brown      # From GCC docs: void __builtin_prefetch(const void *addr,int rw,int locality)
57910699583SJed Brown      #
58010699583SJed Brown      #   The value of rw is a compile-time constant one or zero; one
58110699583SJed Brown      #   means that the prefetch is preparing for a write to the memory
58210699583SJed Brown      #   address and zero, the default, means that the prefetch is
58310699583SJed Brown      #   preparing for a read. The value locality must be a compile-time
58410699583SJed Brown      #   constant integer between zero and three. A value of zero means
58510699583SJed Brown      #   that the data has no temporal locality, so it need not be left
58610699583SJed Brown      #   in the cache after the access. A value of three means that the
58710699583SJed Brown      #   data has a high degree of temporal locality and should be left
58810699583SJed Brown      #   in all levels of cache possible. Values of one and two mean,
58910699583SJed Brown      #   respectively, a low or moderate degree of temporal locality.
59010699583SJed Brown      #
59110699583SJed Brown      # Here we adopt Intel's x86/x86-64 naming scheme for the locality
59210699583SJed Brown      # hints.  Using macros for these values in necessary since some
59310699583SJed Brown      # compilers require an enum.
59410699583SJed Brown      self.addDefine('Prefetch(a,b,c)', '__builtin_prefetch((a),(b),(c))')
59510699583SJed Brown      self.addDefine('PREFETCH_HINT_NTA', '0')
59610699583SJed Brown      self.addDefine('PREFETCH_HINT_T0',  '3')
59710699583SJed Brown      self.addDefine('PREFETCH_HINT_T1',  '2')
59810699583SJed Brown      self.addDefine('PREFETCH_HINT_T2',  '1')
599b2843cf1SBarry Smith    else:
600b2843cf1SBarry Smith      self.addDefine('Prefetch(a,b,c)', ' ')
6017d490b44SBarry Smith    self.popLanguage()
602b2843cf1SBarry Smith
60349fe22e6SSatish Balay  def delGenFiles(self):
60449fe22e6SSatish Balay    '''Delete generated files'''
60549fe22e6SSatish Balay    delfile = os.path.join(self.arch.arch,'lib','petsc','conf','files')
60649fe22e6SSatish Balay    try:
60749fe22e6SSatish Balay      os.unlink(delfile)
60849fe22e6SSatish Balay    except: pass
60949fe22e6SSatish Balay
61009bc878fSSatish Balay  def configureAtoll(self):
61109bc878fSSatish Balay    '''Checks if atoll exists'''
61222164b4cSPierre 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'):
61309bc878fSSatish Balay       self.addDefine('HAVE_ATOLL', '1')
61409bc878fSSatish Balay
6152400fdedSBarry Smith  def configureUnused(self):
6162400fdedSBarry Smith    '''Sees if __attribute((unused)) is supported'''
6171adaff47SSean Farley    if self.framework.argDB['with-ios']:
6182400fdedSBarry Smith      self.addDefine('UNUSED', ' ')
6192400fdedSBarry Smith      return
6202400fdedSBarry Smith    self.pushLanguage(self.languages.clanguage)
62122164b4cSPierre 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'):
6222400fdedSBarry Smith      self.addDefine('UNUSED', '__attribute((unused))')
6232400fdedSBarry Smith    else:
6242400fdedSBarry Smith      self.addDefine('UNUSED', ' ')
6252400fdedSBarry Smith    self.popLanguage()
6262400fdedSBarry Smith
62798ed35c3SBarry Smith  def configureIsatty(self):
62898ed35c3SBarry Smith    '''Check if the Unix C function isatty() works correctly
62998ed35c3SBarry Smith       Actually just assumes it does not work correctly on batch systems'''
63098ed35c3SBarry Smith    if not self.framework.argDB['with-batch']:
63198ed35c3SBarry Smith      self.addDefine('USE_ISATTY',1)
63298ed35c3SBarry Smith
6331ef8df7fSJed Brown  def configureDeprecated(self):
6341ef8df7fSJed Brown    '''Check if __attribute((deprecated)) is supported'''
6357efe37a1SJacob Faibussowitsch    def checkDeprecated(macro_base, src, is_intel):
6367efe37a1SJacob Faibussowitsch      '''
6377efe37a1SJacob Faibussowitsch      run through the various attribute deprecated combinations and define MACRO_BAS(why) to the result
6387efe37a1SJacob Faibussowitsch      it if it compiles.
6397efe37a1SJacob Faibussowitsch
6407efe37a1SJacob Faibussowitsch      If none of the combos work, defines MACRO_BASE(why) as empty
6417efe37a1SJacob Faibussowitsch      '''
6427efe37a1SJacob Faibussowitsch      full_macro_name = macro_base + '(why)'
6437efe37a1SJacob Faibussowitsch      for prefix in ('__attribute__', '__attribute','__declspec'):
6447efe37a1SJacob Faibussowitsch        if prefix == '__declspec':
6457efe37a1SJacob Faibussowitsch          # declspec does not have an extra set of brackets around the arguments
6467efe37a1SJacob Faibussowitsch          attr_bodies = ('deprecated(why)', 'deprecated')
6471ef8df7fSJed Brown        else:
6487efe37a1SJacob Faibussowitsch          attr_bodies = ('(deprecated(why))', '(deprecated)')
6497efe37a1SJacob Faibussowitsch
6507efe37a1SJacob Faibussowitsch        for attr_body in attr_bodies:
6517efe37a1SJacob Faibussowitsch          attr_def = '{}({})'.format(prefix, attr_body)
6527efe37a1SJacob Faibussowitsch          test_src = '\n'.join((
6537efe37a1SJacob Faibussowitsch            '#define {} {}'.format(full_macro_name, attr_def),
6547efe37a1SJacob Faibussowitsch            src.format(macro_base + '("asdasdadsasd")')
6557efe37a1SJacob Faibussowitsch          ))
6567efe37a1SJacob Faibussowitsch          if self.checkCompile(test_src, ''):
6577efe37a1SJacob Faibussowitsch            self.logPrint('configureDeprecated: \'{}\' appears to work'.format(attr_def))
6587efe37a1SJacob Faibussowitsch            if is_intel and '(why)' in attr_body:
6597efe37a1SJacob 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))
6607efe37a1SJacob Faibussowitsch              self.logPrint('*** WE WILL THEREFORE REJECT \'{}\' AND CONTINUE TESTING ***'.format(attr_def))
6617efe37a1SJacob Faibussowitsch              continue
6627efe37a1SJacob Faibussowitsch            self.addDefine(full_macro_name, attr_def)
6637efe37a1SJacob Faibussowitsch            return
6647efe37a1SJacob Faibussowitsch
6657efe37a1SJacob Faibussowitsch      self.addDefine(full_macro_name, ' ')
6667efe37a1SJacob Faibussowitsch      return
6677efe37a1SJacob Faibussowitsch
6687efe37a1SJacob Faibussowitsch    lang = self.languages.clanguage
6697efe37a1SJacob Faibussowitsch    with self.Language(lang):
6707efe37a1SJacob Faibussowitsch      is_intel = self.setCompilers.isIntel(self.getCompiler(lang=lang), self.log)
6717efe37a1SJacob Faibussowitsch      checkDeprecated('DEPRECATED_FUNCTION', '{} int myfunc(void) {{ return 1; }}', is_intel)
6727efe37a1SJacob Faibussowitsch      checkDeprecated('DEPRECATED_TYPEDEF', 'typedef int my_int {};', is_intel)
6737efe37a1SJacob Faibussowitsch      checkDeprecated('DEPRECATED_ENUM', 'enum E {{ oldval {}, newval }};', is_intel)
6747efe37a1SJacob Faibussowitsch      # I was unable to make a CPP macro that takes the old and new values as separate
6757efe37a1SJacob Faibussowitsch      # arguments and builds the message needed by _Pragma hence the deprecation message is
6767efe37a1SJacob Faibussowitsch      # handled as it is
67705de396fSBarry Smith      if self.checkCompile('#define TEST _Pragma("GCC warning \"Testing _Pragma\"") value'):
67805de396fSBarry Smith        self.addDefine('DEPRECATED_MACRO(why)', '_Pragma(why)')
67905de396fSBarry Smith      else:
68005de396fSBarry Smith        self.addDefine('DEPRECATED_MACRO(why)', ' ')
6811ef8df7fSJed Brown
68218f41590SBarry Smith  def configureAlign(self):
6837b7fc14bSLisandro Dalcin    '''Check if __attribute(aligned) is supported'''
6847b7fc14bSLisandro Dalcin    code = '''\
685752d89a4SSatish Balaystruct mystruct {int myint;} __attribute((aligned(16)));
6867b7fc14bSLisandro Dalcinchar assert_aligned[(sizeof(struct mystruct)==16)*2-1];
687752d89a4SSatish Balay'''
688752d89a4SSatish Balay    self.pushLanguage(self.languages.clanguage)
6897b7fc14bSLisandro Dalcin    if self.checkCompile(code):
690752d89a4SSatish Balay      self.addDefine('ATTRIBUTEALIGNED(size)', '__attribute((aligned(size)))')
691752d89a4SSatish Balay      self.addDefine('HAVE_ATTRIBUTEALIGNED', 1)
692752d89a4SSatish Balay    else:
6937b7fc14bSLisandro Dalcin      self.framework.logPrint('Incorrect attribute(aligned)')
694752d89a4SSatish Balay      self.addDefine('ATTRIBUTEALIGNED(size)', ' ')
6957b7fc14bSLisandro Dalcin    self.popLanguage()
696752d89a4SSatish Balay    return
69718f41590SBarry Smith
6989800092aSJed Brown  def configureExpect(self):
6999800092aSJed Brown    '''Sees if the __builtin_expect directive is supported'''
7009800092aSJed Brown    self.pushLanguage(self.languages.clanguage)
7019800092aSJed Brown    if self.checkLink('', 'if (__builtin_expect(0,1)) return 1;'):
7029800092aSJed Brown      self.addDefine('HAVE_BUILTIN_EXPECT', 1)
7039800092aSJed Brown    self.popLanguage()
7049800092aSJed Brown
70553c77d0aSJed Brown  def configureFunctionName(self):
706fbfcfee5SBarry Smith    '''Sees if the compiler supports __func__ or a variant.'''
7071ec50b02SJed Brown    def getFunctionName(lang):
708fbfcfee5SBarry Smith      name = '"unknown"'
7091ec50b02SJed Brown      self.pushLanguage(lang)
710b6ff4c76SKarl Rupp      for fname in ['__func__','__FUNCTION__','__extension__ __func__']:
7110117e5a1SSatish Balay        code = "if ("+fname+"[0] != 'm') return 1;"
7120117e5a1SSatish Balay        if self.checkCompile('',code) and self.checkLink('',code):
7130117e5a1SSatish Balay          name = fname
7140117e5a1SSatish Balay          break
7151ec50b02SJed Brown      self.popLanguage()
7161ec50b02SJed Brown      return name
7171ec50b02SJed Brown    langs = []
718628773c9SSatish Balay
719628773c9SSatish Balay    self.addDefine('FUNCTION_NAME_C', getFunctionName('C'))
7205f6e5f85SSatish Balay    if hasattr(self.compilers, 'CXX'):
721628773c9SSatish Balay      self.addDefine('FUNCTION_NAME_CXX', getFunctionName('Cxx'))
72253c77d0aSJed Brown
723753ebd1dSJed Brown  def configureIntptrt(self):
724753ebd1dSJed Brown    '''Determine what to use for uintptr_t'''
725753ebd1dSJed Brown    def staticAssertSizeMatchesVoidStar(inc,typename):
726753ebd1dSJed Brown      # The declaration is an error if either array size is negative.
727753ebd1dSJed 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
728d26187a0SJed Brown      return self.checkCompile(inc, ('#define STATIC_ASSERT(cond) char negative_length_if_false[2*(!!(cond))-1]\n'
729979939cdSSatish Balay                                     + 'STATIC_ASSERT(sizeof(void*) == sizeof(%s));'%typename))
730753ebd1dSJed Brown    self.pushLanguage(self.languages.clanguage)
73122164b4cSPierre Jolivet    if self.checkCompile('#include <stdint.h>', 'int x; uintptr_t i = (uintptr_t)&x; (void)i'):
732753ebd1dSJed Brown      self.addDefine('UINTPTR_T', 'uintptr_t')
733753ebd1dSJed Brown    elif staticAssertSizeMatchesVoidStar('','unsigned long long'):
734753ebd1dSJed Brown      self.addDefine('UINTPTR_T', 'unsigned long long')
735753ebd1dSJed Brown    elif staticAssertSizeMatchesVoidStar('#include <stdlib.h>','size_t') or staticAssertSizeMatchesVoidStar('#include <string.h>', 'size_t'):
736753ebd1dSJed Brown      self.addDefine('UINTPTR_T', 'size_t')
737c82284b1SJed Brown    elif staticAssertSizeMatchesVoidStar('','unsigned long'):
738c82284b1SJed Brown      self.addDefine('UINTPTR_T', 'unsigned long')
7392d1b7972SSatish Balay    elif staticAssertSizeMatchesVoidStar('','unsigned'):
740753ebd1dSJed Brown      self.addDefine('UINTPTR_T', 'unsigned')
741d26187a0SJed Brown    else:
742d26187a0SJed Brown      raise RuntimeError('Could not find any unsigned integer type matching void*')
743753ebd1dSJed Brown    self.popLanguage()
744753ebd1dSJed Brown
745ed938b00SJed Brown  def configureRTLDDefault(self):
7467b65ca21SBarry Smith    '''Check for dynamic library feature'''
747bfef2c86SBarry Smith    if self.checkCompile('#include <dlfcn.h>\n void *ptr =  RTLD_DEFAULT;'):
748bfef2c86SBarry Smith      self.addDefine('RTLD_DEFAULT','1')
749f8833479SBarry Smith    return
750f8833479SBarry Smith
751f8833479SBarry Smith  def configureSolaris(self):
752f8833479SBarry Smith    '''Solaris specific stuff'''
753f8833479SBarry Smith    if os.path.isdir(os.path.join('/usr','ucblib')):
754f8833479SBarry Smith      try:
755f8833479SBarry Smith        flag = getattr(self.setCompilers, self.language[-1]+'SharedLinkerFlag')
756f8833479SBarry Smith      except AttributeError:
757f8833479SBarry Smith        flag = None
758f8833479SBarry Smith      if flag is None:
759f8833479SBarry Smith        self.compilers.LIBS += ' -L/usr/ucblib'
760f8833479SBarry Smith      else:
761f8833479SBarry Smith        self.compilers.LIBS += ' '+flag+'/usr/ucblib'
762f8833479SBarry Smith    return
763f8833479SBarry Smith
7640f64ec89SBarry Smith  def configureDarwin(self):
7650f64ec89SBarry Smith    '''Log brew configuration for Apple systems'''
7660f64ec89SBarry Smith    try:
7670f64ec89SBarry Smith      self.executeShellCommand(['brew', 'config'], log = self.log)
7680f64ec89SBarry Smith      self.executeShellCommand(['brew', 'info', 'gcc'], log = self.log)
7690f64ec89SBarry Smith    except:
7700f64ec89SBarry Smith      pass
7710f64ec89SBarry Smith    return
7720f64ec89SBarry Smith
773f8833479SBarry Smith  def configureLinux(self):
774f8833479SBarry Smith    '''Linux specific stuff'''
7759f15855cSMatthew G Knepley    # TODO: Test for this by mallocing an odd number of floats and checking the address
776f8833479SBarry Smith    self.addDefine('HAVE_DOUBLE_ALIGN_MALLOC', 1)
777f8833479SBarry Smith    return
778f8833479SBarry Smith
779f8833479SBarry Smith  def configureWin32(self):
780f8833479SBarry Smith    '''Win32 non-cygwin specific stuff'''
781f8833479SBarry Smith    kernel32=0
7824e8afd12SMosè Giordano    if self.libraries.add('Kernel32.lib','GetComputerName',prototype='#include <windows.h>', call='GetComputerName(NULL,NULL);'):
783f8833479SBarry Smith      self.addDefine('HAVE_WINDOWS_H',1)
784f8833479SBarry Smith      self.addDefine('HAVE_GETCOMPUTERNAME',1)
785f8833479SBarry Smith      kernel32=1
7864e8afd12SMosè Giordano    elif self.libraries.add('kernel32','GetComputerName',prototype='#include <windows.h>', call='GetComputerName(NULL,NULL);'):
787f8833479SBarry Smith      self.addDefine('HAVE_WINDOWS_H',1)
788f8833479SBarry Smith      self.addDefine('HAVE_GETCOMPUTERNAME',1)
789f8833479SBarry Smith      kernel32=1
790f8833479SBarry Smith    if kernel32:
791eed94e11SSatish Balay      if self.framework.argDB['with-windows-graphics']:
792eed94e11SSatish Balay        self.addDefine('USE_WINDOWS_GRAPHICS',1)
7934e8afd12SMosè Giordano      if self.checkLink('#include <windows.h>','LoadLibrary(0)'):
794f8833479SBarry Smith        self.addDefine('HAVE_LOADLIBRARY',1)
7954e8afd12SMosè Giordano      if self.checkLink('#include <windows.h>','GetProcAddress(0,0)'):
796b50f6d9eSLisandro Dalcin        self.addDefine('HAVE_GETPROCADDRESS',1)
7974e8afd12SMosè Giordano      if self.checkLink('#include <windows.h>','FreeLibrary(0)'):
798b50f6d9eSLisandro Dalcin        self.addDefine('HAVE_FREELIBRARY',1)
7994e8afd12SMosè Giordano      if self.checkLink('#include <windows.h>','GetLastError()'):
800a21658a3SLisandro Dalcin        self.addDefine('HAVE_GETLASTERROR',1)
8014e8afd12SMosè Giordano      if self.checkLink('#include <windows.h>','SetLastError(0)'):
802a21658a3SLisandro Dalcin        self.addDefine('HAVE_SETLASTERROR',1)
8034e8afd12SMosè Giordano      if self.checkLink('#include <windows.h>\n','QueryPerformanceCounter(0);\n'):
804bea725cfSBarry Smith        self.addDefine('USE_MICROSOFT_TIME',1)
8054e8afd12SMosè Giordano    if self.libraries.add('Advapi32.lib','GetUserName',prototype='#include <windows.h>', call='GetUserName(NULL,NULL);'):
806f8833479SBarry Smith      self.addDefine('HAVE_GET_USER_NAME',1)
8074e8afd12SMosè Giordano    elif self.libraries.add('advapi32','GetUserName',prototype='#include <windows.h>', call='GetUserName(NULL,NULL);'):
808f8833479SBarry Smith      self.addDefine('HAVE_GET_USER_NAME',1)
809f8833479SBarry Smith
8104e8afd12SMosè Giordano    if not self.libraries.add('User32.lib','GetDC',prototype='#include <windows.h>',call='GetDC(0);'):
8114e8afd12SMosè Giordano      self.libraries.add('user32','GetDC',prototype='#include <windows.h>',call='GetDC(0);')
8124e8afd12SMosè Giordano    if not self.libraries.add('Gdi32.lib','CreateCompatibleDC',prototype='#include <windows.h>',call='CreateCompatibleDC(0);'):
8134e8afd12SMosè Giordano      self.libraries.add('gdi32','CreateCompatibleDC',prototype='#include <windows.h>',call='CreateCompatibleDC(0);')
814f8833479SBarry Smith
815f8833479SBarry Smith    self.types.check('int32_t', 'int')
81622164b4cSPierre Jolivet    if not self.checkCompile('#include <sys/types.h>\n','uid_t u;\n(void)u'):
817f8833479SBarry Smith      self.addTypedef('int', 'uid_t')
818f8833479SBarry Smith      self.addTypedef('int', 'gid_t')
81922164b4cSPierre 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'):
820f8833479SBarry Smith      self.framework.addDefine('R_OK', '04')
821f8833479SBarry Smith      self.framework.addDefine('W_OK', '02')
822f8833479SBarry Smith      self.framework.addDefine('X_OK', '01')
823f8833479SBarry Smith    if not self.checkLink('#include <sys/stat.h>\n','int a=0;\nif (S_ISDIR(a)){}\n'):
824f8833479SBarry Smith      self.framework.addDefine('S_ISREG(a)', '(((a)&_S_IFMT) == _S_IFREG)')
825f8833479SBarry Smith      self.framework.addDefine('S_ISDIR(a)', '(((a)&_S_IFMT) == _S_IFDIR)')
8264e8afd12SMosè Giordano    if self.checkCompile('#include <windows.h>\n','LARGE_INTEGER a;\nDWORD b=a.u.HighPart;\n'):
827f8833479SBarry Smith      self.addDefine('HAVE_LARGE_INTEGER_U',1)
828f8833479SBarry Smith
829f8833479SBarry Smith    # Windows requires a Binary file creation flag when creating/opening binary files.  Is a better test in order?
8304e8afd12SMosè Giordano    if self.checkCompile('#include <windows.h>\n#include <fcntl.h>\n', 'int flags = O_BINARY;'):
831f8833479SBarry Smith      self.addDefine('HAVE_O_BINARY',1)
832f8833479SBarry Smith
833f8833479SBarry Smith    if self.compilers.CC.find('win32fe') >= 0:
834ad4212abSSatish Balay      self.addDefine('HAVE_WINDOWS_COMPILERS',1)
835f8833479SBarry Smith      self.addDefine('DIR_SEPARATOR','\'\\\\\'')
836f8833479SBarry Smith      self.addDefine('REPLACE_DIR_SEPARATOR','\'/\'')
837f8833479SBarry Smith      self.addDefine('CANNOT_START_DEBUGGER',1)
8385188cb68SSatish Balay      (petscdir,error,status) = self.executeShellCommand('cygpath -w '+self.installdir.petscDir, log = self.log)
83934531a4dSSatish Balay      self.addDefine('DIR','"'+petscdir.replace('\\','\\\\')+'"')
8405188cb68SSatish Balay      (petscdir,error,status) = self.executeShellCommand('cygpath -m '+self.installdir.petscDir, log = self.log)
841e433681fSSatish Balay      self.addMakeMacro('wPETSC_DIR',petscdir)
8424e00a515SSatish Balay      if self.dataFilesPath.datafilespath:
8434e00a515SSatish Balay        (datafilespath,error,status) = self.executeShellCommand('cygpath -m '+self.dataFilesPath.datafilespath, log = self.log)
8444e00a515SSatish Balay        self.addMakeMacro('DATAFILESPATH',datafilespath)
8454e00a515SSatish Balay
846f8833479SBarry Smith    else:
847f8833479SBarry Smith      self.addDefine('REPLACE_DIR_SEPARATOR','\'\\\\\'')
848f8833479SBarry Smith      self.addDefine('DIR_SEPARATOR','\'/\'')
8495188cb68SSatish Balay      self.addDefine('DIR','"'+self.installdir.petscDir+'"')
8505188cb68SSatish Balay      self.addMakeMacro('wPETSC_DIR',self.installdir.petscDir)
8514e00a515SSatish Balay      if self.dataFilesPath.datafilespath:
8524e00a515SSatish Balay        self.addMakeMacro('DATAFILESPATH',self.dataFilesPath.datafilespath)
8535188cb68SSatish Balay    self.addDefine('ARCH','"'+self.installdir.petscArch+'"')
854f8833479SBarry Smith    return
855f8833479SBarry Smith
8567ce81a4bSJacob Faibussowitsch  def configureCoverage(self, lang, extra_coverage_flags=None, extra_debug_flags=None):
8577ce81a4bSJacob Faibussowitsch    """
8587ce81a4bSJacob Faibussowitsch    Check that a compiler accepts code-coverage flags. If the compiler does accept code-coverage flags
8597ce81a4bSJacob Faibussowitsch    try to set debugging flags equivalent to -Og.
8607ce81a4bSJacob Faibussowitsch
8617ce81a4bSJacob Faibussowitsch    Arguments:
8627ce81a4bSJacob Faibussowitsch    - lang: the language to check the coverage flag for
8637ce81a4bSJacob Faibussowitsch    - extra_coverage_flags: a list of extra flags to use when checking the coverage flags
8647ce81a4bSJacob Faibussowitsch    - extra_debug_flags: a list of extra flags to try when setting debug flags
8657ce81a4bSJacob Faibussowitsch
8667ce81a4bSJacob Faibussowitsch    On success:
8677ce81a4bSJacob Faibussowitsch    - defines PETSC_USE_COVERAGE to 1
8687ce81a4bSJacob Faibussowitsch    """
8697ce81a4bSJacob Faibussowitsch    def log_print(msg, *args, **kwargs):
8707ce81a4bSJacob Faibussowitsch      self.logPrint('checkCoverage: '+str(msg), *args, **kwargs)
8717ce81a4bSJacob Faibussowitsch      return
8727ce81a4bSJacob Faibussowitsch
8737ce81a4bSJacob Faibussowitsch    def quoted(string):
8747ce81a4bSJacob Faibussowitsch      return string.join(("'", "'"))
8757ce81a4bSJacob Faibussowitsch
8767ce81a4bSJacob Faibussowitsch    def make_flag_list(default, extra):
8777ce81a4bSJacob Faibussowitsch      ret = [default]
8787ce81a4bSJacob Faibussowitsch      if extra is not None:
8797ce81a4bSJacob Faibussowitsch        assert isinstance(extra, list)
8807ce81a4bSJacob Faibussowitsch        ret.extend(extra)
8817ce81a4bSJacob Faibussowitsch      return ret
8827ce81a4bSJacob Faibussowitsch
8837ce81a4bSJacob Faibussowitsch    log_print('Checking coverage flag for language {}'.format(lang))
8847ce81a4bSJacob Faibussowitsch    if not self.argDB['with-coverage']:
8857ce81a4bSJacob Faibussowitsch      log_print('coverage was disabled from command line or default')
8867ce81a4bSJacob Faibussowitsch      return
8877ce81a4bSJacob Faibussowitsch
8887ce81a4bSJacob Faibussowitsch    compiler  = self.getCompiler(lang=lang)
8897ce81a4bSJacob Faibussowitsch    is_gnuish = self.setCompilers.isGNU(compiler, self.log) or self.setCompilers.isClang(compiler, self.log)
8907ce81a4bSJacob Faibussowitsch
8917ce81a4bSJacob Faibussowitsch    # if not gnuish and we don't have a set of extra flags, bail
8927ce81a4bSJacob Faibussowitsch    if not is_gnuish and extra_coverage_flags is None:
8937ce81a4bSJacob 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)))
8947ce81a4bSJacob Faibussowitsch      return
8957ce81a4bSJacob Faibussowitsch
8967ce81a4bSJacob Faibussowitsch    coverage_flags = make_flag_list('--coverage', extra_coverage_flags)
8977ce81a4bSJacob Faibussowitsch    log_print('Checking set of coverage flags: {}'.format(coverage_flags))
8987ce81a4bSJacob Faibussowitsch
8997ce81a4bSJacob Faibussowitsch    found = 0
9007ce81a4bSJacob Faibussowitsch    with self.Language(lang):
9017ce81a4bSJacob Faibussowitsch      with self.setCompilers.Language(lang):
9027ce81a4bSJacob Faibussowitsch        for flag in coverage_flags:
9037ce81a4bSJacob Faibussowitsch          if self.setCompilers.checkCompilerFlag(flag) and self.checkLink():
9047ce81a4bSJacob Faibussowitsch            # compilerOnly = False, the linker also needs to see the coverage flag
9057ce81a4bSJacob Faibussowitsch            self.setCompilers.insertCompilerFlag(flag, False)
9067ce81a4bSJacob Faibussowitsch            found = 1
9077ce81a4bSJacob Faibussowitsch            break
9087ce81a4bSJacob Faibussowitsch          log_print(
9097ce81a4bSJacob Faibussowitsch            'Compiler {} did not accept coverage flag {}'.format(quoted(compiler), quoted(flag))
9107ce81a4bSJacob Faibussowitsch          )
9117ce81a4bSJacob Faibussowitsch
9127ce81a4bSJacob Faibussowitsch    if not found:
9137ce81a4bSJacob Faibussowitsch      log_print('Compiler {} did not accept ANY coverage flags: {}, bailing!'.format(quoted(compiler), coverage_flags))
9147ce81a4bSJacob Faibussowitsch      return
9157ce81a4bSJacob Faibussowitsch
9167ce81a4bSJacob Faibussowitsch    if not self.functions.haveFunction('__gcov_dump'):
9177ce81a4bSJacob Faibussowitsch      self.functions.checkClassify(['__gcov_dump'])
9187ce81a4bSJacob Faibussowitsch
9197ce81a4bSJacob Faibussowitsch    # now check if we can override the optimization level. It is only kosher to do so if
9207ce81a4bSJacob Faibussowitsch    # the user did not explicitly set the optimization flags (via CFLAGS, CXXFLAGS,
9217ce81a4bSJacob Faibussowitsch    # CXXOPTFLAGS, etc). If they have done so, we sternly warn them about their lapse in
9227ce81a4bSJacob Faibussowitsch    # judgement
9237ce81a4bSJacob Faibussowitsch    with self.Language(lang):
9247ce81a4bSJacob Faibussowitsch      compiler_flags = self.getCompilerFlags()
9257ce81a4bSJacob Faibussowitsch
9267ce81a4bSJacob Faibussowitsch    user_set          = 0
9277ce81a4bSJacob Faibussowitsch    allowed_opt_flags = re.compile(r'|'.join((r'-O[01g]', r'-g[1-9]*')))
9287ce81a4bSJacob Faibussowitsch    for flagsname in [self.getCompilerFlagsName(lang), self.compilerFlags.getOptionalFlagsName(lang)]:
9297ce81a4bSJacob Faibussowitsch      if flagsname in self.argDB:
9307ce81a4bSJacob Faibussowitsch        opt_flags = [
9317ce81a4bSJacob Faibussowitsch          f for f in self.compilerFlags.findOptFlags(compiler_flags) if not allowed_opt_flags.match(f)
9327ce81a4bSJacob Faibussowitsch        ]
9337ce81a4bSJacob Faibussowitsch        if opt_flags:
9347ce81a4bSJacob 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)))
9357ce81a4bSJacob Faibussowitsch          user_set = 1
9367ce81a4bSJacob Faibussowitsch          break
9377ce81a4bSJacob Faibussowitsch
9387ce81a4bSJacob Faibussowitsch    if not user_set:
9397ce81a4bSJacob Faibussowitsch      debug_flags = make_flag_list('-Og', extra_debug_flags)
9407ce81a4bSJacob Faibussowitsch      with self.setCompilers.Language(lang):
9417ce81a4bSJacob Faibussowitsch        for flag in debug_flags:
9427ce81a4bSJacob Faibussowitsch          try:
9437ce81a4bSJacob Faibussowitsch            self.setCompilers.addCompilerFlag(flag)
9447ce81a4bSJacob Faibussowitsch          except RuntimeError:
9457ce81a4bSJacob Faibussowitsch            continue
9467ce81a4bSJacob Faibussowitsch          break
9477ce81a4bSJacob Faibussowitsch
9487ce81a4bSJacob Faibussowitsch    self.addDefine('USE_COVERAGE', 1)
9497ce81a4bSJacob Faibussowitsch    return
9507ce81a4bSJacob Faibussowitsch
9517ce81a4bSJacob Faibussowitsch  def configureCoverageExecutable(self):
9527ce81a4bSJacob Faibussowitsch    """
9537ce81a4bSJacob Faibussowitsch    Check that a code-coverage collecting tool exists and is on PATH.
9547ce81a4bSJacob Faibussowitsch
9557ce81a4bSJacob Faibussowitsch    On success:
9567ce81a4bSJacob Faibussowitsch    - Adds PETSC_COVERAGE_EXEC make macro containing the full path to the coverage tool executable.
9577ce81a4bSJacob Faibussowitsch
9587ce81a4bSJacob Faibussowitsch    Raises RuntimeError if:
9597ce81a4bSJacob Faibussowitsch    - User explicitly requests auto-detection of the coverage tool from command line, and this
9607ce81a4bSJacob Faibussowitsch      routine fails to guess the suitable tool name.
9617ce81a4bSJacob Faibussowitsch    - The routine fails to find the tool, and --with-coverage is true
9627ce81a4bSJacob Faibussowitsch    """
9637ce81a4bSJacob Faibussowitsch    def log_print(msg, *args, **kwargs):
9647ce81a4bSJacob Faibussowitsch      self.logPrint('checkCoverage: '+str(msg), *args, **kwargs)
9657ce81a4bSJacob Faibussowitsch      return
9667ce81a4bSJacob Faibussowitsch
9677ce81a4bSJacob Faibussowitsch    def quoted(string):
9687ce81a4bSJacob Faibussowitsch      return string.join(("'", "'"))
9697ce81a4bSJacob Faibussowitsch
9707ce81a4bSJacob Faibussowitsch    required         = bool(self.argDB['with-coverage'])
9717ce81a4bSJacob Faibussowitsch    arg_opt          = self.argDB['with-coverage-exec']
9727ce81a4bSJacob Faibussowitsch    use_default_path = True
9737ce81a4bSJacob Faibussowitsch    search_path      = ''
9747ce81a4bSJacob Faibussowitsch
9757ce81a4bSJacob Faibussowitsch    log_print('{} to find an executable'.format('REQUIRED' if required else 'NOT required'))
9767ce81a4bSJacob Faibussowitsch    if arg_opt in {'auto', 'default-auto', '1'}:
9777ce81a4bSJacob Faibussowitsch      # detect it based on the C language compiler, hopefully this does not clash!
9787ce81a4bSJacob Faibussowitsch      compiler = self.getCompiler(lang=self.setCompilers.languages.clanguage)
9797ce81a4bSJacob Faibussowitsch      log_print('User did not explicitly set coverage exec (got {}), trying to auto-detect based on compiler {}'.format(quoted(arg_opt), quoted(compiler)))
9807ce81a4bSJacob Faibussowitsch      if self.setCompilers.isGNU(compiler, self.log):
9817ce81a4bSJacob Faibussowitsch        exec_names = ['gcov']
9827ce81a4bSJacob Faibussowitsch      elif self.setCompilers.isClang(compiler, self.log):
9837ce81a4bSJacob Faibussowitsch        exec_names = ['llvm-cov']
9847ce81a4bSJacob Faibussowitsch        if self.setCompilers.isDarwin(self.log):
9857ce81a4bSJacob Faibussowitsch          # macOS masquerades llvm-cov as just 'gcov', so we add this to the list in case
9867ce81a4bSJacob Faibussowitsch          # bare llvm-cov does not work
9877ce81a4bSJacob Faibussowitsch          exec_names.append('gcov')
9887ce81a4bSJacob Faibussowitsch      elif arg_opt == 'default-auto' and not required:
9897ce81a4bSJacob Faibussowitsch        # default-auto implies the user did not set it via command line!
9907ce81a4bSJacob 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)))
9917ce81a4bSJacob Faibussowitsch        return
9927ce81a4bSJacob Faibussowitsch      else:
9937ce81a4bSJacob Faibussowitsch        # implies 'auto' explicitly set by user, or we were required to find
9947ce81a4bSJacob Faibussowitsch        # something. either way we should error
9957ce81a4bSJacob Faibussowitsch        raise RuntimeError('Could not auto-detect coverage tool for {}, please set coverage tool name explicitly'.format(quoted(compiler)))
9967ce81a4bSJacob Faibussowitsch    else:
9977ce81a4bSJacob Faibussowitsch      log_print('User explicitly set coverage exec as {}'.format(quoted(arg_opt)))
9987ce81a4bSJacob Faibussowitsch      par_dir = os.path.dirname(arg_opt)
9997ce81a4bSJacob Faibussowitsch      if os.path.exists(par_dir):
10007ce81a4bSJacob Faibussowitsch        # arg_opt is path-like, we should only search the provided directory when we go
10017ce81a4bSJacob Faibussowitsch        # looking for the tool
10027ce81a4bSJacob Faibussowitsch        use_default_path = False
10037ce81a4bSJacob Faibussowitsch        search_path      = par_dir
10047ce81a4bSJacob Faibussowitsch      exec_names = [arg_opt]
10057ce81a4bSJacob Faibussowitsch
10067ce81a4bSJacob Faibussowitsch    make_macro_name = 'PETSC_COVERAGE_EXEC'
10077ce81a4bSJacob Faibussowitsch    log_print('Checking for coverage tool(s):\n{}'.format('\n'.join('- '+t for t in exec_names)))
10087ce81a4bSJacob Faibussowitsch    found_exec = self.getExecutables(
10097ce81a4bSJacob Faibussowitsch      exec_names,
10107ce81a4bSJacob Faibussowitsch      path=search_path, getFullPath=True, useDefaultPath=use_default_path, resultName=make_macro_name
10117ce81a4bSJacob Faibussowitsch    )
10127ce81a4bSJacob Faibussowitsch
10137ce81a4bSJacob Faibussowitsch    if found_exec is None:
10147ce81a4bSJacob Faibussowitsch      # didn't find the coverage tool
10157ce81a4bSJacob Faibussowitsch      if required:
10167ce81a4bSJacob Faibussowitsch        raise RuntimeError('Coverage tool(s) {} could not be found. Please provide explicit path to coverage tool'.format(exec_names))
10177ce81a4bSJacob Faibussowitsch      return
10187ce81a4bSJacob Faibussowitsch
10197ce81a4bSJacob Faibussowitsch    found_exec_name = os.path.basename(found_exec)
10207ce81a4bSJacob Faibussowitsch    if 'llvm-cov' in found_exec_name and 'gcov' not in found_exec_name:
10217ce81a4bSJacob Faibussowitsch      # llvm-cov needs to be called as 'llvm-cov gcov' to work
10227ce81a4bSJacob Faibussowitsch      self.addMakeMacro(make_macro_name, found_exec + ' gcov')
10237ce81a4bSJacob Faibussowitsch    return
10247ce81a4bSJacob Faibussowitsch
1025f8833479SBarry Smith#-----------------------------------------------------------------------------------------------------
1026b10d012aSSatish Balay  def configureCygwinBrokenPipe(self):
1027b10d012aSSatish Balay    '''Cygwin version <= 1.7.18 had issues with pipes and long commands invoked from gnu-make
1028b10d012aSSatish Balay    http://cygwin.com/ml/cygwin/2013-05/msg00340.html '''
10297fca349cSMatthew G. Knepley    if config.setCompilers.Configure.isCygwin(self.log):
1030b10d012aSSatish Balay      import platform
1031b10d012aSSatish Balay      import re
1032b10d012aSSatish Balay      r=re.compile("([0-9]+).([0-9]+).([0-9]+)")
1033b10d012aSSatish Balay      m=r.match(platform.release())
1034b10d012aSSatish Balay      major=int(m.group(1))
1035b10d012aSSatish Balay      minor=int(m.group(2))
1036b10d012aSSatish Balay      subminor=int(m.group(3))
1037b10d012aSSatish Balay      if ((major < 1) or (major == 1 and minor < 7) or (major == 1 and minor == 7 and subminor <= 18)):
1038b10d012aSSatish Balay        self.addMakeMacro('PETSC_CYGWIN_BROKEN_PIPE','1')
1039b10d012aSSatish Balay    return
1040b10d012aSSatish Balay
1041b10d012aSSatish Balay#-----------------------------------------------------------------------------------------------------
1042569865ddSSatish Balay  def configureDefaultArch(self):
1043af0996ceSBarry Smith    conffile = os.path.join('lib','petsc','conf', 'petscvariables')
1044569865ddSSatish Balay    if self.framework.argDB['with-default-arch']:
1045c6ef1b5bSJed Brown      fd = open(conffile, 'w')
1046569865ddSSatish Balay      fd.write('PETSC_ARCH='+self.arch.arch+'\n')
1047da93591fSBarry Smith      fd.write('PETSC_DIR='+self.petscdir.dir+'\n')
1048b9b902edSJed Brown      fd.write('include '+os.path.join('$(PETSC_DIR)','$(PETSC_ARCH)','lib','petsc','conf','petscvariables')+'\n')
1049569865ddSSatish Balay      fd.close()
1050569865ddSSatish Balay      self.framework.actions.addArgument('PETSc', 'Build', 'Set default architecture to '+self.arch.arch+' in '+conffile)
1051569865ddSSatish Balay    elif os.path.isfile(conffile):
1052569865ddSSatish Balay      try:
1053569865ddSSatish Balay        os.unlink(conffile)
1054569865ddSSatish Balay      except:
1055569865ddSSatish Balay        raise RuntimeError('Unable to remove file '+conffile+'. Did a different user create it?')
1056569865ddSSatish Balay    return
1057569865ddSSatish Balay
1058569865ddSSatish Balay#-----------------------------------------------------------------------------------------------------
1059f8833479SBarry Smith  def configureScript(self):
1060f8833479SBarry Smith    '''Output a script in the conf directory which will reproduce the configuration'''
1061f8833479SBarry Smith    import nargs
1062495bf1a9SSatish Balay    import sys
1063af0996ceSBarry Smith    scriptName = os.path.join(self.arch.arch,'lib','petsc','conf', 'reconfigure-'+self.arch.arch+'.py')
1064f8833479SBarry Smith    args = dict([(nargs.Arg.parseArgument(arg)[0], arg) for arg in self.framework.clArgs])
1065e97fc2efSSatish Balay    if 'with-clean' in args:
1066e97fc2efSSatish Balay      del args['with-clean']
1067d418e2d7SSatish Balay    if 'force' in args:
1068d418e2d7SSatish Balay      del args['force']
1069f8833479SBarry Smith    if 'configModules' in args:
10701063a081SSatish Balay      if nargs.Arg.parseArgument(args['configModules'])[1] == 'PETSc.Configure':
1071f8833479SBarry Smith        del args['configModules']
1072f8833479SBarry Smith    if 'optionsModule' in args:
107323a19ef1SSatish Balay      if nargs.Arg.parseArgument(args['optionsModule'])[1] == 'config.compilerOptions':
1074f8833479SBarry Smith        del args['optionsModule']
1075f8833479SBarry Smith    if not 'PETSC_ARCH' in args:
10761063a081SSatish Balay      args['PETSC_ARCH'] = 'PETSC_ARCH='+str(self.arch.arch)
1077c6ef1b5bSJed Brown    f = open(scriptName, 'w')
1078495bf1a9SSatish Balay    f.write('#!'+sys.executable+'\n')
1079f8833479SBarry Smith    f.write('if __name__ == \'__main__\':\n')
1080f8833479SBarry Smith    f.write('  import sys\n')
10817561c02cSSatish Balay    f.write('  import os\n')
10827561c02cSSatish Balay    f.write('  sys.path.insert(0, os.path.abspath(\'config\'))\n')
1083f8833479SBarry Smith    f.write('  import configure\n')
10841063a081SSatish Balay    # pretty print repr(args.values())
10851063a081SSatish Balay    f.write('  configure_options = [\n')
10868bec23c5SJed Brown    for itm in sorted(args.values()):
10871063a081SSatish Balay      f.write('    \''+str(itm)+'\',\n')
10881063a081SSatish Balay    f.write('  ]\n')
1089f8833479SBarry Smith    f.write('  configure.petsc_configure(configure_options)\n')
1090f8833479SBarry Smith    f.close()
1091f8833479SBarry Smith    try:
10925b6bfdb9SJed Brown      os.chmod(scriptName, 0o775)
10935b6bfdb9SJed Brown    except OSError as e:
1094f8833479SBarry Smith      self.framework.logPrint('Unable to make reconfigure script executable:\n'+str(e))
1095f8833479SBarry Smith    self.framework.actions.addArgument('PETSc', 'File creation', 'Created '+scriptName+' for automatic reconfiguration')
1096f8833479SBarry Smith    return
1097f8833479SBarry Smith
1098f8833479SBarry Smith  def configureInstall(self):
1099f8833479SBarry Smith    '''Setup the directories for installation'''
1100f8833479SBarry Smith    if self.framework.argDB['prefix']:
11015b4fc442SVaclav Hapla      self.addMakeRule('print_mesg_after_build','',
11025b4fc442SVaclav Hapla       ['-@echo "========================================="',
11035b4fc442SVaclav Hapla        '-@echo "Now to install the libraries do:"',
11045b4fc442SVaclav Hapla        '-@echo "%s${MAKE_USER} PETSC_DIR=${PETSC_DIR} PETSC_ARCH=${PETSC_ARCH} install"' % self.installdir.installSudo,
1105315b77e6SSatish Balay        '-@echo "========================================="'])
1106f8833479SBarry Smith    else:
11075b4fc442SVaclav Hapla      self.addMakeRule('print_mesg_after_build','',
11085b4fc442SVaclav Hapla       ['-@echo "========================================="',
11095b4fc442SVaclav Hapla        '-@echo "Now to check if the libraries are working do:"',
11105b4fc442SVaclav Hapla        '-@echo "${MAKE_USER} PETSC_DIR=${PETSC_DIR} PETSC_ARCH=${PETSC_ARCH} check"',
1111315b77e6SSatish Balay        '-@echo "========================================="'])
1112f8833479SBarry Smith      return
1113f8833479SBarry Smith
111428bb2e72SSatish Balay  def postProcessPackages(self):
111528bb2e72SSatish Balay    postPackages=[]
111628bb2e72SSatish Balay    for i in self.framework.packages:
111728bb2e72SSatish Balay      if hasattr(i,'postProcess'): postPackages.append(i)
111828bb2e72SSatish Balay    if postPackages:
1119e64d19dfSSatish Balay      # ctetgen needs petsc conf files. so attempt to create them early
1120a77eb93bSSatish Balay      self.framework.dumpConfFiles()
1121d9293e7bSBarry Smith      # tacky fix for dependency of Aluimia on Pflotran; requested via petsc-dev Matt provide a correct fix
1122d9293e7bSBarry Smith      for i in postPackages:
1123d9293e7bSBarry Smith        if i.name.upper() in ['PFLOTRAN']:
1124d9293e7bSBarry Smith          i.postProcess()
1125d9293e7bSBarry Smith          postPackages.remove(i)
112628bb2e72SSatish Balay      for i in postPackages: i.postProcess()
1127aa5c8b8eSBarry Smith      for i in postPackages:
1128aa5c8b8eSBarry Smith        if i.installedpetsc:
1129aa5c8b8eSBarry Smith          self.installed = 1
1130aa5c8b8eSBarry Smith          break
113128bb2e72SSatish Balay    return
1132f8833479SBarry Smith
1133f8833479SBarry Smith  def configure(self):
1134bf3e94a3SBarry Smith    if 'package-prefix-hash' in self.argDB:
1135bf3e94a3SBarry Smith      # turn off prefix if it was only used to for installing external packages.
1136bf3e94a3SBarry Smith      self.framework.argDB['prefix'] = ''
1137bf3e94a3SBarry Smith      self.dir = os.path.abspath(os.path.join(self.petscdir.dir, self.arch.arch))
1138bf3e94a3SBarry Smith      self.installdir.dir = self.dir
1139bf3e94a3SBarry Smith      self.installdir.petscDir = self.petscdir.dir
1140bf3e94a3SBarry Smith      self.petscDir = self.petscdir.dir
1141bf3e94a3SBarry Smith      self.petscArch = self.arch.arch
1142bf3e94a3SBarry Smith      self.addMakeMacro('PREFIXDIR',self.dir)
1143bf3e94a3SBarry Smith      self.confDir = os.path.abspath(os.path.join(self.petscdir.dir, self.arch.arch))
1144bf3e94a3SBarry Smith
1145f8833479SBarry Smith    if not os.path.samefile(self.petscdir.dir, os.getcwd()):
1146f8833479SBarry Smith      raise RuntimeError('Wrong PETSC_DIR option specified: '+str(self.petscdir.dir) + '\n  Configure invoked in: '+os.path.realpath(os.getcwd()))
1147550489e3SMatthew 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):
11483552d8fbSSatish Balay      raise RuntimeError('Incorrect option --prefix='+self.framework.argDB['prefix']+' specified. It cannot be same as PETSC_DIR!')
11498fd0dbdbSBarry Smith    if self.framework.argDB['prefix'] and self.framework.argDB['prefix'].find(' ') > -1:
11508fd0dbdbSBarry 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')
1151c16c35a9SSatish 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)):
1152c16c35a9SSatish Balay      raise RuntimeError('Incorrect option --prefix='+self.framework.argDB['prefix']+' specified. It cannot be same as PETSC_DIR/PETSC_ARCH!')
1153f16c1317SJed Brown    self.framework.header          = os.path.join(self.arch.arch,'include','petscconf.h')
1154f16c1317SJed Brown    self.framework.cHeader         = os.path.join(self.arch.arch,'include','petscfix.h')
1155bf113f49SJacob Faibussowitsch    self.framework.poisonheader    = os.path.join(self.arch.arch,'include','petscconf_poison.h')
11569c735a01SStefano Zampini    self.framework.pkgheader       = os.path.join(self.arch.arch,'include','petscpkg_version.h')
1157af0996ceSBarry Smith    self.framework.makeMacroHeader = os.path.join(self.arch.arch,'lib','petsc','conf','petscvariables')
1158af0996ceSBarry Smith    self.framework.makeRuleHeader  = os.path.join(self.arch.arch,'lib','petsc','conf','petscrules')
1159f8833479SBarry Smith    if self.libraries.math is None:
1160f8833479SBarry Smith      raise RuntimeError('PETSc requires a functional math library. Please send configure.log to petsc-maint@mcs.anl.gov.')
1161f8833479SBarry Smith    if self.languages.clanguage == 'Cxx' and not hasattr(self.compilers, 'CXX'):
1162f8833479SBarry Smith      raise RuntimeError('Cannot set C language to C++ without a functional C++ compiler.')
1163ed938b00SJed Brown    self.executeTest(self.configureRTLDDefault)
1164b2843cf1SBarry Smith    self.executeTest(self.configurePrefetch)
11652400fdedSBarry Smith    self.executeTest(self.configureUnused)
11661ef8df7fSJed Brown    self.executeTest(self.configureDeprecated)
116798ed35c3SBarry Smith    self.executeTest(self.configureIsatty)
1168e8e972b2SVaclav Hapla    self.executeTest(self.configureExpect)
1169e8e972b2SVaclav Hapla    self.executeTest(self.configureAlign)
1170e8e972b2SVaclav Hapla    self.executeTest(self.configureFunctionName)
1171e8e972b2SVaclav Hapla    self.executeTest(self.configureIntptrt)
1172f8833479SBarry Smith    self.executeTest(self.configureSolaris)
1173f8833479SBarry Smith    self.executeTest(self.configureLinux)
11740f64ec89SBarry Smith    self.executeTest(self.configureDarwin)
1175f8833479SBarry Smith    self.executeTest(self.configureWin32)
1176b10d012aSSatish Balay    self.executeTest(self.configureCygwinBrokenPipe)
1177569865ddSSatish Balay    self.executeTest(self.configureDefaultArch)
1178f8833479SBarry Smith    self.executeTest(self.configureScript)
1179f8833479SBarry Smith    self.executeTest(self.configureInstall)
118009bc878fSSatish Balay    self.executeTest(self.configureAtoll)
11817ce81a4bSJacob Faibussowitsch    for LANG in ['C', 'Cxx', 'CUDA', 'HIP', 'SYCL', 'FC']:
11827ce81a4bSJacob Faibussowitsch      compilerName = LANG.upper() if LANG in {'Cxx', 'FC'} else LANG+'C'
11837ce81a4bSJacob Faibussowitsch      if hasattr(self.setCompilers, compilerName):
11847ce81a4bSJacob Faibussowitsch        kwargs = {}
11857ce81a4bSJacob Faibussowitsch        if LANG in {'CUDA'}:
11867ce81a4bSJacob Faibussowitsch          # nvcc preprocesses the base file into a bunch of intermediate files, which are
11877ce81a4bSJacob Faibussowitsch          # then compiled by the host compiler. Why is this a problem?  Because the
11887ce81a4bSJacob Faibussowitsch          # generated coverage data is based on these preprocessed source files! So gcov
11897ce81a4bSJacob Faibussowitsch          # tries to read it later, but since its in the tmp directory it cannot. Thus we
11907ce81a4bSJacob Faibussowitsch          # need to keep them around (in a place we know about).
11917ce81a4bSJacob Faibussowitsch          nvcc_tmp_dir = os.path.join(self.petscdir.dir, self.arch.arch, 'nvcc_tmp')
11927ce81a4bSJacob Faibussowitsch          try:
11937ce81a4bSJacob Faibussowitsch            os.mkdir(nvcc_tmp_dir)
11947ce81a4bSJacob Faibussowitsch          except FileExistsError:
11957ce81a4bSJacob Faibussowitsch            pass
11967ce81a4bSJacob Faibussowitsch          kwargs['extra_coverage_flags'] = [
11977ce81a4bSJacob Faibussowitsch            '-Xcompiler --coverage -Xcompiler -fPIC --keep --keep-dir={}'.format(nvcc_tmp_dir)
11987ce81a4bSJacob Faibussowitsch          ]
11997ce81a4bSJacob Faibussowitsch          if self.kokkos.found:
12007ce81a4bSJacob Faibussowitsch            # yet again the kokkos nvcc_wrapper goes out of its way to be as useless as
12017ce81a4bSJacob Faibussowitsch            # possible. Its default arch (sm_35) is actually too low to compile kokkos,
12027ce81a4bSJacob Faibussowitsch            # for whatever reason this works if you dont use the --keep and --keep-dir
12037ce81a4bSJacob Faibussowitsch            # flags above.
12047ce81a4bSJacob Faibussowitsch            kwargs['extra_coverage_flags'].append('-arch=native')
12057ce81a4bSJacob Faibussowitsch          kwargs['extra_debug_flags'] = ['-Xcompiler -Og']
12067ce81a4bSJacob Faibussowitsch        self.executeTest(self.configureCoverage, args=[LANG], kargs=kwargs)
12077ce81a4bSJacob Faibussowitsch    self.executeTest(self.configureCoverageExecutable)
1208f8833479SBarry Smith
1209f8833479SBarry Smith    self.Dump()
1210f8833479SBarry Smith    self.dumpConfigInfo()
12112a4161d9SMatthew G Knepley    self.dumpMachineInfo()
121249fe22e6SSatish Balay    self.delGenFiles()
121340277576SBarry Smith    # need to save the current state of BuildSystem so that postProcess() packages can read it in and perhaps run make install
121440277576SBarry Smith    self.framework.storeSubstitutions(self.framework.argDB)
1215492432c8SJed Brown    self.framework.argDB['configureCache'] = pickle.dumps(self.framework)
121640277576SBarry Smith    self.framework.argDB.save(force = True)
12178244ab14SJed Brown    self.DumpPkgconfig('PETSc.pc')
12188244ab14SJed Brown    self.DumpPkgconfig('petsc.pc')
1219351d3a41SMatthew G Knepley    self.DumpModule()
1220f7ad81e1SBarry Smith    self.postProcessPackages()
1221f8833479SBarry Smith    self.framework.log.write('================================================================================\n')
1222f8833479SBarry Smith    self.logClear()
1223f8833479SBarry Smith    return
1224