xref: /petsc/config/PETSc/Configure.py (revision 749c190bad46ba447444c173d8c7a4080c70750e)
1f8833479SBarry Smithimport config.base
2f8833479SBarry Smith
3f8833479SBarry Smithimport os
46dd73af6SBarry Smithimport sys
5f8833479SBarry Smithimport re
6492432c8SJed Brownimport pickle
7f8833479SBarry Smith
8f8833479SBarry Smithclass Configure(config.base.Configure):
9f8833479SBarry Smith  def __init__(self, framework):
10f8833479SBarry Smith    config.base.Configure.__init__(self, framework)
11f8833479SBarry Smith    self.headerPrefix = 'PETSC'
12f8833479SBarry Smith    self.substPrefix  = 'PETSC'
13aa5c8b8eSBarry Smith    self.installed    = 0 # 1 indicates that Configure itself has already compiled and installed PETSc
144161761dSBarry Smith    self.found        = 1
15f8833479SBarry Smith    return
16f8833479SBarry Smith
177c939e48SSatish Balay  def __str2__(self):
18e4d7ee71SJacob Faibussowitsch    import logger
19e4d7ee71SJacob Faibussowitsch
205b4fc442SVaclav Hapla    desc = ['  Using GNU make: ' + self.make.make]
217ce81a4bSJacob Faibussowitsch    if self.defines.get('USE_COVERAGE'):
227ce81a4bSJacob Faibussowitsch      desc.extend([
237ce81a4bSJacob Faibussowitsch        '  Code coverage: yes',
247ce81a4bSJacob Faibussowitsch        '  Using code coverage executable: {}'.format(self.getMakeMacro('PETSC_COVERAGE_EXEC'))
257ce81a4bSJacob Faibussowitsch      ])
26e4d7ee71SJacob Faibussowitsch    banner_ends   = 'xxx'
27e4d7ee71SJacob Faibussowitsch    banner_middle = '=' * (logger.get_global_divider_length() - 2 * len(banner_ends))
28e4d7ee71SJacob Faibussowitsch    banner_line   = banner_middle.join((banner_ends, banner_ends))
29e4d7ee71SJacob Faibussowitsch    desc.append(banner_line)
30aa5c8b8eSBarry Smith    if not self.installed:
31dc0529c6SBarry Smith      desc.append(' Configure stage complete. Now build PETSc libraries with:')
325b4fc442SVaclav Hapla      desc.append('   %s PETSC_DIR=%s PETSC_ARCH=%s all' % (self.make.make_user, self.petscdir.dir, self.arch.arch))
33aa5c8b8eSBarry Smith    else:
34aa5c8b8eSBarry Smith      desc.append(' Installation complete. You do not need to run make to compile or install the software')
35e4d7ee71SJacob Faibussowitsch    desc.extend([banner_line, ''])
367ce81a4bSJacob Faibussowitsch    return '\n'.join(desc)
37f8833479SBarry Smith
38f8833479SBarry Smith  def setupHelp(self, help):
39f8833479SBarry Smith    import nargs
40ce0b2093SBarry Smith    help.addArgument('PETSc',  '-prefix=<dir>',                              nargs.Arg(None, '', 'Specifiy location to install PETSc (eg. /usr/local)'))
417deb5ab3SBarry Smith    help.addArgument('PETSc',  '-with-prefetch=<bool>',                      nargs.ArgBool(None, 1,'Enable checking for prefetch instructions'))
42eed94e11SSatish Balay    help.addArgument('Windows','-with-windows-graphics=<bool>',              nargs.ArgBool(None, 1,'Enable check for Windows Graphics'))
43569865ddSSatish Balay    help.addArgument('PETSc', '-with-default-arch=<bool>',                   nargs.ArgBool(None, 1, 'Allow using the last configured arch without setting PETSC_ARCH'))
4457cb31baSSatish Balay    help.addArgument('PETSc','-with-single-library=<bool>',                  nargs.ArgBool(None, 1,'Put all PETSc code into the single -lpetsc library'))
45cb297985SSatish Balay    help.addArgument('PETSc','-with-fortran-bindings=<bool>',                nargs.ArgBool(None, 1,'Build PETSc fortran bindings in the library and corresponding module files'))
468143cb67SSatish Balay    help.addArgument('PETSc', '-with-library-name-suffix=<string>',          nargs.Arg(None, '', 'Add a suffix to PETSc library names'))
47525d6f2eSBarry Smith    help.addArgument('PETSc', '-with-ios=<bool>',                            nargs.ArgBool(None, 0, 'Build an iPhone/iPad version of PETSc library'))
4821afe8ebSBarry Smith    help.addArgument('PETSc', '-with-display=<x11display>',                  nargs.Arg(None, '', 'Specifiy DISPLAY environmental variable for use with MATLAB test)'))
492c30b4dfSSatish Balay    help.addArgument('PETSc', '-with-package-scripts=<pyscripts>',           nargs.ArgFileList(None,None,'Specify configure package scripts for user provided packages'))
507ce81a4bSJacob Faibussowitsch    help.addArgument('PETSc', '-with-coverage=<bool>',                       nargs.ArgFuzzyBool(None, value=0, help='Enable or disable code-coverage collection'))
517ce81a4bSJacob 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'))
52a0c7f9aaSSamuel Khuvis    help.addArgument('PETSc', '-with-tau-perfstubs=<bool>',                  nargs.ArgBool(None, 1,'Enable TAU profiler stubs'))
53689a5dfaSJacob Faibussowitsch    help.addArgument('PETSc', '-with-strict-petscerrorcode=<bool>',          nargs.ArgFuzzyBool(None, value=0, help='Enable strict PetscErrorCode mode, which enables additional compile-time checking for misuse of PetscErrorCode and error handling'))
54f8833479SBarry Smith    return
55f8833479SBarry Smith
566dd73af6SBarry Smith  def registerPythonFile(self,filename,directory):
576dd73af6SBarry Smith    ''' Add a python file to the framework and registers its headerprefix, ... externalpackagedir
586dd73af6SBarry Smith        directory is the directory where the file relative to the BuildSystem or config path in python notation with . '''
596dd73af6SBarry Smith    (utilityName, ext) = os.path.splitext(filename)
606dd73af6SBarry Smith    if not utilityName.startswith('.') and not utilityName.startswith('#') and ext == '.py' and not utilityName == '__init__':
616dd73af6SBarry Smith      if directory: directory = directory+'.'
626dd73af6SBarry Smith      utilityObj                             = self.framework.require(directory+utilityName, self)
636dd73af6SBarry Smith      utilityObj.headerPrefix                = self.headerPrefix
646dd73af6SBarry Smith      utilityObj.archProvider                = self.arch
656dd73af6SBarry Smith      utilityObj.languageProvider            = self.languages
666dd73af6SBarry Smith      utilityObj.installDirProvider          = self.installdir
676dd73af6SBarry Smith      utilityObj.externalPackagesDirProvider = self.externalpackagesdir
686dd73af6SBarry Smith      utilityObj.precisionProvider           = self.scalartypes
696dd73af6SBarry Smith      utilityObj.indexProvider               = self.indexTypes
706dd73af6SBarry Smith      setattr(self, utilityName.lower(), utilityObj)
7151294b80SMatthew G. Knepley      return utilityObj
7251294b80SMatthew G. Knepley    return None
736dd73af6SBarry Smith
74f8833479SBarry Smith  def setupDependencies(self, framework):
75f8833479SBarry Smith    config.base.Configure.setupDependencies(self, framework)
76dca78d2bSSatish Balay    self.programs      = framework.require('config.programs',           self)
77f8833479SBarry Smith    self.setCompilers  = framework.require('config.setCompilers',       self)
787ce81a4bSJacob Faibussowitsch    self.compilerFlags = framework.require('config.compilerFlags',      self)
7930b8aa07SMatthew G. Knepley    self.compilers     = framework.require('config.compilers',          self)
809d310bb7SBarry Smith    self.arch          = framework.require('PETSc.options.arch',        self.setCompilers)
819d310bb7SBarry Smith    self.petscdir      = framework.require('PETSc.options.petscdir',    self.arch)
829d310bb7SBarry Smith    self.installdir    = framework.require('PETSc.options.installDir',  self)
834e00a515SSatish Balay    self.dataFilesPath = framework.require('PETSc.options.dataFilesPath',self)
846dd73af6SBarry Smith    self.scalartypes   = framework.require('PETSc.options.scalarTypes', self)
856dd73af6SBarry Smith    self.indexTypes    = framework.require('PETSc.options.indexTypes',  self)
869d310bb7SBarry Smith    self.languages     = framework.require('PETSc.options.languages',   self.setCompilers)
8730b8aa07SMatthew G. Knepley    self.indexTypes    = framework.require('PETSc.options.indexTypes',  self.compilers)
88f8833479SBarry Smith    self.types         = framework.require('config.types',              self)
89f8833479SBarry Smith    self.headers       = framework.require('config.headers',            self)
90f8833479SBarry Smith    self.functions     = framework.require('config.functions',          self)
91f8833479SBarry Smith    self.libraries     = framework.require('config.libraries',          self)
92cd37d877SShri Abhyankar    self.atomics       = framework.require('config.atomics',            self)
939481793eSSatish Balay    self.make          = framework.require('config.packages.make',      self)
949552296fSBarry Smith    self.blasLapack    = framework.require('config.packages.BlasLapack',self)
95e6b0c433SBarry Smith    self.mpi           = framework.require('config.packages.MPI',       self)
960542e31aSBarry Smith    self.fortran       = framework.require('config.compilersFortran',   self)
978d35c829SSatish Balay    self.ftncmdline    = framework.require('config.utilities.fortranCommandLine',self)
980542e31aSBarry Smith    self.externalpackagesdir = framework.require('PETSc.options.externalpackagesdir',self)
9949d43ecaSSatish Balay
10009a6cbfcSBernhard M. Wiedemann    for utility in sorted(os.listdir(os.path.join('config','PETSc','options'))):
1016dd73af6SBarry Smith      self.registerPythonFile(utility,'PETSc.options')
1029d310bb7SBarry Smith
10309a6cbfcSBernhard M. Wiedemann    for utility in sorted(os.listdir(os.path.join('config','BuildSystem','config','utilities'))):
1046dd73af6SBarry Smith      self.registerPythonFile(utility,'config.utilities')
10506e08bc7SBarry Smith
10609a6cbfcSBernhard M. Wiedemann    for package in sorted(os.listdir(os.path.join('config', 'BuildSystem', 'config', 'packages'))):
10751294b80SMatthew G. Knepley      obj = self.registerPythonFile(package,'config.packages')
10851294b80SMatthew G. Knepley      if obj:
10951294b80SMatthew G. Knepley        obj.archProvider                = self.framework.requireModule(obj.archProvider, obj)
11051294b80SMatthew G. Knepley        obj.languageProvider            = self.framework.requireModule(obj.languageProvider, obj)
11151294b80SMatthew G. Knepley        obj.installDirProvider          = self.framework.requireModule(obj.installDirProvider, obj)
11251294b80SMatthew G. Knepley        obj.externalPackagesDirProvider = self.framework.requireModule(obj.externalPackagesDirProvider, obj)
11351294b80SMatthew G. Knepley        obj.precisionProvider           = self.framework.requireModule(obj.precisionProvider, obj)
11451294b80SMatthew G. Knepley        obj.indexProvider               = self.framework.requireModule(obj.indexProvider, obj)
1156dd73af6SBarry Smith
1165faf1eacSMatthew G. Knepley    # Force blaslapack and opencl to depend on scalarType so precision is set before BlasLapack is built
1179d310bb7SBarry Smith    framework.require('PETSc.options.scalarTypes', self.f2cblaslapack)
1189d310bb7SBarry Smith    framework.require('PETSc.options.scalarTypes', self.fblaslapack)
1199d310bb7SBarry Smith    framework.require('PETSc.options.scalarTypes', self.blaslapack)
1205faf1eacSMatthew G. Knepley    framework.require('PETSc.options.scalarTypes', self.opencl)
121f8833479SBarry Smith
122dca78d2bSSatish Balay    self.programs.headerPrefix     = self.headerPrefix
1235b3958d3SJacob Faibussowitsch    self.setCompilers.headerPrefix = self.headerPrefix
124f8833479SBarry Smith    self.compilers.headerPrefix    = self.headerPrefix
1250542e31aSBarry Smith    self.fortran.headerPrefix      = self.headerPrefix
126f8833479SBarry Smith    self.types.headerPrefix        = self.headerPrefix
127f8833479SBarry Smith    self.headers.headerPrefix      = self.headerPrefix
128f8833479SBarry Smith    self.functions.headerPrefix    = self.headerPrefix
129f8833479SBarry Smith    self.libraries.headerPrefix    = self.headerPrefix
1306dd73af6SBarry Smith
1312c30b4dfSSatish Balay    # Register user provided package scripts
1322c30b4dfSSatish Balay    if 'with-package-scripts' in self.framework.argDB:
1332c30b4dfSSatish Balay      for script in self.framework.argDB['with-package-scripts']:
1342c30b4dfSSatish Balay        if os.path.splitext(script)[1] != '.py':
1352c30b4dfSSatish Balay          raise RuntimeError('Only python scripts compatible with configure package script format should be specified! Invalid option -with-package-scripts='+script)
1362c30b4dfSSatish Balay        self.framework.logPrint('User is registering a new package script: '+script)
1372c30b4dfSSatish Balay        dname,fname = os.path.split(script)
1382c30b4dfSSatish Balay        if dname: sys.path.append(dname)
1392c30b4dfSSatish Balay        self.registerPythonFile(fname,'')
1406dd73af6SBarry Smith
1416dd73af6SBarry Smith    # test for a variety of basic headers and functions
1424211eb48SBarry Smith    headersC = map(lambda name: name+'.h',['setjmp','dos','fcntl','float','io','malloc','pwd','strings',
143ace159c0SJed Brown                                            'unistd','machine/endian','sys/param','sys/procfs','sys/resource',
1442475b7caSBarry Smith                                            'sys/systeminfo','sys/times','sys/utsname',
1457e4f0192SMosè Giordano                                            'sys/socket','sys/wait','netinet/in','netdb','direct','time','Ws2tcpip','sys/types',
14618da0197SPierre Jolivet                                            'WindowsX','float','ieeefp','stdint','inttypes','immintrin'])
14745082d64SJed Brown    functions = ['access','_access','clock','drand48','getcwd','_getcwd','getdomainname','gethostname',
1484e04ae5aSSatish Balay                 'posix_memalign','popen','PXFGETARG','rand','getpagesize',
1494211eb48SBarry Smith                 'readlink','realpath','usleep','sleep','_sleep',
1502475b7caSBarry Smith                 'uname','snprintf','_snprintf','lseek','_lseek','time','fork','stricmp',
1512475b7caSBarry Smith                 'strcasecmp','bzero','dlopen','dlsym','dlclose','dlerror',
152c81e5e31SPierre Jolivet                 '_mkdir','socket','gethostbyname','fpresetsticky',
15318da0197SPierre Jolivet                 'fpsetsticky','__gcov_dump']
154b0651e32SBarry Smith    libraries = [(['fpe'],'handle_sigfpes')]
155b0651e32SBarry Smith    librariessock = [(['socket','nsl'],'socket')]
156f8833479SBarry Smith    self.headers.headers.extend(headersC)
157f8833479SBarry Smith    self.functions.functions.extend(functions)
158b0651e32SBarry Smith    self.libraries.libraries.extend(libraries)
159b0651e32SBarry Smith    if not hasattr(self,'socket'):
160b0651e32SBarry Smith      self.libraries.libraries.extend(librariessock)
161f8833479SBarry Smith    return
162f8833479SBarry Smith
1638244ab14SJed Brown  def DumpPkgconfig(self, petsc_pc):
164262119f8SBarry Smith    ''' Create a pkg-config file '''
165262119f8SBarry Smith    if not os.path.exists(os.path.join(self.petscdir.dir,self.arch.arch,'lib','pkgconfig')):
166262119f8SBarry Smith      os.makedirs(os.path.join(self.petscdir.dir,self.arch.arch,'lib','pkgconfig'))
1672eefe1c6SJed Brown    with open(os.path.join(self.petscdir.dir,self.arch.arch,'lib','pkgconfig',petsc_pc),'w') as fd:
1685e3311eeSJed Brown      cflags_inc = ['-I${includedir}']
169262119f8SBarry Smith      if self.framework.argDB['prefix']:
1705bb5b263SMatthew G. Knepley        fd.write('prefix='+self.installdir.dir+'\n')
171262119f8SBarry Smith      else:
172e1e675deSJed Brown        fd.write('prefix='+os.path.join(self.petscdir.dir, self.arch.arch)+'\n')
173e1e675deSJed Brown        cflags_inc.append('-I' + os.path.join(self.petscdir.dir, 'include'))
174262119f8SBarry Smith      fd.write('exec_prefix=${prefix}\n')
175262119f8SBarry Smith      fd.write('includedir=${prefix}/include\n')
1765e3311eeSJed Brown      fd.write('libdir=${prefix}/lib\n')
177262119f8SBarry Smith
1782eefe1c6SJed Brown      with self.setCompilers.Language('C'):
179262119f8SBarry Smith        fd.write('ccompiler='+self.setCompilers.getCompiler()+'\n')
180756c7f9fSJed Brown        fd.write('cflags_extra='+self.setCompilers.getCompilerFlags().strip()+'\n')
181756c7f9fSJed Brown        fd.write('cflags_dep='+self.compilers.dependenciesGenerationFlag.get('C','')+'\n')
182756c7f9fSJed Brown        fd.write('ldflag_rpath='+self.setCompilers.CSharedLinkerFlag+'\n')
183f8b9f887SBarry Smith        fd.write('ldflags='+self.setCompilers.getLinkerFlags().strip()+'\n')
18403e383c8SJed Brown      if hasattr(self.compilers, 'CXX'):
1852eefe1c6SJed Brown        with self.setCompilers.Language('C++'):
186262119f8SBarry Smith          fd.write('cxxcompiler='+self.setCompilers.getCompiler()+'\n')
187756c7f9fSJed Brown          fd.write('cxxflags_extra='+self.setCompilers.getCompilerFlags().strip()+'\n')
188262119f8SBarry Smith      if hasattr(self.compilers, 'FC'):
1892eefe1c6SJed Brown        with self.setCompilers.Language('FC'):
190262119f8SBarry Smith          fd.write('fcompiler='+self.setCompilers.getCompiler()+'\n')
191756c7f9fSJed Brown          fd.write('fflags_extra='+self.setCompilers.getCompilerFlags().strip()+'\n')
19250520af6SPatrick Sanan      if hasattr(self.compilers, 'CUDAC'):
19350520af6SPatrick Sanan        with self.setCompilers.Language('CUDA'):
19450520af6SPatrick Sanan          fd.write('cudacompiler='+self.setCompilers.getCompiler()+'\n')
19550520af6SPatrick Sanan          fd.write('cudaflags_extra='+self.setCompilers.getCompilerFlags().strip()+'\n')
196f605775fSPierre Jolivet          p = self.framework.require('config.packages.CUDA')
19750520af6SPatrick Sanan          fd.write('cudalib='+self.libraries.toStringNoDupes(p.lib)+'\n')
19850520af6SPatrick Sanan          fd.write('cudainclude='+self.headers.toStringNoDupes(p.include)+'\n')
1997ba7a817SBarry Smith          if hasattr(self.setCompilers,'CUDA_CXX'):
2007ba7a817SBarry Smith            fd.write('cuda_cxx='+self.setCompilers.CUDA_CXX+'\n')
2017ba7a817SBarry Smith            fd.write('cuda_cxxflags='+self.setCompilers.CUDA_CXXFLAGS+'\n')
202f8b9f887SBarry Smith      fd.write('mpiexec='+self.mpi.mpiexec+'\n')
203d03e3399SBarry Smith      if self.python.path:
204d03e3399SBarry Smith        if 'PYTHONPATH' in os.environ:
205d03e3399SBarry Smith          fd.write('PETSCPYTHONPATH='+':'.join(self.python.path)+':'+os.environ['PYTHONPATH']+'\n')
206d03e3399SBarry Smith        else:
207d03e3399SBarry Smith          fd.write('PETSCPYTHONPATH='+':'.join(self.python.path)+'\n')
208262119f8SBarry Smith
209262119f8SBarry Smith      fd.write('\n')
210262119f8SBarry Smith      fd.write('Name: PETSc\n')
211262119f8SBarry Smith      fd.write('Description: Library to solve ODEs and algebraic equations\n')
212351d3a41SMatthew G Knepley      fd.write('Version: %s\n' % self.petscdir.version)
2135e3311eeSJed Brown      fd.write('Cflags: ' + ' '.join([self.setCompilers.CPPFLAGS] + cflags_inc) + '\n')
21437371b91SJed Brown      fd.write('Libs: '+self.libraries.toStringNoDupes(['-L${libdir}', self.petsclib], with_rpath=False)+'\n')
2158ebf8858SJed Brown      # Remove RPATH flags from library list.  User can add them using
2168ebf8858SJed Brown      # pkg-config --variable=ldflag_rpath and pkg-config --libs-only-L
217de8f682fSSatish 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')
218262119f8SBarry Smith    return
219262119f8SBarry Smith
220351d3a41SMatthew G Knepley  def DumpModule(self):
221351d3a41SMatthew G Knepley    ''' Create a module file '''
222af0996ceSBarry Smith    if not os.path.exists(os.path.join(self.petscdir.dir,self.arch.arch,'lib','petsc','conf','modules')):
223af0996ceSBarry Smith      os.makedirs(os.path.join(self.petscdir.dir,self.arch.arch,'lib','petsc','conf','modules'))
224af0996ceSBarry Smith    if not os.path.exists(os.path.join(self.petscdir.dir,self.arch.arch,'lib','petsc','conf','modules','petsc')):
225af0996ceSBarry Smith      os.makedirs(os.path.join(self.petscdir.dir,self.arch.arch,'lib','petsc','conf','modules','petsc'))
226351d3a41SMatthew G Knepley    if self.framework.argDB['prefix']:
2275bb5b263SMatthew G. Knepley      installdir  = self.installdir.dir
22855d606a3SSatish Balay      installarch = ''
22955d606a3SSatish Balay      installpath = os.path.join(installdir,'bin')
230351d3a41SMatthew G Knepley    else:
231351d3a41SMatthew G Knepley      installdir  = self.petscdir.dir
23255d606a3SSatish Balay      installarch = self.arch.arch
23355d606a3SSatish Balay      installpath = os.path.join(installdir,installarch,'bin')+':'+os.path.join(installdir,'bin')
234af0996ceSBarry Smith    fd = open(os.path.join(self.petscdir.dir,self.arch.arch,'lib','petsc','conf','modules','petsc',self.petscdir.version),'w')
235351d3a41SMatthew G Knepley    fd.write('''\
236351d3a41SMatthew G Knepley#%%Module
237351d3a41SMatthew G Knepley
238351d3a41SMatthew G Knepleyproc ModulesHelp { } {
239351d3a41SMatthew G Knepley    puts stderr "This module sets the path and environment variables for petsc-%s"
240a17b96a8SKyle Gerard Felker    puts stderr "     see https://petsc.org/ for more information      "
241351d3a41SMatthew G Knepley    puts stderr ""
242351d3a41SMatthew G Knepley}
243351d3a41SMatthew G Knepleymodule-whatis "PETSc - Portable, Extensible Toolkit for Scientific Computation"
244351d3a41SMatthew G Knepley
245dd486775SJed Brownset petsc_dir   "%s"
246dd486775SJed Brownset petsc_arch  "%s"
247351d3a41SMatthew G Knepley
248dd486775SJed Brownsetenv PETSC_ARCH "$petsc_arch"
249dd486775SJed Brownsetenv PETSC_DIR "$petsc_dir"
250dd486775SJed Brownprepend-path PATH "%s"
25155d606a3SSatish Balay''' % (self.petscdir.version, installdir, installarch, installpath))
252351d3a41SMatthew G Knepley    fd.close()
253351d3a41SMatthew G Knepley    return
254351d3a41SMatthew G Knepley
255f8833479SBarry Smith  def Dump(self):
256f8833479SBarry Smith    ''' Actually put the values into the configuration files '''
257f8833479SBarry Smith    # eventually everything between -- should be gone
25817f368bcSBarry Smith    if self.mpi.usingMPIUni:
25917f368bcSBarry Smith      #
26017f368bcSBarry Smith      # Remove any MPI/MPICH include files that may have been put here by previous runs of ./configure
2617908f030SMatthew 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)
26217f368bcSBarry Smith
263b5f71184SBarry Smith    self.logPrintDivider()
264b5f71184SBarry Smith    # Test for compiler-specific macros that need to be defined.
265b5f71184SBarry Smith    if self.setCompilers.isCrayVector('CC', self.log):
266b5f71184SBarry Smith      self.addDefine('HAVE_CRAY_VECTOR','1')
267b5f71184SBarry Smith
268b5f71184SBarry Smith    if self.functions.haveFunction('gethostbyname') and self.functions.haveFunction('socket') and self.headers.haveHeader('netinet/in.h'):
269b5f71184SBarry Smith      self.addDefine('USE_SOCKET_VIEWER','1')
270b5f71184SBarry Smith      if self.checkCompile('#include <sys/socket.h>','setsockopt(0,SOL_SOCKET,SO_REUSEADDR,0,0)'):
271b5f71184SBarry Smith        self.addDefine('HAVE_SO_REUSEADDR','1')
272b5f71184SBarry Smith
273b5f71184SBarry Smith    self.logPrintDivider()
2745f27b2e0SBarry Smith    self.setCompilers.pushLanguage('C')
2755f27b2e0SBarry Smith    compiler = self.setCompilers.getCompiler()
276217fe27eSSatish Balay    if [s for s in ['mpicc','mpiicc'] if os.path.basename(compiler).find(s)>=0]:
2775f27b2e0SBarry Smith      try:
2785f27b2e0SBarry Smith        output   = self.executeShellCommand(compiler + ' -show', log = self.log)[0]
2795f27b2e0SBarry Smith        compiler = output.split(' ')[0]
280f424265bSStefano Zampini        self.addDefine('MPICC_SHOW','"'+output.strip().replace('\n','\\\\n').replace('"','')+'"')
2815f27b2e0SBarry Smith      except:
282c9872b61SBarry Smith        self.addDefine('MPICC_SHOW','"Unavailable"')
283c9872b61SBarry Smith    else:
284c9872b61SBarry Smith      self.addDefine('MPICC_SHOW','"Unavailable"')
2855f27b2e0SBarry Smith    self.setCompilers.popLanguage()
286f8833479SBarry Smith#-----------------------------------------------------------------------------------------------------
287f8833479SBarry Smith
288f8833479SBarry Smith    # Sometimes we need C compiler, even if built with C++
289f8833479SBarry Smith    self.setCompilers.pushLanguage('C')
290e4c30378SBarry Smith    # do not use getCompilerFlags() because that automatically includes the CPPFLAGS so one ends up with duplication flags in makefile usage
291e4c30378SBarry Smith    self.addMakeMacro('CC_FLAGS',self.setCompilers.CFLAGS)
292f8833479SBarry Smith    self.setCompilers.popLanguage()
293f8833479SBarry Smith
29434f774f6SJed Brown    # And sometimes we need a C++ compiler even when PETSc is built with C
29534f774f6SJed Brown    if hasattr(self.compilers, 'CXX'):
29634f774f6SJed Brown      self.setCompilers.pushLanguage('Cxx')
29729921a8fSScott Kruger      self.addDefine('HAVE_CXX','1')
2980b119762SSatish Balay      self.addMakeMacro('CXXPP_FLAGS',self.setCompilers.CXXPPFLAGS)
299e4c30378SBarry Smith      # do not use getCompilerFlags() because that automatically includes the CXXPPFLAGS so one ends up with duplication flags in makefile usage
300e4c30378SBarry Smith      self.addMakeMacro('CXX_FLAGS',self.setCompilers.CXXFLAGS+' '+self.setCompilers.CXX_CXXFLAGS)
3012f4326f3SSatish Balay      cxx_linker = self.setCompilers.getLinker()
3022f4326f3SSatish Balay      self.addMakeMacro('CXX_LINKER',cxx_linker)
3032f4326f3SSatish Balay      self.addMakeMacro('CXX_LINKER_FLAGS',self.setCompilers.getLinkerFlags())
30434f774f6SJed Brown      self.setCompilers.popLanguage()
30505a757cfSSatish Balay    else:
30605a757cfSSatish Balay      self.addMakeMacro('CXX','')
30734f774f6SJed Brown
308f8833479SBarry Smith    # C preprocessor values
3091315f054SBarry Smith    self.addMakeMacro('CPP_FLAGS',self.setCompilers.CPPFLAGS)
310f8833479SBarry Smith
311f8833479SBarry Smith    # compiler values
312f8833479SBarry Smith    self.setCompilers.pushLanguage(self.languages.clanguage)
313f8833479SBarry Smith    self.addMakeMacro('PCC',self.setCompilers.getCompiler())
314e4c30378SBarry Smith    # do not use getCompilerFlags() because that automatically includes the preprocessor flags so one ends up with duplication flags in makefile usage
315e4c30378SBarry Smith    if self.languages.clanguage == 'C':
316e4c30378SBarry Smith      self.addMakeMacro('PCC_FLAGS','$(CC_FLAGS)')
317e4c30378SBarry Smith    else:
318e4c30378SBarry Smith      self.addMakeMacro('PCC_FLAGS','$(CXX_FLAGS)')
319f8833479SBarry Smith    self.setCompilers.popLanguage()
320f8833479SBarry Smith    # .o or .obj
321f8833479SBarry Smith    self.addMakeMacro('CC_SUFFIX','o')
322f8833479SBarry Smith
323f8833479SBarry Smith    # executable linker values
324f8833479SBarry Smith    self.setCompilers.pushLanguage(self.languages.clanguage)
325f8833479SBarry Smith    pcc_linker = self.setCompilers.getLinker()
326f8833479SBarry Smith    self.addMakeMacro('PCC_LINKER',pcc_linker)
327eacb1f64SJunchao Zhang    # We need to add sycl flags when linking petsc. See more in sycl.py.
328eacb1f64SJunchao Zhang    if hasattr(self.compilers, 'SYCLC'):
329eacb1f64SJunchao Zhang      self.addMakeMacro('PCC_LINKER_FLAGS',self.setCompilers.getLinkerFlags()+' '+self.setCompilers.SYCLFLAGS+' '+self.setCompilers.SYCLC_LINKER_FLAGS)
330eacb1f64SJunchao Zhang    else:
331c84a332bSSatish Balay      self.addMakeMacro('PCC_LINKER_FLAGS',self.setCompilers.getLinkerFlags())
332f8833479SBarry Smith    self.setCompilers.popLanguage()
333f8833479SBarry Smith    # '' for Unix, .exe for Windows
334f8833479SBarry Smith    self.addMakeMacro('CC_LINKER_SUFFIX','')
335f8833479SBarry Smith
336f8833479SBarry Smith    if hasattr(self.compilers, 'FC'):
337cb297985SSatish Balay      if self.framework.argDB['with-fortran-bindings']:
338257f4e5aSSatish Balay        if not self.fortran.fortranIsF90:
339ad49c6e3SMartin Diehl          raise RuntimeError('Error! Fortran compiler "'+self.compilers.FC+'" does not support F90! PETSc Fortran bindings require a F90 compiler')
340ad49c6e3SMartin Diehl        if not self.fortran.fortranBoolIsInteroperable:
341ad49c6e3SMartin Diehl          raise RuntimeError('Error! Fortran compiler "'+self.compilers.FC+'" does not have a C-interoperable Bool type! PETSc Fortran bindings require an interoperable Bool')
342fbf9dbe5SBarry Smith        self.addDefine('USE_FORTRAN_BINDINGS','1')
3438d35c829SSatish Balay        if not self.ftncmdline.have_command_argument:
3448d35c829SSatish Balay          raise RuntimeError('Error! Fortran compiler "'+self.compilers.FC+'" does not support F2003 GET_COMMAND_ARGUMENT()!')
3453384ab8aSJose E. Roman        if not self.fortran.fortranInitializePtrInDecl:
3463384ab8aSJose E. Roman          raise RuntimeError('Error! Fortran compiler "'+self.compilers.FC+'" does not support the F2018 standard allowing initializing pointers in the declaration!')
347f8833479SBarry Smith      self.setCompilers.pushLanguage('FC')
348f8833479SBarry Smith      # need FPPFLAGS in config/setCompilers
3490b119762SSatish Balay      self.addMakeMacro('FPP_FLAGS',self.setCompilers.FPPFLAGS)
350f8833479SBarry Smith
351f8833479SBarry Smith      # compiler values
352f8833479SBarry Smith      self.addMakeMacro('FC_FLAGS',self.setCompilers.getCompilerFlags())
353f8833479SBarry Smith      self.setCompilers.popLanguage()
354f8833479SBarry Smith      # .o or .obj
355f8833479SBarry Smith      self.addMakeMacro('FC_SUFFIX','o')
356f8833479SBarry Smith
357f8833479SBarry Smith      # executable linker values
358f8833479SBarry Smith      self.setCompilers.pushLanguage('FC')
359a9acdec7SBarry Smith      self.addMakeMacro('FC_LINKER',self.setCompilers.getLinker())
3606d53d35eSSatish Balay      self.addMakeMacro('FC_LINKER_FLAGS',self.setCompilers.getLinkerFlags())
361bb82cf9cSSatish Balay      self.setCompilers.popLanguage()
3625d631499SMatthew Knepley
3635d631499SMatthew Knepley      # F90 Modules
3645d631499SMatthew Knepley      if self.setCompilers.fortranModuleIncludeFlag:
3655d631499SMatthew Knepley        self.addMakeMacro('FC_MODULE_FLAG', self.setCompilers.fortranModuleIncludeFlag)
3666ddd6694SSatish Balay      else: # for non-f90 compilers like g77
3676ddd6694SSatish Balay        self.addMakeMacro('FC_MODULE_FLAG', '-I')
368a324c51cSMatthew G Knepley      if self.setCompilers.fortranModuleIncludeFlag:
369a324c51cSMatthew G Knepley        self.addMakeMacro('FC_MODULE_OUTPUT_FLAG', self.setCompilers.fortranModuleOutputFlag)
370f8833479SBarry Smith    else:
371f8833479SBarry Smith      self.addMakeMacro('FC','')
372f8833479SBarry Smith
37346a3958fSBarry Smith    if hasattr(self.compilers, 'CUDAC'):
3747ff2890cSSatish Balay      self.setCompilers.pushLanguage('CUDA')
375d93a25ecSSatish Balay      self.addMakeMacro('CUDAC_FLAGS',self.setCompilers.getCompilerFlags())
37650dcbc5aSJunchao Zhang      self.addMakeMacro('CUDAPP_FLAGS',self.setCompilers.CUDAPPFLAGS)
3777ff2890cSSatish Balay      self.setCompilers.popLanguage()
3787ff2890cSSatish Balay
379694a2f0eSJunchao Zhang    if hasattr(self.compilers, 'HIPC'):
38028f796eaSScott Kruger      self.setCompilers.pushLanguage('HIP')
381694a2f0eSJunchao Zhang      self.addMakeMacro('HIPC_FLAGS',self.setCompilers.getCompilerFlags())
3827fb1458fSStefano Zampini      self.addMakeMacro('HIPPP_FLAGS',self.setCompilers.HIPPPFLAGS)
38328f796eaSScott Kruger      self.setCompilers.popLanguage()
38428f796eaSScott Kruger
38550dcbc5aSJunchao Zhang    if hasattr(self.compilers, 'SYCLC'):
38628f796eaSScott Kruger      self.setCompilers.pushLanguage('SYCL')
38750dcbc5aSJunchao Zhang      self.addMakeMacro('SYCLC_FLAGS',self.setCompilers.getCompilerFlags())
388eacb1f64SJunchao Zhang      self.addMakeMacro('SYCLC_LINKER_FLAGS',self.setCompilers.getLinkerFlags())
38950dcbc5aSJunchao Zhang      self.addMakeMacro('SYCLPP_FLAGS',self.setCompilers.SYCLPPFLAGS)
39028f796eaSScott Kruger      self.setCompilers.popLanguage()
39128f796eaSScott Kruger
392f8833479SBarry Smith    # shared library linker values
393f8833479SBarry Smith    self.setCompilers.pushLanguage(self.languages.clanguage)
394f8833479SBarry Smith    # need to fix BuildSystem to collect these separately
395f8833479SBarry Smith    self.addMakeMacro('SL_LINKER',self.setCompilers.getLinker())
39670db8aa6SSatish Balay    self.addMakeMacro('SL_LINKER_FLAGS','${PCC_LINKER_FLAGS}')
397f8833479SBarry Smith    self.setCompilers.popLanguage()
398f8833479SBarry Smith    # One of 'a', 'so', 'lib', 'dll', 'dylib' (perhaps others also?) depending on the library generator and architecture
399f8833479SBarry Smith    # Note: . is not included in this macro, consistent with AR_LIB_SUFFIX
400f8833479SBarry Smith    if self.setCompilers.sharedLibraryExt == self.setCompilers.AR_LIB_SUFFIX:
401f8833479SBarry Smith      self.addMakeMacro('SL_LINKER_SUFFIX', '')
40246bc77b6SBarry Smith      self.addDefine('SLSUFFIX','""')
403f8833479SBarry Smith    else:
404f8833479SBarry Smith      self.addMakeMacro('SL_LINKER_SUFFIX', self.setCompilers.sharedLibraryExt)
40546bc77b6SBarry Smith      self.addDefine('SLSUFFIX','"'+self.setCompilers.sharedLibraryExt+'"')
406bb82cf9cSSatish Balay
40723e93537SBarry Smith    self.addMakeMacro('SL_LINKER_LIBS','${PETSC_EXTERNAL_LIB_BASIC}')
408bb82cf9cSSatish Balay
409f8833479SBarry Smith#-----------------------------------------------------------------------------------------------------
410f8833479SBarry Smith
411f8833479SBarry Smith    # CONLY or CPP. We should change the PETSc makefiles to do this better
412f8833479SBarry Smith    if self.languages.clanguage == 'C': lang = 'CONLY'
413f8833479SBarry Smith    else: lang = 'CXXONLY'
414f8833479SBarry Smith    self.addMakeMacro('PETSC_LANGUAGE',lang)
415f8833479SBarry Smith
416f8833479SBarry Smith    # real or complex
417f8833479SBarry Smith    self.addMakeMacro('PETSC_SCALAR',self.scalartypes.scalartype)
418f8833479SBarry Smith    # double or float
419f8833479SBarry Smith    self.addMakeMacro('PETSC_PRECISION',self.scalartypes.precision)
420f8833479SBarry Smith
421f8833479SBarry Smith    if self.framework.argDB['with-batch']:
422f8833479SBarry Smith      self.addMakeMacro('PETSC_WITH_BATCH','1')
423f8833479SBarry Smith
424f8833479SBarry Smith#-----------------------------------------------------------------------------------------------------
425a6cc6bb1SBarry Smith    # print include and lib for makefiles
426b5f71184SBarry Smith    self.logPrintDivider()
427f8833479SBarry Smith    self.framework.packages.reverse()
4285a21677cSJed Brown    petscincludes = [os.path.join(self.petscdir.dir,'include'),os.path.join(self.petscdir.dir,self.arch.arch,'include')]
4298749e224SJunchao Zhang    petscincludes_install = [os.path.join(self.installdir.dir, 'include')] if self.framework.argDB['prefix'] else petscincludes
4305a21677cSJed Brown    includes = []
431de8f682fSSatish Balay    self.packagelibs = []
432f8833479SBarry Smith    for i in self.framework.packages:
4337f0ff1afSBarry Smith      if not i.required:
4343972cb20SJacob Faibussowitsch        if i.devicePackage:
4353972cb20SJacob Faibussowitsch          self.addDefine('HAVE_DEVICE',1)
436c95305b8SPierre Jolivet          if self.languages.devicelanguage == 'C':
437c95305b8SPierre Jolivet            raise RuntimeError('Cannot use --with-devicelanguage=C with the device package ' + i.name)
438c95305b8SPierre Jolivet          self.languages.devicelanguage = 'Cxx'
439eeb16384SBarry Smith        self.addDefine('HAVE_'+i.PACKAGE.replace('-','_'), 1)  # ONLY list package if it is used directly by PETSc (and not only by another package)
440f8833479SBarry Smith      if not isinstance(i.lib, list):
441f8833479SBarry Smith        i.lib = [i.lib]
442de8f682fSSatish Balay      if i.linkedbypetsc: self.packagelibs.extend(i.lib)
443eeb16384SBarry Smith      self.addMakeMacro(i.PACKAGE.replace('-','_')+'_LIB', self.libraries.toStringNoDupes(i.lib))
444f8833479SBarry Smith      if hasattr(i,'include'):
445f8833479SBarry Smith        if not isinstance(i.include,list):
446f8833479SBarry Smith          i.include = [i.include]
447ac9e4c42SSatish Balay        includes.extend(i.include)
448eeb16384SBarry Smith        self.addMakeMacro(i.PACKAGE.replace('-','_')+'_INCLUDE',self.headers.toStringNoDupes(i.include))
449c95305b8SPierre Jolivet    if self.languages.devicelanguage == '': self.languages.devicelanguage = self.languages.clanguage
450c95305b8SPierre Jolivet    elif self.languages.devicelanguage == 'Cxx' and not hasattr(self.compilers, 'CXX'):
451c95305b8SPierre Jolivet      raise RuntimeError('Cannot use --with-devicelanguage=C++ without a valid C++ compiler')
452c95305b8SPierre Jolivet    if self.getMakeMacro('DEVICELANGUAGE') is None:
453c95305b8SPierre Jolivet      self.addDefine('DEVICELANGUAGE_'+self.languages.devicelanguage.upper(),'1')
454c95305b8SPierre Jolivet      self.addMakeMacro('DEVICELANGUAGE',self.languages.devicelanguage.upper())
455571416bbSBarry Smith
456de8f682fSSatish Balay    self.complibs = self.compilers.flibs+self.compilers.cxxlibs+self.compilers.LIBS.split()
457de8f682fSSatish Balay    self.PETSC_EXTERNAL_LIB_BASIC = self.libraries.toStringNoDupes(self.packagelibs+self.complibs)
458de8f682fSSatish Balay
459de8f682fSSatish Balay    self.addMakeMacro('PETSC_EXTERNAL_LIB_BASIC',self.PETSC_EXTERNAL_LIB_BASIC)
4605a21677cSJed Brown    allincludes = petscincludes + includes
4615a21677cSJed Brown    allincludes_install = petscincludes_install + includes
4625a21677cSJed Brown    self.PETSC_CC_INCLUDES = self.headers.toStringNoDupes(allincludes)
4635a21677cSJed Brown    self.PETSC_CC_INCLUDES_INSTALL = self.headers.toStringNoDupes(allincludes_install)
4645a21677cSJed Brown    self.addMakeMacro('PETSC_CC_INCLUDES',self.PETSC_CC_INCLUDES)
4655a21677cSJed Brown    self.addMakeMacro('PETSC_CC_INCLUDES_INSTALL', self.PETSC_CC_INCLUDES_INSTALL)
466cbd5cc15SBarry Smith    if hasattr(self.compilers, 'FC'):
4675a21677cSJed Brown      def modinc(includes):
4680542e31aSBarry Smith        return includes if self.fortran.fortranIsF90 else []
4695a21677cSJed Brown      self.addMakeMacro('PETSC_FC_INCLUDES',self.headers.toStringNoDupes(allincludes,modinc(allincludes)))
4705a21677cSJed Brown      self.addMakeMacro('PETSC_FC_INCLUDES_INSTALL',self.headers.toStringNoDupes(allincludes_install,modinc(allincludes_install)))
471f8833479SBarry Smith
472571416bbSBarry Smith    LIB_DIR = os.path.join(self.installdir.dir,'lib')
473571416bbSBarry Smith    self.addDefine('LIB_DIR','"'+LIB_DIR+'"')
4742d10696aSSatish Balay    # Use build dir here for 'make check' to work before 'make install'
4752d10696aSSatish Balay    PREINSTALL_LIB_DIR = os.path.join(self.petscdir.dir,self.arch.arch,'lib')
476f8833479SBarry Smith
4778143cb67SSatish Balay    self.LIB_NAME_SUFFIX = self.framework.argDB['with-library-name-suffix']
4788143cb67SSatish Balay    self.addMakeMacro('LIB_NAME_SUFFIX', self.LIB_NAME_SUFFIX)
479859f1752SFrancesco Ballarin    self.addDefine('LIB_NAME_SUFFIX', '"'+self.LIB_NAME_SUFFIX+'"')
4808143cb67SSatish Balay
4810f3b21c2SBarry Smith    if self.framework.argDB['with-single-library']:
4828143cb67SSatish Balay      self.petsclib = '-lpetsc'+self.LIB_NAME_SUFFIX
483571416bbSBarry Smith      self.addDefine('USE_SINGLE_LIBRARY', '1')
4848143cb67SSatish Balay      self.addMakeMacro('LIBNAME','${INSTALL_LIB_DIR}/libpetsc${LIB_NAME_SUFFIX}.${AR_LIB_SUFFIX}')
48557cb31baSSatish Balay      self.addMakeMacro('SHLIBS','libpetsc')
4868143cb67SSatish Balay      self.addMakeMacro('PETSC_WITH_EXTERNAL_LIB',self.libraries.toStringNoDupes(['-L'+PREINSTALL_LIB_DIR, '-lpetsc'+self.LIB_NAME_SUFFIX]+self.packagelibs+self.complibs))
487ea820d49SSatish Balay      self.addMakeMacro('PETSC_SYS_LIB','${PETSC_WITH_EXTERNAL_LIB}')
488ea820d49SSatish Balay      self.addMakeMacro('PETSC_VEC_LIB','${PETSC_WITH_EXTERNAL_LIB}')
489ea820d49SSatish Balay      self.addMakeMacro('PETSC_MAT_LIB','${PETSC_WITH_EXTERNAL_LIB}')
490ea820d49SSatish Balay      self.addMakeMacro('PETSC_DM_LIB','${PETSC_WITH_EXTERNAL_LIB}')
491ea820d49SSatish Balay      self.addMakeMacro('PETSC_KSP_LIB','${PETSC_WITH_EXTERNAL_LIB}')
492ea820d49SSatish Balay      self.addMakeMacro('PETSC_SNES_LIB','${PETSC_WITH_EXTERNAL_LIB}')
493ea820d49SSatish Balay      self.addMakeMacro('PETSC_TS_LIB','${PETSC_WITH_EXTERNAL_LIB}')
494b0a7d7e7SSatish Balay      self.addMakeMacro('PETSC_TAO_LIB','${PETSC_WITH_EXTERNAL_LIB}')
49534b254c5SRichard Tran Mills      self.addMakeMacro('PETSC_ML_LIB','${PETSC_WITH_EXTERNAL_LIB}')
496571416bbSBarry Smith    else:
49734b254c5SRichard Tran Mills      pkgs = ['ml', 'tao', 'ts', 'snes', 'ksp', 'dm', 'mat', 'vec', 'sys']
4988143cb67SSatish Balay      def liblist_basic(libs):
4998143cb67SSatish Balay        return [ '-lpetsc'+lib+self.LIB_NAME_SUFFIX for lib in libs]
5008143cb67SSatish Balay      def liblist(libs):
5018143cb67SSatish Balay        return self.libraries.toStringNoDupes(['-L'+PREINSTALL_LIB_DIR]+liblist_basic(libs)+self.packagelibs+self.complibs)
5028143cb67SSatish Balay      self.petsclib = ' '.join(liblist_basic(pkgs))
5038143cb67SSatish Balay      self.addMakeMacro('PETSC_SYS_LIB', liblist(pkgs[-1:]))
5048143cb67SSatish Balay      self.addMakeMacro('PETSC_VEC_LIB', liblist(pkgs[-2:]))
5058143cb67SSatish Balay      self.addMakeMacro('PETSC_MAT_LIB', liblist(pkgs[-3:]))
5068143cb67SSatish Balay      self.addMakeMacro('PETSC_DM_LIB',  liblist(pkgs[-4:]))
5078143cb67SSatish Balay      self.addMakeMacro('PETSC_KSP_LIB', liblist(pkgs[-5:]))
5088143cb67SSatish Balay      self.addMakeMacro('PETSC_SNES_LIB',liblist(pkgs[-6:]))
5098143cb67SSatish Balay      self.addMakeMacro('PETSC_TS_LIB',  liblist(pkgs[-7:]))
5108143cb67SSatish Balay      self.addMakeMacro('PETSC_TAO_LIB', liblist(pkgs[-8:]))
51134b254c5SRichard Tran Mills      self.addMakeMacro('PETSC_ML_LIB', liblist(pkgs[-9:]))
51234b254c5SRichard Tran Mills    self.addMakeMacro('PETSC_LIB','${PETSC_ML_LIB}')
513571416bbSBarry Smith    self.addMakeMacro('PETSC_LIB_BASIC',self.petsclib)
5140f3b21c2SBarry Smith
515f8833479SBarry Smith    if not os.path.exists(os.path.join(self.petscdir.dir,self.arch.arch,'lib')):
516f8833479SBarry Smith      os.makedirs(os.path.join(self.petscdir.dir,self.arch.arch,'lib'))
517f8833479SBarry Smith
518d03e3399SBarry Smith    # add a makefile entry for display
51913f9d092SSatish Balay    if self.framework.argDB['with-display']:
52013f9d092SSatish Balay      self.addMakeMacro('DISPLAY',self.framework.argDB['with-display'])
52113f9d092SSatish Balay
522f8833479SBarry Smith    # add a makefile entry for configure options
523f8833479SBarry Smith    self.addMakeMacro('CONFIGURE_OPTIONS', self.framework.getOptionsString(['configModules', 'optionsModule']).replace('\"','\\"'))
524a0c7f9aaSSamuel Khuvis
525a0c7f9aaSSamuel Khuvis    if self.framework.argDB['with-tau-perfstubs']:
526a0c7f9aaSSamuel Khuvis      self.addDefine('HAVE_TAU_PERFSTUBS',1)
527d03e3399SBarry Smith
528d03e3399SBarry Smith    if self.python.path:
529d03e3399SBarry Smith      if 'PYTHONPATH' in os.environ:
530d03e3399SBarry Smith        self.addMakeMacro('PETSCPYTHONPATH',':'.join(self.python.path)+':'+os.environ['PYTHONPATH'])
531d03e3399SBarry Smith      else:
532d03e3399SBarry Smith        self.addMakeMacro('PETSCPYTHONPATH',':'.join(self.python.path))
533f8833479SBarry Smith
534f8833479SBarry Smith  def dumpConfigInfo(self):
535f8833479SBarry Smith    import time
536c6ef1b5bSJed Brown    fd = open(os.path.join(self.arch.arch,'include','petscconfiginfo.h'),'w')
537dc25a686SPierre Jolivet    fd.write('static const char *petscconfigureoptions = "'+self.framework.getOptionsString(['configModules', 'optionsModule']).replace('\"','\\"').replace('\\ ','\\\\ ')+'";\n')
538f8833479SBarry Smith    fd.close()
539f8833479SBarry Smith    return
540f8833479SBarry Smith
5412a4161d9SMatthew G Knepley  def dumpMachineInfo(self):
5422a4161d9SMatthew G Knepley    import platform
543a970bd74SBernhard M. Wiedemann    import datetime
5442a4161d9SMatthew G Knepley    import time
54540373944SSatish Balay    import script
546ca77dbeeSGeoffrey Irving    def escape(s):
547e08ecd42SSatish Balay      return s.replace('"',r'\"').replace(r'\ ',r'\\ ') # novermin
548c6ef1b5bSJed Brown    fd = open(os.path.join(self.arch.arch,'include','petscmachineinfo.h'),'w')
5492a4161d9SMatthew G Knepley    fd.write('static const char *petscmachineinfo = \"\\n\"\n')
5502a4161d9SMatthew G Knepley    fd.write('\"-----------------------------------------\\n\"\n')
551a970bd74SBernhard M. Wiedemann    buildhost = platform.node()
552a970bd74SBernhard M. Wiedemann    if os.environ.get('SOURCE_DATE_EPOCH'):
553a970bd74SBernhard M. Wiedemann      buildhost = "reproducible"
554a970bd74SBernhard M. Wiedemann    buildtime = datetime.datetime.utcfromtimestamp(int(os.environ.get('SOURCE_DATE_EPOCH', time.time())))
555a970bd74SBernhard M. Wiedemann    fd.write('\"Libraries compiled on %s on %s\\n\"\n' % (buildtime, buildhost))
55660acdfe7SSatish Balay    fd.write('\"Machine characteristics: %s\\n\"\n' % (platform.platform()))
5575188cb68SSatish Balay    fd.write('\"Using PETSc directory: %s\\n\"\n' % (escape(self.installdir.petscDir)))
5585188cb68SSatish Balay    fd.write('\"Using PETSc arch: %s\\n\"\n' % (escape(self.installdir.petscArch)))
559cdec380aSBarry Smith    fd.write('\"-----------------------------------------\\n\";\n')
5602a4161d9SMatthew G Knepley    fd.write('static const char *petsccompilerinfo = \"\\n\"\n')
5612a4161d9SMatthew G Knepley    self.setCompilers.pushLanguage(self.languages.clanguage)
5625f27b2e0SBarry Smith    fd.write('\"Using C compiler: %s %s\\n\"\n' % (escape(self.setCompilers.getCompiler()), escape(self.setCompilers.getCompilerFlags())))
5632a4161d9SMatthew G Knepley    self.setCompilers.popLanguage()
5648782282cSMatthew G Knepley    if hasattr(self.compilers, 'FC'):
5652a4161d9SMatthew G Knepley      self.setCompilers.pushLanguage('FC')
5665f27b2e0SBarry Smith      fd.write('\"Using Fortran compiler: %s %s %s\\n\"\n' % (escape(self.setCompilers.getCompiler()), escape(self.setCompilers.getCompilerFlags()), escape(self.setCompilers.CPPFLAGS)))
5672a4161d9SMatthew G Knepley      self.setCompilers.popLanguage()
568cdec380aSBarry Smith    fd.write('\"-----------------------------------------\\n\";\n')
5692a4161d9SMatthew G Knepley    fd.write('static const char *petsccompilerflagsinfo = \"\\n\"\n')
5705a21677cSJed Brown    fd.write('\"Using include paths: %s\\n\"\n' % (escape(self.PETSC_CC_INCLUDES_INSTALL.replace('${PETSC_DIR}', self.installdir.petscDir))))
571cdec380aSBarry Smith    fd.write('\"-----------------------------------------\\n\";\n')
5722a4161d9SMatthew G Knepley    fd.write('static const char *petsclinkerinfo = \"\\n\"\n')
5732a4161d9SMatthew G Knepley    self.setCompilers.pushLanguage(self.languages.clanguage)
574ca77dbeeSGeoffrey Irving    fd.write('\"Using C linker: %s\\n\"\n' % (escape(self.setCompilers.getLinker())))
5752a4161d9SMatthew G Knepley    self.setCompilers.popLanguage()
5768782282cSMatthew G Knepley    if hasattr(self.compilers, 'FC'):
5772a4161d9SMatthew G Knepley      self.setCompilers.pushLanguage('FC')
578ca77dbeeSGeoffrey Irving      fd.write('\"Using Fortran linker: %s\\n\"\n' % (escape(self.setCompilers.getLinker())))
5792a4161d9SMatthew G Knepley      self.setCompilers.popLanguage()
5805188cb68SSatish 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)))
581cdec380aSBarry Smith    fd.write('\"-----------------------------------------\\n\";\n')
5822a4161d9SMatthew G Knepley    fd.close()
5832a4161d9SMatthew G Knepley    return
584b2843cf1SBarry Smith
585b2843cf1SBarry Smith  def configurePrefetch(self):
586b2843cf1SBarry Smith    '''Sees if there are any prefetch functions supported'''
5877fca349cSMatthew G. Knepley    if config.setCompilers.Configure.isSolaris(self.log) or self.framework.argDB['with-ios'] or not self.framework.argDB['with-prefetch']:
58893f78423SSatish Balay      self.addDefine('Prefetch(a,b,c)', ' ')
58993f78423SSatish Balay      return
590ec284106SBarry Smith    self.pushLanguage(self.languages.clanguage)
59110699583SJed Brown    if self.checkLink('#include <xmmintrin.h>', 'void *v = 0;_mm_prefetch((const char*)v,_MM_HINT_NTA);\n'):
59250d8bf02SJed Brown      # The Intel Intrinsics manual [1] specifies the prototype
59350d8bf02SJed Brown      #
59450d8bf02SJed Brown      #   void _mm_prefetch(char const *a, int sel);
59550d8bf02SJed Brown      #
59650d8bf02SJed Brown      # but other vendors seem to insist on using subtly different
59750d8bf02SJed Brown      # prototypes, including void* for the pointer, and an enum for
59850d8bf02SJed Brown      # sel.  These are both reasonable changes, but negatively impact
59950d8bf02SJed Brown      # portability.
60050d8bf02SJed Brown      #
601a8d69d7bSBarry Smith      # [1] https://software.intel.com/file/6373
60250d8bf02SJed Brown      self.addDefine('HAVE_XMMINTRIN_H', 1)
60350d8bf02SJed Brown      self.addDefine('Prefetch(a,b,c)', '_mm_prefetch((const char*)(a),(c))')
60450d8bf02SJed Brown      self.addDefine('PREFETCH_HINT_NTA', '_MM_HINT_NTA')
60550d8bf02SJed Brown      self.addDefine('PREFETCH_HINT_T0',  '_MM_HINT_T0')
60650d8bf02SJed Brown      self.addDefine('PREFETCH_HINT_T1',  '_MM_HINT_T1')
60750d8bf02SJed Brown      self.addDefine('PREFETCH_HINT_T2',  '_MM_HINT_T2')
60850d8bf02SJed Brown    elif self.checkLink('#include <xmmintrin.h>', 'void *v = 0;_mm_prefetch(v,_MM_HINT_NTA);\n'):
60950d8bf02SJed Brown      self.addDefine('HAVE_XMMINTRIN_H', 1)
61050d8bf02SJed Brown      self.addDefine('Prefetch(a,b,c)', '_mm_prefetch((const void*)(a),(c))')
61150d8bf02SJed Brown      self.addDefine('PREFETCH_HINT_NTA', '_MM_HINT_NTA')
61250d8bf02SJed Brown      self.addDefine('PREFETCH_HINT_T0',  '_MM_HINT_T0')
61350d8bf02SJed Brown      self.addDefine('PREFETCH_HINT_T1',  '_MM_HINT_T1')
61450d8bf02SJed Brown      self.addDefine('PREFETCH_HINT_T2',  '_MM_HINT_T2')
61510699583SJed Brown    elif self.checkLink('', 'void *v = 0;__builtin_prefetch(v,0,0);\n'):
61610699583SJed Brown      # From GCC docs: void __builtin_prefetch(const void *addr,int rw,int locality)
61710699583SJed Brown      #
61810699583SJed Brown      #   The value of rw is a compile-time constant one or zero; one
61910699583SJed Brown      #   means that the prefetch is preparing for a write to the memory
62010699583SJed Brown      #   address and zero, the default, means that the prefetch is
62110699583SJed Brown      #   preparing for a read. The value locality must be a compile-time
62210699583SJed Brown      #   constant integer between zero and three. A value of zero means
62310699583SJed Brown      #   that the data has no temporal locality, so it need not be left
62410699583SJed Brown      #   in the cache after the access. A value of three means that the
62510699583SJed Brown      #   data has a high degree of temporal locality and should be left
62610699583SJed Brown      #   in all levels of cache possible. Values of one and two mean,
62710699583SJed Brown      #   respectively, a low or moderate degree of temporal locality.
62810699583SJed Brown      #
62910699583SJed Brown      # Here we adopt Intel's x86/x86-64 naming scheme for the locality
63010699583SJed Brown      # hints.  Using macros for these values in necessary since some
63110699583SJed Brown      # compilers require an enum.
63210699583SJed Brown      self.addDefine('Prefetch(a,b,c)', '__builtin_prefetch((a),(b),(c))')
63310699583SJed Brown      self.addDefine('PREFETCH_HINT_NTA', '0')
63410699583SJed Brown      self.addDefine('PREFETCH_HINT_T0',  '3')
63510699583SJed Brown      self.addDefine('PREFETCH_HINT_T1',  '2')
63610699583SJed Brown      self.addDefine('PREFETCH_HINT_T2',  '1')
637b2843cf1SBarry Smith    else:
638b2843cf1SBarry Smith      self.addDefine('Prefetch(a,b,c)', ' ')
6397d490b44SBarry Smith    self.popLanguage()
640b2843cf1SBarry Smith
64149fe22e6SSatish Balay  def delGenFiles(self):
64249fe22e6SSatish Balay    '''Delete generated files'''
64349fe22e6SSatish Balay    delfile = os.path.join(self.arch.arch,'lib','petsc','conf','files')
64449fe22e6SSatish Balay    try:
64549fe22e6SSatish Balay      os.unlink(delfile)
64649fe22e6SSatish Balay    except: pass
64749fe22e6SSatish Balay
64809bc878fSSatish Balay  def configureAtoll(self):
64909bc878fSSatish Balay    '''Checks if atoll exists'''
65022164b4cSPierre 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'):
65109bc878fSSatish Balay       self.addDefine('HAVE_ATOLL', '1')
65209bc878fSSatish Balay
653648c30bcSBarry Smith  def configureSanitize(self):
654648c30bcSBarry Smith    '''Checks if fsanitize is supported'''
655648c30bcSBarry Smith    if self.checkLink('#if defined(__has_feature)\n#if !__has_feature(address_sanitizer)\nGarbage\n#endif\n#else\nGarbage\n#endif\n'):
656648c30bcSBarry Smith      self.addDefine('HAVE_SANITIZER', '1')
6570ef292d3SStefano Zampini    elif self.checkLink('#if !defined(__SANITIZE_ADDRESS__)\nGarbage\n#endif\n'):
6580ef292d3SStefano Zampini      self.addDefine('HAVE_SANITIZER', '1')
659648c30bcSBarry Smith
6602400fdedSBarry Smith  def configureUnused(self):
6612400fdedSBarry Smith    '''Sees if __attribute((unused)) is supported'''
6621adaff47SSean Farley    if self.framework.argDB['with-ios']:
6632400fdedSBarry Smith      self.addDefine('UNUSED', ' ')
6642400fdedSBarry Smith      return
6652400fdedSBarry Smith    self.pushLanguage(self.languages.clanguage)
66622164b4cSPierre 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'):
6672400fdedSBarry Smith      self.addDefine('UNUSED', '__attribute((unused))')
6682400fdedSBarry Smith    else:
6692400fdedSBarry Smith      self.addDefine('UNUSED', ' ')
6702400fdedSBarry Smith    self.popLanguage()
6712400fdedSBarry Smith
67298ed35c3SBarry Smith  def configureIsatty(self):
67398ed35c3SBarry Smith    '''Check if the Unix C function isatty() works correctly
67498ed35c3SBarry Smith       Actually just assumes it does not work correctly on batch systems'''
67598ed35c3SBarry Smith    if not self.framework.argDB['with-batch']:
67698ed35c3SBarry Smith      self.addDefine('USE_ISATTY',1)
67798ed35c3SBarry Smith
6781ef8df7fSJed Brown  def configureDeprecated(self):
6791ef8df7fSJed Brown    '''Check if __attribute((deprecated)) is supported'''
6807efe37a1SJacob Faibussowitsch    def checkDeprecated(macro_base, src, is_intel):
6817efe37a1SJacob Faibussowitsch      '''
6827efe37a1SJacob Faibussowitsch      run through the various attribute deprecated combinations and define MACRO_BAS(why) to the result
6837efe37a1SJacob Faibussowitsch      it if it compiles.
6847efe37a1SJacob Faibussowitsch
6857efe37a1SJacob Faibussowitsch      If none of the combos work, defines MACRO_BASE(why) as empty
6867efe37a1SJacob Faibussowitsch      '''
6875029be03SJacob Faibussowitsch      full_macro_name = macro_base + '(string_literal_why)'
6887efe37a1SJacob Faibussowitsch      for prefix in ('__attribute__', '__attribute','__declspec'):
6897efe37a1SJacob Faibussowitsch        if prefix == '__declspec':
6907efe37a1SJacob Faibussowitsch          # declspec does not have an extra set of brackets around the arguments
6915029be03SJacob Faibussowitsch          attr_bodies = ('deprecated(string_literal_why)', 'deprecated')
6921ef8df7fSJed Brown        else:
6935029be03SJacob Faibussowitsch          attr_bodies = ('(deprecated(string_literal_why))', '(deprecated)')
6947efe37a1SJacob Faibussowitsch
6957efe37a1SJacob Faibussowitsch        for attr_body in attr_bodies:
6967efe37a1SJacob Faibussowitsch          attr_def = '{}({})'.format(prefix, attr_body)
6977efe37a1SJacob Faibussowitsch          test_src = '\n'.join((
6987efe37a1SJacob Faibussowitsch            '#define {} {}'.format(full_macro_name, attr_def),
6997efe37a1SJacob Faibussowitsch            src.format(macro_base + '("asdasdadsasd")')
7007efe37a1SJacob Faibussowitsch          ))
7017efe37a1SJacob Faibussowitsch          if self.checkCompile(test_src, ''):
7027efe37a1SJacob Faibussowitsch            self.logPrint('configureDeprecated: \'{}\' appears to work'.format(attr_def))
7037efe37a1SJacob Faibussowitsch            if is_intel and '(why)' in attr_body:
7047efe37a1SJacob 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))
7057efe37a1SJacob Faibussowitsch              self.logPrint('*** WE WILL THEREFORE REJECT \'{}\' AND CONTINUE TESTING ***'.format(attr_def))
7067efe37a1SJacob Faibussowitsch              continue
7077efe37a1SJacob Faibussowitsch            self.addDefine(full_macro_name, attr_def)
7087efe37a1SJacob Faibussowitsch            return
7097efe37a1SJacob Faibussowitsch
7107efe37a1SJacob Faibussowitsch      self.addDefine(full_macro_name, ' ')
7117efe37a1SJacob Faibussowitsch      return
7127efe37a1SJacob Faibussowitsch
7137efe37a1SJacob Faibussowitsch    lang = self.languages.clanguage
7147efe37a1SJacob Faibussowitsch    with self.Language(lang):
7157efe37a1SJacob Faibussowitsch      is_intel = self.setCompilers.isIntel(self.getCompiler(lang=lang), self.log)
7165029be03SJacob Faibussowitsch      checkDeprecated('DEPRECATED_FUNCTION_BASE', '{} int myfunc(void) {{ return 1; }}', is_intel)
7175029be03SJacob Faibussowitsch      checkDeprecated('DEPRECATED_TYPEDEF_BASE', 'typedef int my_int {};', is_intel)
7185029be03SJacob Faibussowitsch      checkDeprecated('DEPRECATED_ENUM_BASE', 'enum E {{ oldval {}, newval }};', is_intel)
719a14d4ff0SToby Isaac      checkDeprecated('DEPRECATED_OBJECT_BASE', '{} int x;', is_intel)
7207efe37a1SJacob Faibussowitsch      # I was unable to make a CPP macro that takes the old and new values as separate
7217efe37a1SJacob Faibussowitsch      # arguments and builds the message needed by _Pragma hence the deprecation message is
7227efe37a1SJacob Faibussowitsch      # handled as it is
72305de396fSBarry Smith      if self.checkCompile('#define TEST _Pragma("GCC warning \"Testing _Pragma\"") value'):
7245029be03SJacob Faibussowitsch        self.addDefine('DEPRECATED_MACRO_BASE_(why)', '_Pragma(#why)')
7255029be03SJacob Faibussowitsch        self.addDefine('DEPRECATED_MACRO_BASE(string_literal_why)', self.substPrefix + '_DEPRECATED_MACRO_BASE_(GCC warning string_literal_why)')
72605de396fSBarry Smith      else:
7275029be03SJacob Faibussowitsch        self.addDefine('DEPRECATED_MACRO_BASE(why)', ' ')
7281ef8df7fSJed Brown
72918f41590SBarry Smith  def configureAlign(self):
7307b7fc14bSLisandro Dalcin    '''Check if __attribute(aligned) is supported'''
7317b7fc14bSLisandro Dalcin    code = '''\
732752d89a4SSatish Balaystruct mystruct {int myint;} __attribute((aligned(16)));
7337b7fc14bSLisandro Dalcinchar assert_aligned[(sizeof(struct mystruct)==16)*2-1];
734752d89a4SSatish Balay'''
735752d89a4SSatish Balay    self.pushLanguage(self.languages.clanguage)
7367b7fc14bSLisandro Dalcin    if self.checkCompile(code):
737752d89a4SSatish Balay      self.addDefine('ATTRIBUTEALIGNED(size)', '__attribute((aligned(size)))')
738752d89a4SSatish Balay      self.addDefine('HAVE_ATTRIBUTEALIGNED', 1)
739752d89a4SSatish Balay    else:
7407b7fc14bSLisandro Dalcin      self.framework.logPrint('Incorrect attribute(aligned)')
741752d89a4SSatish Balay      self.addDefine('ATTRIBUTEALIGNED(size)', ' ')
7427b7fc14bSLisandro Dalcin    self.popLanguage()
743752d89a4SSatish Balay    return
74418f41590SBarry Smith
7459800092aSJed Brown  def configureExpect(self):
7469800092aSJed Brown    '''Sees if the __builtin_expect directive is supported'''
7479800092aSJed Brown    self.pushLanguage(self.languages.clanguage)
7489800092aSJed Brown    if self.checkLink('', 'if (__builtin_expect(0,1)) return 1;'):
7499800092aSJed Brown      self.addDefine('HAVE_BUILTIN_EXPECT', 1)
7509800092aSJed Brown    self.popLanguage()
7519800092aSJed Brown
75253c77d0aSJed Brown  def configureFunctionName(self):
753fbfcfee5SBarry Smith    '''Sees if the compiler supports __func__ or a variant.'''
7541ec50b02SJed Brown    def getFunctionName(lang):
755fbfcfee5SBarry Smith      name = '"unknown"'
7561ec50b02SJed Brown      self.pushLanguage(lang)
757b6ff4c76SKarl Rupp      for fname in ['__func__','__FUNCTION__','__extension__ __func__']:
7580117e5a1SSatish Balay        code = "if ("+fname+"[0] != 'm') return 1;"
7590117e5a1SSatish Balay        if self.checkCompile('',code) and self.checkLink('',code):
7600117e5a1SSatish Balay          name = fname
7610117e5a1SSatish Balay          break
7621ec50b02SJed Brown      self.popLanguage()
7631ec50b02SJed Brown      return name
7641ec50b02SJed Brown    langs = []
765628773c9SSatish Balay
766628773c9SSatish Balay    self.addDefine('FUNCTION_NAME_C', getFunctionName('C'))
7675f6e5f85SSatish Balay    if hasattr(self.compilers, 'CXX'):
768628773c9SSatish Balay      self.addDefine('FUNCTION_NAME_CXX', getFunctionName('Cxx'))
76953c77d0aSJed Brown
770753ebd1dSJed Brown  def configureIntptrt(self):
771f18a5f7eSJacob Faibussowitsch    '''Determine what to use for uintptr_t and intptr_t'''
772753ebd1dSJed Brown    def staticAssertSizeMatchesVoidStar(inc,typename):
773753ebd1dSJed Brown      # The declaration is an error if either array size is negative.
774753ebd1dSJed 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
775d26187a0SJed Brown      return self.checkCompile(inc, ('#define STATIC_ASSERT(cond) char negative_length_if_false[2*(!!(cond))-1]\n'
776979939cdSSatish Balay                                     + 'STATIC_ASSERT(sizeof(void*) == sizeof(%s));'%typename))
777f18a5f7eSJacob Faibussowitsch
778f18a5f7eSJacob Faibussowitsch    def generate_uintptr_guesses():
779f18a5f7eSJacob Faibussowitsch      for suff in ('max', '64', '32', '16'):
780f18a5f7eSJacob Faibussowitsch        yield '#include <stdint.h>', 'uint{}_t'.format(suff), 'PRIx{}'.format(suff.upper())
781f18a5f7eSJacob Faibussowitsch      yield '#include <stdlib.h>\n#include <string.h>', 'size_t', 'zx'
782f18a5f7eSJacob Faibussowitsch      yield '', 'unsigned long long', 'llx'
783f18a5f7eSJacob Faibussowitsch      yield '', 'unsigned long', 'lx'
784f18a5f7eSJacob Faibussowitsch      yield '', 'unsigned', 'x'
785f18a5f7eSJacob Faibussowitsch
786f18a5f7eSJacob Faibussowitsch    def generate_intptr_guesses():
787f18a5f7eSJacob Faibussowitsch      for suff in ('max', '64', '32', '16'):
788f18a5f7eSJacob Faibussowitsch        yield '#include <stdint.h>', 'int{}_t'.format(suff), 'PRIx{}'.format(suff.upper())
789f18a5f7eSJacob Faibussowitsch      yield '', 'long long', 'llx'
790f18a5f7eSJacob Faibussowitsch      yield '', 'long', 'lx'
791f18a5f7eSJacob Faibussowitsch      yield '', 'int', 'x'
792f18a5f7eSJacob Faibussowitsch
793f18a5f7eSJacob Faibussowitsch    def check(default_typename, generator):
794f18a5f7eSJacob Faibussowitsch      macro_name = default_typename.upper()
795f18a5f7eSJacob Faibussowitsch      with self.Language(self.languages.clanguage):
796f18a5f7eSJacob Faibussowitsch        if self.checkCompile(
797f18a5f7eSJacob Faibussowitsch            '#include <stdint.h>',
798f18a5f7eSJacob Faibussowitsch            'int x; {type_name} i = ({type_name})&x; (void)i'.format(type_name=default_typename)
799f18a5f7eSJacob Faibussowitsch        ):
800f18a5f7eSJacob Faibussowitsch          typename     = default_typename
801f18a5f7eSJacob Faibussowitsch          print_format = 'PRIxPTR'
802d26187a0SJed Brown        else:
803f18a5f7eSJacob Faibussowitsch          for include, typename, print_format in generator():
804f18a5f7eSJacob Faibussowitsch            if staticAssertSizeMatchesVoidStar(include, typename):
805f18a5f7eSJacob Faibussowitsch              break
806f18a5f7eSJacob Faibussowitsch          else:
807f18a5f7eSJacob Faibussowitsch            raise RuntimeError('Could not find any {} type matching void*'.format(macro_name))
808f18a5f7eSJacob Faibussowitsch      self.addDefine(macro_name         , typename)
809f18a5f7eSJacob Faibussowitsch      self.addDefine(macro_name + '_FMT', '\"#\" ' + print_format)
810f18a5f7eSJacob Faibussowitsch      return
811f18a5f7eSJacob Faibussowitsch
812f18a5f7eSJacob Faibussowitsch    check('uintptr_t', generate_uintptr_guesses)
813f18a5f7eSJacob Faibussowitsch    check('intptr_t', generate_intptr_guesses)
814f18a5f7eSJacob Faibussowitsch    return
815753ebd1dSJed Brown
816ed938b00SJed Brown  def configureRTLDDefault(self):
8177b65ca21SBarry Smith    '''Check for dynamic library feature'''
818bfef2c86SBarry Smith    if self.checkCompile('#include <dlfcn.h>\n void *ptr =  RTLD_DEFAULT;'):
8199fb7294dSPierre Jolivet      self.addDefine('HAVE_RTLD_DEFAULT','1')
820f8833479SBarry Smith    return
821f8833479SBarry Smith
822f8833479SBarry Smith  def configureSolaris(self):
823f8833479SBarry Smith    '''Solaris specific stuff'''
824f8833479SBarry Smith    if os.path.isdir(os.path.join('/usr','ucblib')):
825f8833479SBarry Smith      try:
826f8833479SBarry Smith        flag = getattr(self.setCompilers, self.language[-1]+'SharedLinkerFlag')
827f8833479SBarry Smith      except AttributeError:
828f8833479SBarry Smith        flag = None
829f8833479SBarry Smith      if flag is None:
830f8833479SBarry Smith        self.compilers.LIBS += ' -L/usr/ucblib'
831f8833479SBarry Smith      else:
832f8833479SBarry Smith        self.compilers.LIBS += ' '+flag+'/usr/ucblib'
833f8833479SBarry Smith    return
834f8833479SBarry Smith
8350f64ec89SBarry Smith  def configureDarwin(self):
8360f64ec89SBarry Smith    '''Log brew configuration for Apple systems'''
8370f64ec89SBarry Smith    try:
8380f64ec89SBarry Smith      self.executeShellCommand(['brew', 'config'], log = self.log)
8390f64ec89SBarry Smith      self.executeShellCommand(['brew', 'info', 'gcc'], log = self.log)
8400f64ec89SBarry Smith    except:
8410f64ec89SBarry Smith      pass
8420f64ec89SBarry Smith    return
8430f64ec89SBarry Smith
844f8833479SBarry Smith  def configureLinux(self):
845f8833479SBarry Smith    '''Linux specific stuff'''
8469f15855cSMatthew G Knepley    # TODO: Test for this by mallocing an odd number of floats and checking the address
847f8833479SBarry Smith    self.addDefine('HAVE_DOUBLE_ALIGN_MALLOC', 1)
848f8833479SBarry Smith    return
849f8833479SBarry Smith
850f8833479SBarry Smith  def configureWin32(self):
851f8833479SBarry Smith    '''Win32 non-cygwin specific stuff'''
852f8833479SBarry Smith    kernel32=0
8534e8afd12SMosè Giordano    if self.libraries.add('Kernel32.lib','GetComputerName',prototype='#include <windows.h>', call='GetComputerName(NULL,NULL);'):
854f8833479SBarry Smith      self.addDefine('HAVE_WINDOWS_H',1)
855f8833479SBarry Smith      self.addDefine('HAVE_GETCOMPUTERNAME',1)
856f8833479SBarry Smith      kernel32=1
8574e8afd12SMosè Giordano    elif self.libraries.add('kernel32','GetComputerName',prototype='#include <windows.h>', call='GetComputerName(NULL,NULL);'):
858f8833479SBarry Smith      self.addDefine('HAVE_WINDOWS_H',1)
859f8833479SBarry Smith      self.addDefine('HAVE_GETCOMPUTERNAME',1)
860f8833479SBarry Smith      kernel32=1
861f8833479SBarry Smith    if kernel32:
862eed94e11SSatish Balay      if self.framework.argDB['with-windows-graphics']:
863eed94e11SSatish Balay        self.addDefine('USE_WINDOWS_GRAPHICS',1)
8644e8afd12SMosè Giordano      if self.checkLink('#include <windows.h>','LoadLibrary(0)'):
865f8833479SBarry Smith        self.addDefine('HAVE_LOADLIBRARY',1)
8664e8afd12SMosè Giordano      if self.checkLink('#include <windows.h>','GetProcAddress(0,0)'):
867b50f6d9eSLisandro Dalcin        self.addDefine('HAVE_GETPROCADDRESS',1)
8684e8afd12SMosè Giordano      if self.checkLink('#include <windows.h>','FreeLibrary(0)'):
869b50f6d9eSLisandro Dalcin        self.addDefine('HAVE_FREELIBRARY',1)
8704e8afd12SMosè Giordano      if self.checkLink('#include <windows.h>','GetLastError()'):
871a21658a3SLisandro Dalcin        self.addDefine('HAVE_GETLASTERROR',1)
8724e8afd12SMosè Giordano      if self.checkLink('#include <windows.h>','SetLastError(0)'):
873a21658a3SLisandro Dalcin        self.addDefine('HAVE_SETLASTERROR',1)
8744e8afd12SMosè Giordano      if self.checkLink('#include <windows.h>\n','QueryPerformanceCounter(0);\n'):
875bea725cfSBarry Smith        self.addDefine('USE_MICROSOFT_TIME',1)
8764e8afd12SMosè Giordano    if self.libraries.add('Advapi32.lib','GetUserName',prototype='#include <windows.h>', call='GetUserName(NULL,NULL);'):
877f8833479SBarry Smith      self.addDefine('HAVE_GET_USER_NAME',1)
8784e8afd12SMosè Giordano    elif self.libraries.add('advapi32','GetUserName',prototype='#include <windows.h>', call='GetUserName(NULL,NULL);'):
879f8833479SBarry Smith      self.addDefine('HAVE_GET_USER_NAME',1)
880f8833479SBarry Smith
8814e8afd12SMosè Giordano    if not self.libraries.add('User32.lib','GetDC',prototype='#include <windows.h>',call='GetDC(0);'):
8824e8afd12SMosè Giordano      self.libraries.add('user32','GetDC',prototype='#include <windows.h>',call='GetDC(0);')
8834e8afd12SMosè Giordano    if not self.libraries.add('Gdi32.lib','CreateCompatibleDC',prototype='#include <windows.h>',call='CreateCompatibleDC(0);'):
8844e8afd12SMosè Giordano      self.libraries.add('gdi32','CreateCompatibleDC',prototype='#include <windows.h>',call='CreateCompatibleDC(0);')
885f8833479SBarry Smith
88622164b4cSPierre Jolivet    if not self.checkCompile('#include <sys/types.h>\n','uid_t u;\n(void)u'):
887f8833479SBarry Smith      self.addTypedef('int', 'uid_t')
888f8833479SBarry Smith      self.addTypedef('int', 'gid_t')
88922164b4cSPierre 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'):
890f8833479SBarry Smith      self.framework.addDefine('R_OK', '04')
891f8833479SBarry Smith      self.framework.addDefine('W_OK', '02')
892f8833479SBarry Smith      self.framework.addDefine('X_OK', '01')
893f8833479SBarry Smith    if not self.checkLink('#include <sys/stat.h>\n','int a=0;\nif (S_ISDIR(a)){}\n'):
894f8833479SBarry Smith      self.framework.addDefine('S_ISREG(a)', '(((a)&_S_IFMT) == _S_IFREG)')
895f8833479SBarry Smith      self.framework.addDefine('S_ISDIR(a)', '(((a)&_S_IFMT) == _S_IFDIR)')
8964e8afd12SMosè Giordano    if self.checkCompile('#include <windows.h>\n','LARGE_INTEGER a;\nDWORD b=a.u.HighPart;\n'):
897f8833479SBarry Smith      self.addDefine('HAVE_LARGE_INTEGER_U',1)
898f8833479SBarry Smith
899f8833479SBarry Smith    # Windows requires a Binary file creation flag when creating/opening binary files.  Is a better test in order?
9004e8afd12SMosè Giordano    if self.checkCompile('#include <windows.h>\n#include <fcntl.h>\n', 'int flags = O_BINARY;'):
901f8833479SBarry Smith      self.addDefine('HAVE_O_BINARY',1)
902f8833479SBarry Smith
903f8833479SBarry Smith    if self.compilers.CC.find('win32fe') >= 0:
904ad4212abSSatish Balay      self.addDefine('HAVE_WINDOWS_COMPILERS',1)
905f8833479SBarry Smith      self.addDefine('DIR_SEPARATOR','\'\\\\\'')
906f8833479SBarry Smith      self.addDefine('REPLACE_DIR_SEPARATOR','\'/\'')
907f8833479SBarry Smith      self.addDefine('CANNOT_START_DEBUGGER',1)
9085188cb68SSatish Balay      (petscdir,error,status) = self.executeShellCommand('cygpath -w '+self.installdir.petscDir, log = self.log)
90934531a4dSSatish Balay      self.addDefine('DIR','"'+petscdir.replace('\\','\\\\')+'"')
9105188cb68SSatish Balay      (petscdir,error,status) = self.executeShellCommand('cygpath -m '+self.installdir.petscDir, log = self.log)
911e433681fSSatish Balay      self.addMakeMacro('wPETSC_DIR',petscdir)
9124e00a515SSatish Balay      if self.dataFilesPath.datafilespath:
9134e00a515SSatish Balay        (datafilespath,error,status) = self.executeShellCommand('cygpath -m '+self.dataFilesPath.datafilespath, log = self.log)
9144e00a515SSatish Balay        self.addMakeMacro('DATAFILESPATH',datafilespath)
9154e00a515SSatish Balay
916f8833479SBarry Smith    else:
917f8833479SBarry Smith      self.addDefine('REPLACE_DIR_SEPARATOR','\'\\\\\'')
918f8833479SBarry Smith      self.addDefine('DIR_SEPARATOR','\'/\'')
9195188cb68SSatish Balay      self.addDefine('DIR','"'+self.installdir.petscDir+'"')
9205188cb68SSatish Balay      self.addMakeMacro('wPETSC_DIR',self.installdir.petscDir)
9214e00a515SSatish Balay      if self.dataFilesPath.datafilespath:
9224e00a515SSatish Balay        self.addMakeMacro('DATAFILESPATH',self.dataFilesPath.datafilespath)
9235188cb68SSatish Balay    self.addDefine('ARCH','"'+self.installdir.petscArch+'"')
924f8833479SBarry Smith    return
925f8833479SBarry Smith
926d3d5cfdcSJacob Faibussowitsch  def configureCoverageForLang(self, log_printer_cls, lang, extra_coverage_flags=None, extra_debug_flags=None):
9277ce81a4bSJacob Faibussowitsch    """
9287ce81a4bSJacob Faibussowitsch    Check that a compiler accepts code-coverage flags. If the compiler does accept code-coverage flags
9297ce81a4bSJacob Faibussowitsch    try to set debugging flags equivalent to -Og.
9307ce81a4bSJacob Faibussowitsch
9317ce81a4bSJacob Faibussowitsch    Arguments:
9327ce81a4bSJacob Faibussowitsch    - lang: the language to check the coverage flag for
9337ce81a4bSJacob Faibussowitsch    - extra_coverage_flags: a list of extra flags to use when checking the coverage flags
9347ce81a4bSJacob Faibussowitsch    - extra_debug_flags: a list of extra flags to try when setting debug flags
9357ce81a4bSJacob Faibussowitsch
9367ce81a4bSJacob Faibussowitsch    On success:
9377ce81a4bSJacob Faibussowitsch    - defines PETSC_USE_COVERAGE to 1
9387ce81a4bSJacob Faibussowitsch    """
939d3d5cfdcSJacob Faibussowitsch    log_print = log_printer_cls(self)
9407ce81a4bSJacob Faibussowitsch
9417ce81a4bSJacob Faibussowitsch    def quoted(string):
9427ce81a4bSJacob Faibussowitsch      return string.join(("'", "'"))
9437ce81a4bSJacob Faibussowitsch
9447ce81a4bSJacob Faibussowitsch    def make_flag_list(default, extra):
9457ce81a4bSJacob Faibussowitsch      ret = [default]
9467ce81a4bSJacob Faibussowitsch      if extra is not None:
9477ce81a4bSJacob Faibussowitsch        assert isinstance(extra, list)
9487ce81a4bSJacob Faibussowitsch        ret.extend(extra)
9497ce81a4bSJacob Faibussowitsch      return ret
9507ce81a4bSJacob Faibussowitsch
9517ce81a4bSJacob Faibussowitsch    log_print('Checking coverage flag for language {}'.format(lang))
9529b81490aSJacob Faibussowitsch
9539b81490aSJacob Faibussowitsch    compiler = self.getCompiler(lang=lang)
9549b81490aSJacob Faibussowitsch    if self.setCompilers.isGNU(compiler, self.log):
955d3d5cfdcSJacob Faibussowitsch      is_gnuish = True
9569b81490aSJacob Faibussowitsch    elif self.setCompilers.isClang(compiler, self.log):
957d3d5cfdcSJacob Faibussowitsch      is_gnuish = True
9589b81490aSJacob Faibussowitsch    else:
959d3d5cfdcSJacob Faibussowitsch      is_gnuish = False
9607ce81a4bSJacob Faibussowitsch
9617ce81a4bSJacob Faibussowitsch    # if not gnuish and we don't have a set of extra flags, bail
962d3d5cfdcSJacob Faibussowitsch    if not is_gnuish and extra_coverage_flags is None:
9637ce81a4bSJacob 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)))
9647ce81a4bSJacob Faibussowitsch      return
9657ce81a4bSJacob Faibussowitsch
9667ce81a4bSJacob Faibussowitsch    coverage_flags = make_flag_list('--coverage', extra_coverage_flags)
9677ce81a4bSJacob Faibussowitsch    log_print('Checking set of coverage flags: {}'.format(coverage_flags))
9687ce81a4bSJacob Faibussowitsch
9699b81490aSJacob Faibussowitsch    found = None
9707ce81a4bSJacob Faibussowitsch    with self.Language(lang):
9717ce81a4bSJacob Faibussowitsch      with self.setCompilers.Language(lang):
9727ce81a4bSJacob Faibussowitsch        for flag in coverage_flags:
9739b81490aSJacob Faibussowitsch          # the linker also needs to see the coverage flag
9749b81490aSJacob Faibussowitsch          with self.setCompilers.extraCompilerFlags([flag], compilerOnly=False) as skip_flags:
9759b81490aSJacob Faibussowitsch            if not skip_flags and self.checkRun():
9769b81490aSJacob Faibussowitsch              # flag was accepted
9779b81490aSJacob Faibussowitsch              found = flag
9787ce81a4bSJacob Faibussowitsch              break
9799b81490aSJacob Faibussowitsch
9807ce81a4bSJacob Faibussowitsch          log_print(
9817ce81a4bSJacob Faibussowitsch            'Compiler {} did not accept coverage flag {}'.format(quoted(compiler), quoted(flag))
9827ce81a4bSJacob Faibussowitsch          )
9837ce81a4bSJacob Faibussowitsch
9849b81490aSJacob Faibussowitsch        if found is None:
9859b81490aSJacob Faibussowitsch          log_print(
9869b81490aSJacob Faibussowitsch            'Compiler {} did not accept ANY coverage flags: {}, bailing!'.format(
9879b81490aSJacob Faibussowitsch              quoted(compiler), coverage_flags
9889b81490aSJacob Faibussowitsch            )
9899b81490aSJacob Faibussowitsch          )
9907ce81a4bSJacob Faibussowitsch          return
9917ce81a4bSJacob Faibussowitsch
9929b81490aSJacob Faibussowitsch        # must do this exactly here since:
9939b81490aSJacob Faibussowitsch        #
9949b81490aSJacob Faibussowitsch        # 1. setCompilers.extraCompilerFlags() will reset the compiler flags on __exit__()
9959b81490aSJacob Faibussowitsch        #    (so cannot do it in the loop)
9969b81490aSJacob Faibussowitsch        # 2. we need to set the compiler flag while setCompilers.Language() is still in
9979b81490aSJacob Faibussowitsch        #    effect (so cannot do it outside the with statements)
9989b81490aSJacob Faibussowitsch        self.setCompilers.insertCompilerFlag(flag, False)
9999b81490aSJacob Faibussowitsch
10007ce81a4bSJacob Faibussowitsch    if not self.functions.haveFunction('__gcov_dump'):
10017ce81a4bSJacob Faibussowitsch      self.functions.checkClassify(['__gcov_dump'])
10027ce81a4bSJacob Faibussowitsch
10037ce81a4bSJacob Faibussowitsch    # now check if we can override the optimization level. It is only kosher to do so if
10047ce81a4bSJacob Faibussowitsch    # the user did not explicitly set the optimization flags (via CFLAGS, CXXFLAGS,
10057ce81a4bSJacob Faibussowitsch    # CXXOPTFLAGS, etc). If they have done so, we sternly warn them about their lapse in
10067ce81a4bSJacob Faibussowitsch    # judgement
10077ce81a4bSJacob Faibussowitsch    with self.Language(lang):
10087ce81a4bSJacob Faibussowitsch      compiler_flags = self.getCompilerFlags()
10097ce81a4bSJacob Faibussowitsch
10107ce81a4bSJacob Faibussowitsch    user_set          = 0
10117ce81a4bSJacob Faibussowitsch    allowed_opt_flags = re.compile(r'|'.join((r'-O[01g]', r'-g[1-9]*')))
10127ce81a4bSJacob Faibussowitsch    for flagsname in [self.getCompilerFlagsName(lang), self.compilerFlags.getOptionalFlagsName(lang)]:
10137ce81a4bSJacob Faibussowitsch      if flagsname in self.argDB:
10147ce81a4bSJacob Faibussowitsch        opt_flags = [
10157ce81a4bSJacob Faibussowitsch          f for f in self.compilerFlags.findOptFlags(compiler_flags) if not allowed_opt_flags.match(f)
10167ce81a4bSJacob Faibussowitsch        ]
10177ce81a4bSJacob Faibussowitsch        if opt_flags:
10187ce81a4bSJacob 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)))
10197ce81a4bSJacob Faibussowitsch          user_set = 1
10207ce81a4bSJacob Faibussowitsch          break
10217ce81a4bSJacob Faibussowitsch
102210b490f0SJacob Faibussowitsch    # disable this for now, the warning should be sufficient. If the user still chooses to
102310b490f0SJacob Faibussowitsch    # ignore it, then that's on them
102410b490f0SJacob Faibussowitsch    if 0 and not user_set:
10257ce81a4bSJacob Faibussowitsch      debug_flags = make_flag_list('-Og', extra_debug_flags)
10267ce81a4bSJacob Faibussowitsch      with self.setCompilers.Language(lang):
10277ce81a4bSJacob Faibussowitsch        for flag in debug_flags:
10287ce81a4bSJacob Faibussowitsch          try:
10297ce81a4bSJacob Faibussowitsch            self.setCompilers.addCompilerFlag(flag)
10307ce81a4bSJacob Faibussowitsch          except RuntimeError:
10317ce81a4bSJacob Faibussowitsch            continue
10327ce81a4bSJacob Faibussowitsch          break
10337ce81a4bSJacob Faibussowitsch
10347ce81a4bSJacob Faibussowitsch    self.addDefine('USE_COVERAGE', 1)
10357ce81a4bSJacob Faibussowitsch    return
10367ce81a4bSJacob Faibussowitsch
10379b81490aSJacob Faibussowitsch  def configureCoverage(self):
1038d3d5cfdcSJacob Faibussowitsch    """
1039d3d5cfdcSJacob Faibussowitsch    Configure coverage for all available languages.
1040d3d5cfdcSJacob Faibussowitsch
1041d3d5cfdcSJacob Faibussowitsch    If user did not request coverage, this function does nothing and returns immediatel.
1042d3d5cfdcSJacob Faibussowitsch    Therefore the following only apply to the case where the user requested coverage.
1043d3d5cfdcSJacob Faibussowitsch
1044d3d5cfdcSJacob Faibussowitsch    On success:
1045d3d5cfdcSJacob Faibussowitsch    - defines PETSC_USE_COVERAGE to 1
1046d3d5cfdcSJacob Faibussowitsch
1047d3d5cfdcSJacob Faibussowitsch    On failure:
1048d3d5cfdcSJacob Faibussowitsch    - If no compilers supported the coverage flag, throws RuntimeError
1049d3d5cfdcSJacob Faibussowitsch    -
1050d3d5cfdcSJacob Faibussowitsch    """
1051d3d5cfdcSJacob Faibussowitsch    class LogPrinter:
1052d3d5cfdcSJacob Faibussowitsch      def __init__(self, cfg):
1053d3d5cfdcSJacob Faibussowitsch        self.cfg = cfg
1054d3d5cfdcSJacob Faibussowitsch        try:
1055d3d5cfdcSJacob Faibussowitsch          import inspect
1056d3d5cfdcSJacob Faibussowitsch
1057d3d5cfdcSJacob Faibussowitsch          calling_func_stack = inspect.stack()[1]
1058d3d5cfdcSJacob Faibussowitsch          if sys.version_info >= (3, 5):
1059d3d5cfdcSJacob Faibussowitsch            func_name = calling_func_stack.function
1060d3d5cfdcSJacob Faibussowitsch          else:
1061d3d5cfdcSJacob Faibussowitsch            func_name = calling_func_stack[3]
1062d3d5cfdcSJacob Faibussowitsch        except:
1063d3d5cfdcSJacob Faibussowitsch          func_name = 'Unknown'
1064d3d5cfdcSJacob Faibussowitsch        self.fmt_str = func_name + '(): {}'
1065d3d5cfdcSJacob Faibussowitsch
1066d3d5cfdcSJacob Faibussowitsch      def __call__(self, msg, *args, **kwargs):
1067d3d5cfdcSJacob Faibussowitsch        return self.cfg.logPrint(self.fmt_str.format(msg), *args, **kwargs)
1068d3d5cfdcSJacob Faibussowitsch
1069d3d5cfdcSJacob Faibussowitsch    argdb_flag = 'with-coverage'
1070d3d5cfdcSJacob Faibussowitsch    log_print  = LogPrinter(self)
1071d3d5cfdcSJacob Faibussowitsch    if not self.argDB[argdb_flag]:
1072d3d5cfdcSJacob Faibussowitsch      log_print('coverage was disabled from command line or default')
1073d3d5cfdcSJacob Faibussowitsch      return
1074d3d5cfdcSJacob Faibussowitsch
1075d3d5cfdcSJacob Faibussowitsch    tested_langs = []
10769b81490aSJacob Faibussowitsch    for LANG in ['C', 'Cxx', 'CUDA', 'HIP', 'SYCL', 'FC']:
10779b81490aSJacob Faibussowitsch      compilerName = LANG.upper() if LANG in {'Cxx', 'FC'} else LANG + 'C'
10789b81490aSJacob Faibussowitsch      if hasattr(self.setCompilers, compilerName):
10799b81490aSJacob Faibussowitsch        kwargs = {}
10809b81490aSJacob Faibussowitsch        if LANG in {'CUDA'}:
10819b81490aSJacob Faibussowitsch          # nvcc preprocesses the base file into a bunch of intermediate files, which are
10829b81490aSJacob Faibussowitsch          # then compiled by the host compiler. Why is this a problem?  Because the
10839b81490aSJacob Faibussowitsch          # generated coverage data is based on these preprocessed source files! So gcov
10849b81490aSJacob Faibussowitsch          # tries to read it later, but since its in the tmp directory it cannot. Thus we
10859b81490aSJacob Faibussowitsch          # need to keep them around (in a place we know about).
10869b81490aSJacob Faibussowitsch          nvcc_tmp_dir = os.path.join(self.petscdir.dir, self.arch.arch, 'nvcc_tmp')
10879b81490aSJacob Faibussowitsch          try:
10889b81490aSJacob Faibussowitsch            os.mkdir(nvcc_tmp_dir)
10899b81490aSJacob Faibussowitsch          except FileExistsError:
10909b81490aSJacob Faibussowitsch            pass
10919b81490aSJacob Faibussowitsch          kwargs['extra_coverage_flags'] = [
10929b81490aSJacob Faibussowitsch            '-Xcompiler --coverage -Xcompiler -fPIC --keep --keep-dir={}'.format(nvcc_tmp_dir)
10939b81490aSJacob Faibussowitsch          ]
10949b81490aSJacob Faibussowitsch          if self.kokkos.found:
10959b81490aSJacob Faibussowitsch            # yet again the kokkos nvcc_wrapper goes out of its way to be as useless as
10969b81490aSJacob Faibussowitsch            # possible. Its default arch (sm_35) is actually too low to compile kokkos,
10979b81490aSJacob Faibussowitsch            # for whatever reason this works if you dont use the --keep and --keep-dir
10989b81490aSJacob Faibussowitsch            # flags above.
10999b81490aSJacob Faibussowitsch            kwargs['extra_coverage_flags'].append('-arch=native')
11009b81490aSJacob Faibussowitsch            kwargs['extra_debug_flags'] = ['-Xcompiler -Og']
1101d3d5cfdcSJacob Faibussowitsch        tested_langs.append(LANG)
1102d3d5cfdcSJacob Faibussowitsch        self.executeTest(self.configureCoverageForLang, args=[LogPrinter, LANG], kargs=kwargs)
1103d3d5cfdcSJacob Faibussowitsch
1104d3d5cfdcSJacob Faibussowitsch    if not self.defines.get('USE_COVERAGE'):
1105d3d5cfdcSJacob Faibussowitsch      # coverage was requested but no compilers accepted it, this is an error
1106d3d5cfdcSJacob Faibussowitsch      raise RuntimeError(
1107d3d5cfdcSJacob Faibussowitsch        'Coverage was requested (--{}={}) but none of the compilers supported it:\n{}\n'.format(
1108d3d5cfdcSJacob Faibussowitsch          argdb_flag, self.argDB[argdb_flag],
1109d3d5cfdcSJacob Faibussowitsch          '\n'.join(['  - {} ({})'.format(self.getCompiler(lang=lang), lang) for lang in tested_langs])
1110d3d5cfdcSJacob Faibussowitsch        )
1111d3d5cfdcSJacob Faibussowitsch      )
1112d3d5cfdcSJacob Faibussowitsch
1113d3d5cfdcSJacob Faibussowitsch    return
1114d3d5cfdcSJacob Faibussowitsch    # Disabled for now, since this does not really work. It solves the problem of
1115d3d5cfdcSJacob Faibussowitsch    # "undefined reference to __gcov_flush()" but if we add -lgcov we get:
1116d3d5cfdcSJacob Faibussowitsch    #
1117d3d5cfdcSJacob Faibussowitsch    # duplicate symbol '___gcov_reset' in:
1118d3d5cfdcSJacob Faibussowitsch    #     /Library/.../libclang_rt.profile_osx.a(GCDAProfiling.c.o)
1119d3d5cfdcSJacob Faibussowitsch    #     /opt/.../libgcov.a(_gcov_reset.o)
1120d3d5cfdcSJacob Faibussowitsch    # duplicate symbol '___gcov_dump' in:
1121d3d5cfdcSJacob Faibussowitsch    #     /opt/.../libgcov.a(_gcov_dump.o)
1122d3d5cfdcSJacob Faibussowitsch    #     /Library/.../libclang_rt.profile_osx.a(GCDAProfiling.c.o)
1123d3d5cfdcSJacob Faibussowitsch    # duplicate symbol '___gcov_fork' in:
1124d3d5cfdcSJacob Faibussowitsch    #     /opt/.../libgcov.a(_gcov_fork.o)
1125d3d5cfdcSJacob Faibussowitsch    #     /Library/.../libclang_rt.profile_osx.a(GCDAProfiling.c.o)
1126d3d5cfdcSJacob Faibussowitsch    #
1127d3d5cfdcSJacob Faibussowitsch    # I don't know how to solve this.
1128d3d5cfdcSJacob Faibussowitsch
1129d3d5cfdcSJacob Faibussowitsch    log_print('Checking if compilers can cross-link disparate coverage libraries')
1130d3d5cfdcSJacob Faibussowitsch    # At least one of the compilers has coverage enabled. Now need to make sure multiple
1131d3d5cfdcSJacob Faibussowitsch    # code coverage impls work together, specifically when using clang C/C++ compiler with
1132d3d5cfdcSJacob Faibussowitsch    # gfortran.
1133d3d5cfdcSJacob Faibussowitsch    if not hasattr(self.setCompilers, 'FC'):
1134d3d5cfdcSJacob Faibussowitsch      log_print('No fortran compiler detected. No need to check cross-linking!')
1135d3d5cfdcSJacob Faibussowitsch      return
1136d3d5cfdcSJacob Faibussowitsch
1137d3d5cfdcSJacob Faibussowitsch    c_lang = self.languages.clanguage
1138d3d5cfdcSJacob Faibussowitsch    if not self.setCompilers.isClang(self.getCompiler(lang=c_lang), self.log):
1139d3d5cfdcSJacob Faibussowitsch      # must be GCC
1140d3d5cfdcSJacob Faibussowitsch      log_print('C-language ({}) compiler is not clang, assuming it is GCC, so cross-linking with FC ({}) assumed to be OK'.format(c_lang, self.getCompiler(lang='FC')))
1141d3d5cfdcSJacob Faibussowitsch      return
1142d3d5cfdcSJacob Faibussowitsch
1143d3d5cfdcSJacob Faibussowitsch    # If we are here we:
1144d3d5cfdcSJacob Faibussowitsch    #   1. Have both C/C++ compiler and fortran compiler
1145d3d5cfdcSJacob Faibussowitsch    #   2. The C/C++ compiler is *not* the same as the fortran compiler (unless we start
1146d3d5cfdcSJacob Faibussowitsch    #      using flang)
1147d3d5cfdcSJacob Faibussowitsch    #
1148d3d5cfdcSJacob Faibussowitsch    # Now we check if we can cross-link
1149d3d5cfdcSJacob Faibussowitsch    def can_cross_link(**kwargs):
1150d3d5cfdcSJacob Faibussowitsch      f_body = "      subroutine foo()\n      print*,'testing'\n      return\n      end\n"
1151d3d5cfdcSJacob Faibussowitsch      c_body = "int main() { }"
1152d3d5cfdcSJacob Faibussowitsch
1153d3d5cfdcSJacob Faibussowitsch      return self.compilers.checkCrossLink(
1154d3d5cfdcSJacob Faibussowitsch        f_body, c_body, language1='FC', language2=c_lang, extralibs=self.compilers.flibs, **kwargs
1155d3d5cfdcSJacob Faibussowitsch      )
1156d3d5cfdcSJacob Faibussowitsch
1157d3d5cfdcSJacob Faibussowitsch    log_print('Trying to cross-link WITHOUT extra libs')
1158d3d5cfdcSJacob Faibussowitsch    if can_cross_link():
1159d3d5cfdcSJacob Faibussowitsch      log_print('Successfully cross-linked WITHOUT extra libs')
1160d3d5cfdcSJacob Faibussowitsch      # success, we already can cross-link
1161d3d5cfdcSJacob Faibussowitsch      return
1162d3d5cfdcSJacob Faibussowitsch
1163d3d5cfdcSJacob Faibussowitsch    extra_libs = ['-lgcov']
1164d3d5cfdcSJacob Faibussowitsch    log_print('Trying to cross-link with extra libs: {}'.format(extra_libs))
1165d3d5cfdcSJacob Faibussowitsch    if can_cross_link(extraObjs=extra_libs):
1166d3d5cfdcSJacob Faibussowitsch      log_print(
1167d3d5cfdcSJacob Faibussowitsch        'Successfully cross-linked using extra libs: {}, adding them to LIBS'.format(extra_libs)
1168d3d5cfdcSJacob Faibussowitsch      )
1169d3d5cfdcSJacob Faibussowitsch      self.setCompilers.LIBS += ' ' + ' '.join(extra_libs)
1170d3d5cfdcSJacob Faibussowitsch    else:
1171d3d5cfdcSJacob Faibussowitsch      # maybe should be an error?
1172d3d5cfdcSJacob Faibussowitsch      self.logPrintWarning("Could not successfully cross-link covered code between {} and FC. Sometimes this is a false positive. Assuming this does eventually end up working when the full link-line is assembled when building PETSc. If you later encounter linker errors about missing __gcov_exit(), __gcov_init(), __llvm_cov_flush() etc. this is why!".format(c_lang))
11739b81490aSJacob Faibussowitsch    return
11749b81490aSJacob Faibussowitsch
11757ce81a4bSJacob Faibussowitsch  def configureCoverageExecutable(self):
11767ce81a4bSJacob Faibussowitsch    """
11777ce81a4bSJacob Faibussowitsch    Check that a code-coverage collecting tool exists and is on PATH.
11787ce81a4bSJacob Faibussowitsch
11797ce81a4bSJacob Faibussowitsch    On success:
11807ce81a4bSJacob Faibussowitsch    - Adds PETSC_COVERAGE_EXEC make macro containing the full path to the coverage tool executable.
11817ce81a4bSJacob Faibussowitsch
11827ce81a4bSJacob Faibussowitsch    Raises RuntimeError if:
11837ce81a4bSJacob Faibussowitsch    - User explicitly requests auto-detection of the coverage tool from command line, and this
11847ce81a4bSJacob Faibussowitsch      routine fails to guess the suitable tool name.
11857ce81a4bSJacob Faibussowitsch    - The routine fails to find the tool, and --with-coverage is true
11867ce81a4bSJacob Faibussowitsch    """
11877ce81a4bSJacob Faibussowitsch    def log_print(msg, *args, **kwargs):
11887ce81a4bSJacob Faibussowitsch      self.logPrint('checkCoverage: '+str(msg), *args, **kwargs)
11897ce81a4bSJacob Faibussowitsch      return
11907ce81a4bSJacob Faibussowitsch
11917ce81a4bSJacob Faibussowitsch    def quoted(string):
11927ce81a4bSJacob Faibussowitsch      return string.join(("'", "'"))
11937ce81a4bSJacob Faibussowitsch
11947ce81a4bSJacob Faibussowitsch    required         = bool(self.argDB['with-coverage'])
11957ce81a4bSJacob Faibussowitsch    arg_opt          = self.argDB['with-coverage-exec']
11967ce81a4bSJacob Faibussowitsch    use_default_path = True
11977ce81a4bSJacob Faibussowitsch    search_path      = ''
11987ce81a4bSJacob Faibussowitsch
11997ce81a4bSJacob Faibussowitsch    log_print('{} to find an executable'.format('REQUIRED' if required else 'NOT required'))
12007ce81a4bSJacob Faibussowitsch    if arg_opt in {'auto', 'default-auto', '1'}:
12017ce81a4bSJacob Faibussowitsch      # detect it based on the C language compiler, hopefully this does not clash!
1202095fb05fSJacob Faibussowitsch      lang     = self.setCompilers.languages.clanguage
1203095fb05fSJacob Faibussowitsch      compiler = self.getCompiler(lang=lang)
12047ce81a4bSJacob Faibussowitsch      log_print('User did not explicitly set coverage exec (got {}), trying to auto-detect based on compiler {}'.format(quoted(arg_opt), quoted(compiler)))
12057ce81a4bSJacob Faibussowitsch      if self.setCompilers.isGNU(compiler, self.log):
1206095fb05fSJacob Faibussowitsch        compiler_version_re = re.compile(r'[gG][cC\+\-]+[0-9]* \(.+\) (\d+)\.(\d+)\.(\d+)')
12077ce81a4bSJacob Faibussowitsch        exec_names          = ['gcov']
12087ce81a4bSJacob Faibussowitsch      elif self.setCompilers.isClang(compiler, self.log):
1209095fb05fSJacob Faibussowitsch        compiler_version_re = re.compile(r'clang version (\d+)\.(\d+)\.(\d+)')
12107ce81a4bSJacob Faibussowitsch        exec_names          = ['llvm-cov']
12117ce81a4bSJacob Faibussowitsch        if self.setCompilers.isDarwin(self.log):
12127ce81a4bSJacob Faibussowitsch          # macOS masquerades llvm-cov as just 'gcov', so we add this to the list in case
12137ce81a4bSJacob Faibussowitsch          # bare llvm-cov does not work
12147ce81a4bSJacob Faibussowitsch          exec_names.append('gcov')
12157ce81a4bSJacob Faibussowitsch      elif arg_opt == 'default-auto' and not required:
12167ce81a4bSJacob Faibussowitsch        # default-auto implies the user did not set it via command line!
12177ce81a4bSJacob 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)))
12187ce81a4bSJacob Faibussowitsch        return
12197ce81a4bSJacob Faibussowitsch      else:
12207ce81a4bSJacob Faibussowitsch        # implies 'auto' explicitly set by user, or we were required to find
12217ce81a4bSJacob Faibussowitsch        # something. either way we should error
12227ce81a4bSJacob Faibussowitsch        raise RuntimeError('Could not auto-detect coverage tool for {}, please set coverage tool name explicitly'.format(quoted(compiler)))
1223095fb05fSJacob Faibussowitsch
1224095fb05fSJacob Faibussowitsch      try:
1225095fb05fSJacob Faibussowitsch        compiler_version_str = self.compilerFlags.version[lang]
1226095fb05fSJacob Faibussowitsch      except KeyError:
1227095fb05fSJacob Faibussowitsch        compiler_version_str = 'Unknown'
1228095fb05fSJacob Faibussowitsch
1229095fb05fSJacob Faibussowitsch      log_print('Searching version string {} (for compiler {}) using pattern {}'.format(quoted(compiler_version_str), quoted(compiler), quoted(compiler_version_re.pattern)))
1230095fb05fSJacob Faibussowitsch      compiler_version = compiler_version_re.search(compiler_version_str)
1231095fb05fSJacob Faibussowitsch      if compiler_version is not None:
1232095fb05fSJacob Faibussowitsch        log_print('Found major = {}, minor = {}, patch = {}'.format(compiler_version.group(1), compiler_version.group(2), compiler_version.group(3)))
1233095fb05fSJacob Faibussowitsch        # form [llvm-cov-14, llvm-cov-14.0, llvm-cov, etc.]
1234095fb05fSJacob Faibussowitsch        cov_exec_name = exec_names[0]
1235095fb05fSJacob Faibussowitsch        exec_names    = [
1236095fb05fSJacob Faibussowitsch          # llvm-cov-14
1237095fb05fSJacob Faibussowitsch          '{}-{}'.format(cov_exec_name, compiler_version.group(1)),
1238095fb05fSJacob Faibussowitsch           # llvm-cov-14.0
1239095fb05fSJacob Faibussowitsch          '{}-{}.{}'.format(cov_exec_name, compiler_version.group(1), compiler_version.group(2))
1240095fb05fSJacob Faibussowitsch        ] + exec_names
12417ce81a4bSJacob Faibussowitsch    else:
12427ce81a4bSJacob Faibussowitsch      log_print('User explicitly set coverage exec as {}'.format(quoted(arg_opt)))
12437ce81a4bSJacob Faibussowitsch      par_dir = os.path.dirname(arg_opt)
12447ce81a4bSJacob Faibussowitsch      if os.path.exists(par_dir):
12457ce81a4bSJacob Faibussowitsch        # arg_opt is path-like, we should only search the provided directory when we go
12467ce81a4bSJacob Faibussowitsch        # looking for the tool
12477ce81a4bSJacob Faibussowitsch        use_default_path = False
12487ce81a4bSJacob Faibussowitsch        search_path      = par_dir
12497ce81a4bSJacob Faibussowitsch      exec_names = [arg_opt]
12507ce81a4bSJacob Faibussowitsch
12517ce81a4bSJacob Faibussowitsch    make_macro_name = 'PETSC_COVERAGE_EXEC'
12527ce81a4bSJacob Faibussowitsch    log_print('Checking for coverage tool(s):\n{}'.format('\n'.join('- '+t for t in exec_names)))
12537ce81a4bSJacob Faibussowitsch    found_exec = self.getExecutables(
12547ce81a4bSJacob Faibussowitsch      exec_names,
12557ce81a4bSJacob Faibussowitsch      path=search_path, getFullPath=True, useDefaultPath=use_default_path, resultName=make_macro_name
12567ce81a4bSJacob Faibussowitsch    )
12577ce81a4bSJacob Faibussowitsch
12587ce81a4bSJacob Faibussowitsch    if found_exec is None:
12597ce81a4bSJacob Faibussowitsch      # didn't find the coverage tool
12607ce81a4bSJacob Faibussowitsch      if required:
12617ce81a4bSJacob Faibussowitsch        raise RuntimeError('Coverage tool(s) {} could not be found. Please provide explicit path to coverage tool'.format(exec_names))
12627ce81a4bSJacob Faibussowitsch      return
12637ce81a4bSJacob Faibussowitsch
12647ce81a4bSJacob Faibussowitsch    found_exec_name = os.path.basename(found_exec)
12657ce81a4bSJacob Faibussowitsch    if 'llvm-cov' in found_exec_name and 'gcov' not in found_exec_name:
12667ce81a4bSJacob Faibussowitsch      # llvm-cov needs to be called as 'llvm-cov gcov' to work
12677ce81a4bSJacob Faibussowitsch      self.addMakeMacro(make_macro_name, found_exec + ' gcov')
12687ce81a4bSJacob Faibussowitsch    return
12697ce81a4bSJacob Faibussowitsch
1270689a5dfaSJacob Faibussowitsch  def configureStrictPetscErrorCode(self):
1271689a5dfaSJacob Faibussowitsch    """
1272689a5dfaSJacob Faibussowitsch    Enables or disables strict PetscErrorCode checking.
1273689a5dfaSJacob Faibussowitsch
1274689a5dfaSJacob Faibussowitsch    If --with-strict-petscerrorcode = 1:
1275689a5dfaSJacob Faibussowitsch    - defines PETSC_USE_STRICT_PETSCERRORCODE to 1
1276689a5dfaSJacob Faibussowitsch
1277689a5dfaSJacob Faibussowitsch    Else:
1278689a5dfaSJacob Faibussowitsch    - deletes any prior PETSC_USE_STRICT_PETSCERRORCODE definitions (if they exist)
1279689a5dfaSJacob Faibussowitsch    """
1280689a5dfaSJacob Faibussowitsch    define_name = 'USE_STRICT_PETSCERRORCODE'
1281689a5dfaSJacob Faibussowitsch    if self.argDB['with-strict-petscerrorcode']:
1282689a5dfaSJacob Faibussowitsch      self.addDefine(define_name, 1)
1283689a5dfaSJacob Faibussowitsch    else:
1284689a5dfaSJacob Faibussowitsch      # in case it was somehow added previously
1285689a5dfaSJacob Faibussowitsch      self.delDefine(define_name)
1286689a5dfaSJacob Faibussowitsch    return
1287689a5dfaSJacob Faibussowitsch
1288f8833479SBarry Smith#-----------------------------------------------------------------------------------------------------
1289b10d012aSSatish Balay  def configureCygwinBrokenPipe(self):
1290b10d012aSSatish Balay    '''Cygwin version <= 1.7.18 had issues with pipes and long commands invoked from gnu-make
1291b10d012aSSatish Balay    http://cygwin.com/ml/cygwin/2013-05/msg00340.html '''
12927fca349cSMatthew G. Knepley    if config.setCompilers.Configure.isCygwin(self.log):
1293b10d012aSSatish Balay      import platform
1294b10d012aSSatish Balay      import re
1295b10d012aSSatish Balay      r=re.compile("([0-9]+).([0-9]+).([0-9]+)")
1296b10d012aSSatish Balay      m=r.match(platform.release())
1297b10d012aSSatish Balay      major=int(m.group(1))
1298b10d012aSSatish Balay      minor=int(m.group(2))
1299b10d012aSSatish Balay      subminor=int(m.group(3))
1300b10d012aSSatish Balay      if ((major < 1) or (major == 1 and minor < 7) or (major == 1 and minor == 7 and subminor <= 18)):
1301b10d012aSSatish Balay        self.addMakeMacro('PETSC_CYGWIN_BROKEN_PIPE','1')
1302b10d012aSSatish Balay    return
1303b10d012aSSatish Balay
1304b10d012aSSatish Balay#-----------------------------------------------------------------------------------------------------
1305569865ddSSatish Balay  def configureDefaultArch(self):
1306af0996ceSBarry Smith    conffile = os.path.join('lib','petsc','conf', 'petscvariables')
1307569865ddSSatish Balay    if self.framework.argDB['with-default-arch']:
1308c6ef1b5bSJed Brown      fd = open(conffile, 'w')
1309569865ddSSatish Balay      fd.write('PETSC_ARCH='+self.arch.arch+'\n')
1310da93591fSBarry Smith      fd.write('PETSC_DIR='+self.petscdir.dir+'\n')
1311b9b902edSJed Brown      fd.write('include '+os.path.join('$(PETSC_DIR)','$(PETSC_ARCH)','lib','petsc','conf','petscvariables')+'\n')
1312569865ddSSatish Balay      fd.close()
1313569865ddSSatish Balay      self.framework.actions.addArgument('PETSc', 'Build', 'Set default architecture to '+self.arch.arch+' in '+conffile)
1314569865ddSSatish Balay    elif os.path.isfile(conffile):
1315569865ddSSatish Balay      try:
1316569865ddSSatish Balay        os.unlink(conffile)
1317569865ddSSatish Balay      except:
1318569865ddSSatish Balay        raise RuntimeError('Unable to remove file '+conffile+'. Did a different user create it?')
1319569865ddSSatish Balay    return
1320569865ddSSatish Balay
1321569865ddSSatish Balay#-----------------------------------------------------------------------------------------------------
1322f8833479SBarry Smith  def configureScript(self):
1323f8833479SBarry Smith    '''Output a script in the conf directory which will reproduce the configuration'''
1324f8833479SBarry Smith    import nargs
1325495bf1a9SSatish Balay    import sys
1326af0996ceSBarry Smith    scriptName = os.path.join(self.arch.arch,'lib','petsc','conf', 'reconfigure-'+self.arch.arch+'.py')
1327f8833479SBarry Smith    args = dict([(nargs.Arg.parseArgument(arg)[0], arg) for arg in self.framework.clArgs])
1328e97fc2efSSatish Balay    if 'with-clean' in args:
1329e97fc2efSSatish Balay      del args['with-clean']
1330d418e2d7SSatish Balay    if 'force' in args:
1331d418e2d7SSatish Balay      del args['force']
1332f8833479SBarry Smith    if 'configModules' in args:
13331063a081SSatish Balay      if nargs.Arg.parseArgument(args['configModules'])[1] == 'PETSc.Configure':
1334f8833479SBarry Smith        del args['configModules']
1335f8833479SBarry Smith    if 'optionsModule' in args:
133623a19ef1SSatish Balay      if nargs.Arg.parseArgument(args['optionsModule'])[1] == 'config.compilerOptions':
1337f8833479SBarry Smith        del args['optionsModule']
1338f8833479SBarry Smith    if not 'PETSC_ARCH' in args:
13391063a081SSatish Balay      args['PETSC_ARCH'] = 'PETSC_ARCH='+str(self.arch.arch)
1340c6ef1b5bSJed Brown    f = open(scriptName, 'w')
1341495bf1a9SSatish Balay    f.write('#!'+sys.executable+'\n')
1342f8833479SBarry Smith    f.write('if __name__ == \'__main__\':\n')
1343f8833479SBarry Smith    f.write('  import sys\n')
13447561c02cSSatish Balay    f.write('  import os\n')
13457561c02cSSatish Balay    f.write('  sys.path.insert(0, os.path.abspath(\'config\'))\n')
1346f8833479SBarry Smith    f.write('  import configure\n')
13471063a081SSatish Balay    # pretty print repr(args.values())
13481063a081SSatish Balay    f.write('  configure_options = [\n')
13498bec23c5SJed Brown    for itm in sorted(args.values()):
13501063a081SSatish Balay      f.write('    \''+str(itm)+'\',\n')
13511063a081SSatish Balay    f.write('  ]\n')
1352f8833479SBarry Smith    f.write('  configure.petsc_configure(configure_options)\n')
1353f8833479SBarry Smith    f.close()
1354f8833479SBarry Smith    try:
13555b6bfdb9SJed Brown      os.chmod(scriptName, 0o775)
13565b6bfdb9SJed Brown    except OSError as e:
1357f8833479SBarry Smith      self.framework.logPrint('Unable to make reconfigure script executable:\n'+str(e))
1358f8833479SBarry Smith    self.framework.actions.addArgument('PETSc', 'File creation', 'Created '+scriptName+' for automatic reconfiguration')
1359f8833479SBarry Smith    return
1360f8833479SBarry Smith
1361f8833479SBarry Smith  def configureInstall(self):
1362f8833479SBarry Smith    '''Setup the directories for installation'''
1363f8833479SBarry Smith    if self.framework.argDB['prefix']:
13645b4fc442SVaclav Hapla      self.addMakeRule('print_mesg_after_build','',
13655b4fc442SVaclav Hapla       ['-@echo "========================================="',
13665b4fc442SVaclav Hapla        '-@echo "Now to install the libraries do:"',
13675b4fc442SVaclav Hapla        '-@echo "%s${MAKE_USER} PETSC_DIR=${PETSC_DIR} PETSC_ARCH=${PETSC_ARCH} install"' % self.installdir.installSudo,
1368315b77e6SSatish Balay        '-@echo "========================================="'])
1369f8833479SBarry Smith    else:
13705b4fc442SVaclav Hapla      self.addMakeRule('print_mesg_after_build','',
13715b4fc442SVaclav Hapla       ['-@echo "========================================="',
13725b4fc442SVaclav Hapla        '-@echo "Now to check if the libraries are working do:"',
13735b4fc442SVaclav Hapla        '-@echo "${MAKE_USER} PETSC_DIR=${PETSC_DIR} PETSC_ARCH=${PETSC_ARCH} check"',
1374315b77e6SSatish Balay        '-@echo "========================================="'])
1375f8833479SBarry Smith      return
1376f8833479SBarry Smith
137728bb2e72SSatish Balay  def postProcessPackages(self):
137828bb2e72SSatish Balay    postPackages=[]
137928bb2e72SSatish Balay    for i in self.framework.packages:
138028bb2e72SSatish Balay      if hasattr(i,'postProcess'): postPackages.append(i)
138128bb2e72SSatish Balay    if postPackages:
1382f0b74427SPierre Jolivet      # ctetgen needs PETSc conf files. so attempt to create them early
1383a77eb93bSSatish Balay      self.framework.dumpConfFiles()
1384d9293e7bSBarry Smith      # tacky fix for dependency of Aluimia on Pflotran; requested via petsc-dev Matt provide a correct fix
1385d9293e7bSBarry Smith      for i in postPackages:
1386d9293e7bSBarry Smith        if i.name.upper() in ['PFLOTRAN']:
1387d9293e7bSBarry Smith          i.postProcess()
1388d9293e7bSBarry Smith          postPackages.remove(i)
138928bb2e72SSatish Balay      for i in postPackages: i.postProcess()
1390aa5c8b8eSBarry Smith      for i in postPackages:
1391aa5c8b8eSBarry Smith        if i.installedpetsc:
1392aa5c8b8eSBarry Smith          self.installed = 1
1393aa5c8b8eSBarry Smith          break
139428bb2e72SSatish Balay    return
1395f8833479SBarry Smith
1396ce78bad3SBarry Smith  def generateFortranBindings(self):
139784585aa5SBarry Smith    '''Remove any current Fortran bindings from previous ./configure runs because they may not be needed this run'''
139884585aa5SBarry Smith    import shutil
139984585aa5SBarry Smith    dir = os.path.join(self.arch.arch,'ftn')
140084585aa5SBarry Smith    if os.path.isdir(dir): shutil.rmtree(dir)
1401ce78bad3SBarry Smith    if hasattr(self.compilers, 'FC') and self.framework.argDB['with-fortran-bindings']:
1402ce78bad3SBarry Smith      self.logPrintBox('Generating Fortran binding')
1403ce78bad3SBarry Smith      try:
14044cc8716dSJose E. Roman        sys.path.append(os.path.join(self.petscdir.dir,'lib','petsc','bin'))
14054cc8716dSJose E. Roman        import generatefortranbindings
1406*b06eb4cdSBarry Smith        generatefortranbindings.main(self.petscdir.dir, '', self.arch.arch, self.mpi.mpi_f08)
1407ce78bad3SBarry Smith      except RuntimeError as e:
1408ce78bad3SBarry Smith        raise RuntimeError('*******Error generating Fortran stubs: '+str(e)+'*******\n')
1409ce78bad3SBarry Smith
1410f8833479SBarry Smith  def configure(self):
1411bf3e94a3SBarry Smith    if 'package-prefix-hash' in self.argDB:
1412bf3e94a3SBarry Smith      # turn off prefix if it was only used to for installing external packages.
1413bf3e94a3SBarry Smith      self.framework.argDB['prefix'] = ''
1414bf3e94a3SBarry Smith      self.dir = os.path.abspath(os.path.join(self.petscdir.dir, self.arch.arch))
1415bf3e94a3SBarry Smith      self.installdir.dir = self.dir
1416bf3e94a3SBarry Smith      self.installdir.petscDir = self.petscdir.dir
1417bf3e94a3SBarry Smith      self.petscDir = self.petscdir.dir
1418bf3e94a3SBarry Smith      self.petscArch = self.arch.arch
1419bf3e94a3SBarry Smith      self.addMakeMacro('PREFIXDIR',self.dir)
1420bf3e94a3SBarry Smith      self.confDir = os.path.abspath(os.path.join(self.petscdir.dir, self.arch.arch))
1421bf3e94a3SBarry Smith
1422f8833479SBarry Smith    if not os.path.samefile(self.petscdir.dir, os.getcwd()):
1423f8833479SBarry Smith      raise RuntimeError('Wrong PETSC_DIR option specified: '+str(self.petscdir.dir) + '\n  Configure invoked in: '+os.path.realpath(os.getcwd()))
1424550489e3SMatthew 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):
14253552d8fbSSatish Balay      raise RuntimeError('Incorrect option --prefix='+self.framework.argDB['prefix']+' specified. It cannot be same as PETSC_DIR!')
14268fd0dbdbSBarry Smith    if self.framework.argDB['prefix'] and self.framework.argDB['prefix'].find(' ') > -1:
14278fd0dbdbSBarry 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')
1428c16c35a9SSatish 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)):
1429c16c35a9SSatish Balay      raise RuntimeError('Incorrect option --prefix='+self.framework.argDB['prefix']+' specified. It cannot be same as PETSC_DIR/PETSC_ARCH!')
1430f16c1317SJed Brown    self.framework.header          = os.path.join(self.arch.arch,'include','petscconf.h')
1431f16c1317SJed Brown    self.framework.cHeader         = os.path.join(self.arch.arch,'include','petscfix.h')
1432bf113f49SJacob Faibussowitsch    self.framework.poisonheader    = os.path.join(self.arch.arch,'include','petscconf_poison.h')
14339c735a01SStefano Zampini    self.framework.pkgheader       = os.path.join(self.arch.arch,'include','petscpkg_version.h')
1434af0996ceSBarry Smith    self.framework.makeMacroHeader = os.path.join(self.arch.arch,'lib','petsc','conf','petscvariables')
1435af0996ceSBarry Smith    self.framework.makeRuleHeader  = os.path.join(self.arch.arch,'lib','petsc','conf','petscrules')
1436f8833479SBarry Smith    if self.libraries.math is None:
1437f8833479SBarry Smith      raise RuntimeError('PETSc requires a functional math library. Please send configure.log to petsc-maint@mcs.anl.gov.')
1438f8833479SBarry Smith    if self.languages.clanguage == 'Cxx' and not hasattr(self.compilers, 'CXX'):
1439f8833479SBarry Smith      raise RuntimeError('Cannot set C language to C++ without a functional C++ compiler.')
1440ed938b00SJed Brown    self.executeTest(self.configureRTLDDefault)
1441b2843cf1SBarry Smith    self.executeTest(self.configurePrefetch)
14422400fdedSBarry Smith    self.executeTest(self.configureUnused)
14431ef8df7fSJed Brown    self.executeTest(self.configureDeprecated)
144498ed35c3SBarry Smith    self.executeTest(self.configureIsatty)
1445e8e972b2SVaclav Hapla    self.executeTest(self.configureExpect)
1446e8e972b2SVaclav Hapla    self.executeTest(self.configureAlign)
1447e8e972b2SVaclav Hapla    self.executeTest(self.configureFunctionName)
1448e8e972b2SVaclav Hapla    self.executeTest(self.configureIntptrt)
1449f8833479SBarry Smith    self.executeTest(self.configureSolaris)
1450f8833479SBarry Smith    self.executeTest(self.configureLinux)
14510f64ec89SBarry Smith    self.executeTest(self.configureDarwin)
1452f8833479SBarry Smith    self.executeTest(self.configureWin32)
1453b10d012aSSatish Balay    self.executeTest(self.configureCygwinBrokenPipe)
1454569865ddSSatish Balay    self.executeTest(self.configureDefaultArch)
1455f8833479SBarry Smith    self.executeTest(self.configureScript)
1456f8833479SBarry Smith    self.executeTest(self.configureInstall)
145709bc878fSSatish Balay    self.executeTest(self.configureAtoll)
14589b81490aSJacob Faibussowitsch    self.executeTest(self.configureCoverage)
14597ce81a4bSJacob Faibussowitsch    self.executeTest(self.configureCoverageExecutable)
1460689a5dfaSJacob Faibussowitsch    self.executeTest(self.configureStrictPetscErrorCode)
1461648c30bcSBarry Smith    self.executeTest(self.configureSanitize)
1462ce78bad3SBarry Smith    self.executeTest(self.generateFortranBindings)
1463f8833479SBarry Smith
1464f8833479SBarry Smith    self.Dump()
1465f8833479SBarry Smith    self.dumpConfigInfo()
14662a4161d9SMatthew G Knepley    self.dumpMachineInfo()
146749fe22e6SSatish Balay    self.delGenFiles()
146840277576SBarry Smith    # need to save the current state of BuildSystem so that postProcess() packages can read it in and perhaps run make install
146940277576SBarry Smith    self.framework.storeSubstitutions(self.framework.argDB)
1470492432c8SJed Brown    self.framework.argDB['configureCache'] = pickle.dumps(self.framework)
147140277576SBarry Smith    self.framework.argDB.save(force = True)
14728244ab14SJed Brown    self.DumpPkgconfig('PETSc.pc')
14738244ab14SJed Brown    self.DumpPkgconfig('petsc.pc')
1474351d3a41SMatthew G Knepley    self.DumpModule()
1475f7ad81e1SBarry Smith    self.postProcessPackages()
1476f8833479SBarry Smith    self.framework.log.write('================================================================================\n')
1477f8833479SBarry Smith    self.logClear()
1478f8833479SBarry Smith    return
1479