xref: /petsc/config/PETSc/Configure.py (revision 3384ab8a42e580776d97539215944c00bb10b696)
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')
1968f561fa3SPatrick Sanan          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')
203262119f8SBarry Smith
204262119f8SBarry Smith      fd.write('\n')
205262119f8SBarry Smith      fd.write('Name: PETSc\n')
206262119f8SBarry Smith      fd.write('Description: Library to solve ODEs and algebraic equations\n')
207351d3a41SMatthew G Knepley      fd.write('Version: %s\n' % self.petscdir.version)
2085e3311eeSJed Brown      fd.write('Cflags: ' + ' '.join([self.setCompilers.CPPFLAGS] + cflags_inc) + '\n')
20937371b91SJed Brown      fd.write('Libs: '+self.libraries.toStringNoDupes(['-L${libdir}', self.petsclib], with_rpath=False)+'\n')
2108ebf8858SJed Brown      # Remove RPATH flags from library list.  User can add them using
2118ebf8858SJed Brown      # pkg-config --variable=ldflag_rpath and pkg-config --libs-only-L
212de8f682fSSatish 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')
213262119f8SBarry Smith    return
214262119f8SBarry Smith
215351d3a41SMatthew G Knepley  def DumpModule(self):
216351d3a41SMatthew G Knepley    ''' Create a module file '''
217af0996ceSBarry Smith    if not os.path.exists(os.path.join(self.petscdir.dir,self.arch.arch,'lib','petsc','conf','modules')):
218af0996ceSBarry Smith      os.makedirs(os.path.join(self.petscdir.dir,self.arch.arch,'lib','petsc','conf','modules'))
219af0996ceSBarry Smith    if not os.path.exists(os.path.join(self.petscdir.dir,self.arch.arch,'lib','petsc','conf','modules','petsc')):
220af0996ceSBarry Smith      os.makedirs(os.path.join(self.petscdir.dir,self.arch.arch,'lib','petsc','conf','modules','petsc'))
221351d3a41SMatthew G Knepley    if self.framework.argDB['prefix']:
2225bb5b263SMatthew G. Knepley      installdir  = self.installdir.dir
22355d606a3SSatish Balay      installarch = ''
22455d606a3SSatish Balay      installpath = os.path.join(installdir,'bin')
225351d3a41SMatthew G Knepley    else:
226351d3a41SMatthew G Knepley      installdir  = self.petscdir.dir
22755d606a3SSatish Balay      installarch = self.arch.arch
22855d606a3SSatish Balay      installpath = os.path.join(installdir,installarch,'bin')+':'+os.path.join(installdir,'bin')
229af0996ceSBarry Smith    fd = open(os.path.join(self.petscdir.dir,self.arch.arch,'lib','petsc','conf','modules','petsc',self.petscdir.version),'w')
230351d3a41SMatthew G Knepley    fd.write('''\
231351d3a41SMatthew G Knepley#%%Module
232351d3a41SMatthew G Knepley
233351d3a41SMatthew G Knepleyproc ModulesHelp { } {
234351d3a41SMatthew G Knepley    puts stderr "This module sets the path and environment variables for petsc-%s"
235a17b96a8SKyle Gerard Felker    puts stderr "     see https://petsc.org/ for more information      "
236351d3a41SMatthew G Knepley    puts stderr ""
237351d3a41SMatthew G Knepley}
238351d3a41SMatthew G Knepleymodule-whatis "PETSc - Portable, Extensible Toolkit for Scientific Computation"
239351d3a41SMatthew G Knepley
240dd486775SJed Brownset petsc_dir   "%s"
241dd486775SJed Brownset petsc_arch  "%s"
242351d3a41SMatthew G Knepley
243dd486775SJed Brownsetenv PETSC_ARCH "$petsc_arch"
244dd486775SJed Brownsetenv PETSC_DIR "$petsc_dir"
245dd486775SJed Brownprepend-path PATH "%s"
24655d606a3SSatish Balay''' % (self.petscdir.version, installdir, installarch, installpath))
247351d3a41SMatthew G Knepley    fd.close()
248351d3a41SMatthew G Knepley    return
249351d3a41SMatthew G Knepley
250f8833479SBarry Smith  def Dump(self):
251f8833479SBarry Smith    ''' Actually put the values into the configuration files '''
252f8833479SBarry Smith    # eventually everything between -- should be gone
25317f368bcSBarry Smith    if self.mpi.usingMPIUni:
25417f368bcSBarry Smith      #
25517f368bcSBarry Smith      # Remove any MPI/MPICH include files that may have been put here by previous runs of ./configure
2567908f030SMatthew 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)
25717f368bcSBarry Smith
258b5f71184SBarry Smith    self.logPrintDivider()
259b5f71184SBarry Smith    # Test for compiler-specific macros that need to be defined.
260b5f71184SBarry Smith    if self.setCompilers.isCrayVector('CC', self.log):
261b5f71184SBarry Smith      self.addDefine('HAVE_CRAY_VECTOR','1')
262b5f71184SBarry Smith
263b5f71184SBarry Smith    if self.functions.haveFunction('gethostbyname') and self.functions.haveFunction('socket') and self.headers.haveHeader('netinet/in.h'):
264b5f71184SBarry Smith      self.addDefine('USE_SOCKET_VIEWER','1')
265b5f71184SBarry Smith      if self.checkCompile('#include <sys/socket.h>','setsockopt(0,SOL_SOCKET,SO_REUSEADDR,0,0)'):
266b5f71184SBarry Smith        self.addDefine('HAVE_SO_REUSEADDR','1')
267b5f71184SBarry Smith
268b5f71184SBarry Smith    self.logPrintDivider()
2695f27b2e0SBarry Smith    self.setCompilers.pushLanguage('C')
2705f27b2e0SBarry Smith    compiler = self.setCompilers.getCompiler()
271217fe27eSSatish Balay    if [s for s in ['mpicc','mpiicc'] if os.path.basename(compiler).find(s)>=0]:
2725f27b2e0SBarry Smith      try:
2735f27b2e0SBarry Smith        output   = self.executeShellCommand(compiler + ' -show', log = self.log)[0]
2745f27b2e0SBarry Smith        compiler = output.split(' ')[0]
275f424265bSStefano Zampini        self.addDefine('MPICC_SHOW','"'+output.strip().replace('\n','\\\\n').replace('"','')+'"')
2765f27b2e0SBarry Smith      except:
277c9872b61SBarry Smith        self.addDefine('MPICC_SHOW','"Unavailable"')
278c9872b61SBarry Smith    else:
279c9872b61SBarry Smith      self.addDefine('MPICC_SHOW','"Unavailable"')
2805f27b2e0SBarry Smith    self.setCompilers.popLanguage()
281f8833479SBarry Smith#-----------------------------------------------------------------------------------------------------
282f8833479SBarry Smith
283f8833479SBarry Smith    # Sometimes we need C compiler, even if built with C++
284f8833479SBarry Smith    self.setCompilers.pushLanguage('C')
285e4c30378SBarry Smith    # do not use getCompilerFlags() because that automatically includes the CPPFLAGS so one ends up with duplication flags in makefile usage
286e4c30378SBarry Smith    self.addMakeMacro('CC_FLAGS',self.setCompilers.CFLAGS)
287f8833479SBarry Smith    self.setCompilers.popLanguage()
288f8833479SBarry Smith
28934f774f6SJed Brown    # And sometimes we need a C++ compiler even when PETSc is built with C
29034f774f6SJed Brown    if hasattr(self.compilers, 'CXX'):
29134f774f6SJed Brown      self.setCompilers.pushLanguage('Cxx')
29229921a8fSScott Kruger      self.addDefine('HAVE_CXX','1')
2930b119762SSatish Balay      self.addMakeMacro('CXXPP_FLAGS',self.setCompilers.CXXPPFLAGS)
294e4c30378SBarry Smith      # do not use getCompilerFlags() because that automatically includes the CXXPPFLAGS so one ends up with duplication flags in makefile usage
295e4c30378SBarry Smith      self.addMakeMacro('CXX_FLAGS',self.setCompilers.CXXFLAGS+' '+self.setCompilers.CXX_CXXFLAGS)
2962f4326f3SSatish Balay      cxx_linker = self.setCompilers.getLinker()
2972f4326f3SSatish Balay      self.addMakeMacro('CXX_LINKER',cxx_linker)
2982f4326f3SSatish Balay      self.addMakeMacro('CXX_LINKER_FLAGS',self.setCompilers.getLinkerFlags())
29934f774f6SJed Brown      self.setCompilers.popLanguage()
30005a757cfSSatish Balay    else:
30105a757cfSSatish Balay      self.addMakeMacro('CXX','')
30234f774f6SJed Brown
303f8833479SBarry Smith    # C preprocessor values
3041315f054SBarry Smith    self.addMakeMacro('CPP_FLAGS',self.setCompilers.CPPFLAGS)
305f8833479SBarry Smith
306f8833479SBarry Smith    # compiler values
307f8833479SBarry Smith    self.setCompilers.pushLanguage(self.languages.clanguage)
308f8833479SBarry Smith    self.addMakeMacro('PCC',self.setCompilers.getCompiler())
309e4c30378SBarry Smith    # do not use getCompilerFlags() because that automatically includes the preprocessor flags so one ends up with duplication flags in makefile usage
310e4c30378SBarry Smith    if self.languages.clanguage == 'C':
311e4c30378SBarry Smith      self.addMakeMacro('PCC_FLAGS','$(CC_FLAGS)')
312e4c30378SBarry Smith    else:
313e4c30378SBarry Smith      self.addMakeMacro('PCC_FLAGS','$(CXX_FLAGS)')
314f8833479SBarry Smith    self.setCompilers.popLanguage()
315f8833479SBarry Smith    # .o or .obj
316f8833479SBarry Smith    self.addMakeMacro('CC_SUFFIX','o')
317f8833479SBarry Smith
318f8833479SBarry Smith    # executable linker values
319f8833479SBarry Smith    self.setCompilers.pushLanguage(self.languages.clanguage)
320f8833479SBarry Smith    pcc_linker = self.setCompilers.getLinker()
321f8833479SBarry Smith    self.addMakeMacro('PCC_LINKER',pcc_linker)
322eacb1f64SJunchao Zhang    # We need to add sycl flags when linking petsc. See more in sycl.py.
323eacb1f64SJunchao Zhang    if hasattr(self.compilers, 'SYCLC'):
324eacb1f64SJunchao Zhang      self.addMakeMacro('PCC_LINKER_FLAGS',self.setCompilers.getLinkerFlags()+' '+self.setCompilers.SYCLFLAGS+' '+self.setCompilers.SYCLC_LINKER_FLAGS)
325eacb1f64SJunchao Zhang    else:
326c84a332bSSatish Balay      self.addMakeMacro('PCC_LINKER_FLAGS',self.setCompilers.getLinkerFlags())
327f8833479SBarry Smith    self.setCompilers.popLanguage()
328f8833479SBarry Smith    # '' for Unix, .exe for Windows
329f8833479SBarry Smith    self.addMakeMacro('CC_LINKER_SUFFIX','')
330f8833479SBarry Smith
331f8833479SBarry Smith    if hasattr(self.compilers, 'FC'):
332cb297985SSatish Balay      if self.framework.argDB['with-fortran-bindings']:
333257f4e5aSSatish Balay        if not self.fortran.fortranIsF90:
334257f4e5aSSatish Balay          raise RuntimeError('Error! Fortran compiler "'+self.compilers.FC+'" does not support F90! PETSc fortran bindings require a F90 compiler')
335fbf9dbe5SBarry Smith        self.addDefine('USE_FORTRAN_BINDINGS','1')
3368d35c829SSatish Balay        if not self.ftncmdline.have_command_argument:
3378d35c829SSatish Balay          raise RuntimeError('Error! Fortran compiler "'+self.compilers.FC+'" does not support F2003 GET_COMMAND_ARGUMENT()!')
338*3384ab8aSJose E. Roman        if not self.fortran.fortranInitializePtrInDecl:
339*3384ab8aSJose E. Roman          raise RuntimeError('Error! Fortran compiler "'+self.compilers.FC+'" does not support the F2018 standard allowing initializing pointers in the declaration!')
340f8833479SBarry Smith      self.setCompilers.pushLanguage('FC')
341f8833479SBarry Smith      # need FPPFLAGS in config/setCompilers
3420b119762SSatish Balay      self.addMakeMacro('FPP_FLAGS',self.setCompilers.FPPFLAGS)
343f8833479SBarry Smith
344f8833479SBarry Smith      # compiler values
345f8833479SBarry Smith      self.addMakeMacro('FC_FLAGS',self.setCompilers.getCompilerFlags())
346f8833479SBarry Smith      self.setCompilers.popLanguage()
347f8833479SBarry Smith      # .o or .obj
348f8833479SBarry Smith      self.addMakeMacro('FC_SUFFIX','o')
349f8833479SBarry Smith
350f8833479SBarry Smith      # executable linker values
351f8833479SBarry Smith      self.setCompilers.pushLanguage('FC')
352a9acdec7SBarry Smith      self.addMakeMacro('FC_LINKER',self.setCompilers.getLinker())
3536d53d35eSSatish Balay      self.addMakeMacro('FC_LINKER_FLAGS',self.setCompilers.getLinkerFlags())
354bb82cf9cSSatish Balay      self.setCompilers.popLanguage()
3555d631499SMatthew Knepley
3565d631499SMatthew Knepley      # F90 Modules
3575d631499SMatthew Knepley      if self.setCompilers.fortranModuleIncludeFlag:
3585d631499SMatthew Knepley        self.addMakeMacro('FC_MODULE_FLAG', self.setCompilers.fortranModuleIncludeFlag)
3596ddd6694SSatish Balay      else: # for non-f90 compilers like g77
3606ddd6694SSatish Balay        self.addMakeMacro('FC_MODULE_FLAG', '-I')
361a324c51cSMatthew G Knepley      if self.setCompilers.fortranModuleIncludeFlag:
362a324c51cSMatthew G Knepley        self.addMakeMacro('FC_MODULE_OUTPUT_FLAG', self.setCompilers.fortranModuleOutputFlag)
363f8833479SBarry Smith    else:
364f8833479SBarry Smith      self.addMakeMacro('FC','')
365f8833479SBarry Smith
36646a3958fSBarry Smith    if hasattr(self.compilers, 'CUDAC'):
3677ff2890cSSatish Balay      self.setCompilers.pushLanguage('CUDA')
368d93a25ecSSatish Balay      self.addMakeMacro('CUDAC_FLAGS',self.setCompilers.getCompilerFlags())
36950dcbc5aSJunchao Zhang      self.addMakeMacro('CUDAPP_FLAGS',self.setCompilers.CUDAPPFLAGS)
3707ff2890cSSatish Balay      self.setCompilers.popLanguage()
3717ff2890cSSatish Balay
372694a2f0eSJunchao Zhang    if hasattr(self.compilers, 'HIPC'):
37328f796eaSScott Kruger      self.setCompilers.pushLanguage('HIP')
374694a2f0eSJunchao Zhang      self.addMakeMacro('HIPC_FLAGS',self.setCompilers.getCompilerFlags())
3757fb1458fSStefano Zampini      self.addMakeMacro('HIPPP_FLAGS',self.setCompilers.HIPPPFLAGS)
37628f796eaSScott Kruger      self.setCompilers.popLanguage()
37728f796eaSScott Kruger
37850dcbc5aSJunchao Zhang    if hasattr(self.compilers, 'SYCLC'):
37928f796eaSScott Kruger      self.setCompilers.pushLanguage('SYCL')
38050dcbc5aSJunchao Zhang      self.addMakeMacro('SYCLC_FLAGS',self.setCompilers.getCompilerFlags())
381eacb1f64SJunchao Zhang      self.addMakeMacro('SYCLC_LINKER_FLAGS',self.setCompilers.getLinkerFlags())
38250dcbc5aSJunchao Zhang      self.addMakeMacro('SYCLPP_FLAGS',self.setCompilers.SYCLPPFLAGS)
38328f796eaSScott Kruger      self.setCompilers.popLanguage()
38428f796eaSScott Kruger
385f8833479SBarry Smith    # shared library linker values
386f8833479SBarry Smith    self.setCompilers.pushLanguage(self.languages.clanguage)
387f8833479SBarry Smith    # need to fix BuildSystem to collect these separately
388f8833479SBarry Smith    self.addMakeMacro('SL_LINKER',self.setCompilers.getLinker())
38970db8aa6SSatish Balay    self.addMakeMacro('SL_LINKER_FLAGS','${PCC_LINKER_FLAGS}')
390f8833479SBarry Smith    self.setCompilers.popLanguage()
391f8833479SBarry Smith    # One of 'a', 'so', 'lib', 'dll', 'dylib' (perhaps others also?) depending on the library generator and architecture
392f8833479SBarry Smith    # Note: . is not included in this macro, consistent with AR_LIB_SUFFIX
393f8833479SBarry Smith    if self.setCompilers.sharedLibraryExt == self.setCompilers.AR_LIB_SUFFIX:
394f8833479SBarry Smith      self.addMakeMacro('SL_LINKER_SUFFIX', '')
39546bc77b6SBarry Smith      self.addDefine('SLSUFFIX','""')
396f8833479SBarry Smith    else:
397f8833479SBarry Smith      self.addMakeMacro('SL_LINKER_SUFFIX', self.setCompilers.sharedLibraryExt)
39846bc77b6SBarry Smith      self.addDefine('SLSUFFIX','"'+self.setCompilers.sharedLibraryExt+'"')
399bb82cf9cSSatish Balay
40023e93537SBarry Smith    self.addMakeMacro('SL_LINKER_LIBS','${PETSC_EXTERNAL_LIB_BASIC}')
401bb82cf9cSSatish Balay
402f8833479SBarry Smith#-----------------------------------------------------------------------------------------------------
403f8833479SBarry Smith
404f8833479SBarry Smith    # CONLY or CPP. We should change the PETSc makefiles to do this better
405f8833479SBarry Smith    if self.languages.clanguage == 'C': lang = 'CONLY'
406f8833479SBarry Smith    else: lang = 'CXXONLY'
407f8833479SBarry Smith    self.addMakeMacro('PETSC_LANGUAGE',lang)
408f8833479SBarry Smith
409f8833479SBarry Smith    # real or complex
410f8833479SBarry Smith    self.addMakeMacro('PETSC_SCALAR',self.scalartypes.scalartype)
411f8833479SBarry Smith    # double or float
412f8833479SBarry Smith    self.addMakeMacro('PETSC_PRECISION',self.scalartypes.precision)
413f8833479SBarry Smith
414f8833479SBarry Smith    if self.framework.argDB['with-batch']:
415f8833479SBarry Smith      self.addMakeMacro('PETSC_WITH_BATCH','1')
416f8833479SBarry Smith
417f8833479SBarry Smith#-----------------------------------------------------------------------------------------------------
418a6cc6bb1SBarry Smith    # print include and lib for makefiles
419b5f71184SBarry Smith    self.logPrintDivider()
420f8833479SBarry Smith    self.framework.packages.reverse()
4215a21677cSJed Brown    petscincludes = [os.path.join(self.petscdir.dir,'include'),os.path.join(self.petscdir.dir,self.arch.arch,'include')]
4228749e224SJunchao Zhang    petscincludes_install = [os.path.join(self.installdir.dir, 'include')] if self.framework.argDB['prefix'] else petscincludes
4235a21677cSJed Brown    includes = []
424de8f682fSSatish Balay    self.packagelibs = []
425f8833479SBarry Smith    for i in self.framework.packages:
4267f0ff1afSBarry Smith      if not i.required:
4273972cb20SJacob Faibussowitsch        if i.devicePackage:
4283972cb20SJacob Faibussowitsch          self.addDefine('HAVE_DEVICE',1)
429eeb16384SBarry Smith        self.addDefine('HAVE_'+i.PACKAGE.replace('-','_'), 1)  # ONLY list package if it is used directly by PETSc (and not only by another package)
430f8833479SBarry Smith      if not isinstance(i.lib, list):
431f8833479SBarry Smith        i.lib = [i.lib]
432de8f682fSSatish Balay      if i.linkedbypetsc: self.packagelibs.extend(i.lib)
433eeb16384SBarry Smith      self.addMakeMacro(i.PACKAGE.replace('-','_')+'_LIB', self.libraries.toStringNoDupes(i.lib))
434f8833479SBarry Smith      if hasattr(i,'include'):
435f8833479SBarry Smith        if not isinstance(i.include,list):
436f8833479SBarry Smith          i.include = [i.include]
437ac9e4c42SSatish Balay        includes.extend(i.include)
438eeb16384SBarry Smith        self.addMakeMacro(i.PACKAGE.replace('-','_')+'_INCLUDE',self.headers.toStringNoDupes(i.include))
439571416bbSBarry Smith
440de8f682fSSatish Balay    self.complibs = self.compilers.flibs+self.compilers.cxxlibs+self.compilers.LIBS.split()
441de8f682fSSatish Balay    self.PETSC_EXTERNAL_LIB_BASIC = self.libraries.toStringNoDupes(self.packagelibs+self.complibs)
442de8f682fSSatish Balay
443de8f682fSSatish Balay    self.addMakeMacro('PETSC_EXTERNAL_LIB_BASIC',self.PETSC_EXTERNAL_LIB_BASIC)
4445a21677cSJed Brown    allincludes = petscincludes + includes
4455a21677cSJed Brown    allincludes_install = petscincludes_install + includes
4465a21677cSJed Brown    self.PETSC_CC_INCLUDES = self.headers.toStringNoDupes(allincludes)
4475a21677cSJed Brown    self.PETSC_CC_INCLUDES_INSTALL = self.headers.toStringNoDupes(allincludes_install)
4485a21677cSJed Brown    self.addMakeMacro('PETSC_CC_INCLUDES',self.PETSC_CC_INCLUDES)
4495a21677cSJed Brown    self.addMakeMacro('PETSC_CC_INCLUDES_INSTALL', self.PETSC_CC_INCLUDES_INSTALL)
450cbd5cc15SBarry Smith    if hasattr(self.compilers, 'FC'):
4515a21677cSJed Brown      def modinc(includes):
4520542e31aSBarry Smith        return includes if self.fortran.fortranIsF90 else []
4535a21677cSJed Brown      self.addMakeMacro('PETSC_FC_INCLUDES',self.headers.toStringNoDupes(allincludes,modinc(allincludes)))
4545a21677cSJed Brown      self.addMakeMacro('PETSC_FC_INCLUDES_INSTALL',self.headers.toStringNoDupes(allincludes_install,modinc(allincludes_install)))
455f8833479SBarry Smith
456571416bbSBarry Smith    LIB_DIR = os.path.join(self.installdir.dir,'lib')
457571416bbSBarry Smith    self.addDefine('LIB_DIR','"'+LIB_DIR+'"')
4582d10696aSSatish Balay    # Use build dir here for 'make check' to work before 'make install'
4592d10696aSSatish Balay    PREINSTALL_LIB_DIR = os.path.join(self.petscdir.dir,self.arch.arch,'lib')
460f8833479SBarry Smith
4618143cb67SSatish Balay    self.LIB_NAME_SUFFIX = self.framework.argDB['with-library-name-suffix']
4628143cb67SSatish Balay    self.addMakeMacro('LIB_NAME_SUFFIX', self.LIB_NAME_SUFFIX)
463859f1752SFrancesco Ballarin    self.addDefine('LIB_NAME_SUFFIX', '"'+self.LIB_NAME_SUFFIX+'"')
4648143cb67SSatish Balay
4650f3b21c2SBarry Smith    if self.framework.argDB['with-single-library']:
4668143cb67SSatish Balay      self.petsclib = '-lpetsc'+self.LIB_NAME_SUFFIX
467571416bbSBarry Smith      self.addDefine('USE_SINGLE_LIBRARY', '1')
4688143cb67SSatish Balay      self.addMakeMacro('LIBNAME','${INSTALL_LIB_DIR}/libpetsc${LIB_NAME_SUFFIX}.${AR_LIB_SUFFIX}')
46957cb31baSSatish Balay      self.addMakeMacro('SHLIBS','libpetsc')
4708143cb67SSatish Balay      self.addMakeMacro('PETSC_WITH_EXTERNAL_LIB',self.libraries.toStringNoDupes(['-L'+PREINSTALL_LIB_DIR, '-lpetsc'+self.LIB_NAME_SUFFIX]+self.packagelibs+self.complibs))
471ea820d49SSatish Balay      self.addMakeMacro('PETSC_SYS_LIB','${PETSC_WITH_EXTERNAL_LIB}')
472ea820d49SSatish Balay      self.addMakeMacro('PETSC_VEC_LIB','${PETSC_WITH_EXTERNAL_LIB}')
473ea820d49SSatish Balay      self.addMakeMacro('PETSC_MAT_LIB','${PETSC_WITH_EXTERNAL_LIB}')
474ea820d49SSatish Balay      self.addMakeMacro('PETSC_DM_LIB','${PETSC_WITH_EXTERNAL_LIB}')
475ea820d49SSatish Balay      self.addMakeMacro('PETSC_KSP_LIB','${PETSC_WITH_EXTERNAL_LIB}')
476ea820d49SSatish Balay      self.addMakeMacro('PETSC_SNES_LIB','${PETSC_WITH_EXTERNAL_LIB}')
477ea820d49SSatish Balay      self.addMakeMacro('PETSC_TS_LIB','${PETSC_WITH_EXTERNAL_LIB}')
478b0a7d7e7SSatish Balay      self.addMakeMacro('PETSC_TAO_LIB','${PETSC_WITH_EXTERNAL_LIB}')
479571416bbSBarry Smith    else:
4808143cb67SSatish Balay      pkgs = ['tao', 'ts', 'snes', 'ksp', 'dm', 'mat', 'vec', 'sys']
4818143cb67SSatish Balay      def liblist_basic(libs):
4828143cb67SSatish Balay        return [ '-lpetsc'+lib+self.LIB_NAME_SUFFIX for lib in libs]
4838143cb67SSatish Balay      def liblist(libs):
4848143cb67SSatish Balay        return self.libraries.toStringNoDupes(['-L'+PREINSTALL_LIB_DIR]+liblist_basic(libs)+self.packagelibs+self.complibs)
4858143cb67SSatish Balay      self.petsclib = ' '.join(liblist_basic(pkgs))
4868143cb67SSatish Balay      self.addMakeMacro('PETSC_SYS_LIB', liblist(pkgs[-1:]))
4878143cb67SSatish Balay      self.addMakeMacro('PETSC_VEC_LIB', liblist(pkgs[-2:]))
4888143cb67SSatish Balay      self.addMakeMacro('PETSC_MAT_LIB', liblist(pkgs[-3:]))
4898143cb67SSatish Balay      self.addMakeMacro('PETSC_DM_LIB',  liblist(pkgs[-4:]))
4908143cb67SSatish Balay      self.addMakeMacro('PETSC_KSP_LIB', liblist(pkgs[-5:]))
4918143cb67SSatish Balay      self.addMakeMacro('PETSC_SNES_LIB',liblist(pkgs[-6:]))
4928143cb67SSatish Balay      self.addMakeMacro('PETSC_TS_LIB',  liblist(pkgs[-7:]))
4938143cb67SSatish Balay      self.addMakeMacro('PETSC_TAO_LIB', liblist(pkgs[-8:]))
494571416bbSBarry Smith    self.addMakeMacro('PETSC_LIB','${PETSC_TAO_LIB}')
495571416bbSBarry Smith    self.addMakeMacro('PETSC_LIB_BASIC',self.petsclib)
4960f3b21c2SBarry Smith
497f8833479SBarry Smith    if not os.path.exists(os.path.join(self.petscdir.dir,self.arch.arch,'lib')):
498f8833479SBarry Smith      os.makedirs(os.path.join(self.petscdir.dir,self.arch.arch,'lib'))
499f8833479SBarry Smith
50013f9d092SSatish Balay# add a makefile endtry for display
50113f9d092SSatish Balay    if self.framework.argDB['with-display']:
50213f9d092SSatish Balay      self.addMakeMacro('DISPLAY',self.framework.argDB['with-display'])
50313f9d092SSatish Balay
504f8833479SBarry Smith    # add a makefile entry for configure options
505f8833479SBarry Smith    self.addMakeMacro('CONFIGURE_OPTIONS', self.framework.getOptionsString(['configModules', 'optionsModule']).replace('\"','\\"'))
506a0c7f9aaSSamuel Khuvis
507a0c7f9aaSSamuel Khuvis    if self.framework.argDB['with-tau-perfstubs']:
508a0c7f9aaSSamuel Khuvis      self.addDefine('HAVE_TAU_PERFSTUBS',1)
509f8833479SBarry Smith    return
510f8833479SBarry Smith
511f8833479SBarry Smith  def dumpConfigInfo(self):
512f8833479SBarry Smith    import time
513c6ef1b5bSJed Brown    fd = open(os.path.join(self.arch.arch,'include','petscconfiginfo.h'),'w')
514dc25a686SPierre Jolivet    fd.write('static const char *petscconfigureoptions = "'+self.framework.getOptionsString(['configModules', 'optionsModule']).replace('\"','\\"').replace('\\ ','\\\\ ')+'";\n')
515f8833479SBarry Smith    fd.close()
516f8833479SBarry Smith    return
517f8833479SBarry Smith
5182a4161d9SMatthew G Knepley  def dumpMachineInfo(self):
5192a4161d9SMatthew G Knepley    import platform
520a970bd74SBernhard M. Wiedemann    import datetime
5212a4161d9SMatthew G Knepley    import time
52240373944SSatish Balay    import script
523ca77dbeeSGeoffrey Irving    def escape(s):
524e08ecd42SSatish Balay      return s.replace('"',r'\"').replace(r'\ ',r'\\ ') # novermin
525c6ef1b5bSJed Brown    fd = open(os.path.join(self.arch.arch,'include','petscmachineinfo.h'),'w')
5262a4161d9SMatthew G Knepley    fd.write('static const char *petscmachineinfo = \"\\n\"\n')
5272a4161d9SMatthew G Knepley    fd.write('\"-----------------------------------------\\n\"\n')
528a970bd74SBernhard M. Wiedemann    buildhost = platform.node()
529a970bd74SBernhard M. Wiedemann    if os.environ.get('SOURCE_DATE_EPOCH'):
530a970bd74SBernhard M. Wiedemann      buildhost = "reproducible"
531a970bd74SBernhard M. Wiedemann    buildtime = datetime.datetime.utcfromtimestamp(int(os.environ.get('SOURCE_DATE_EPOCH', time.time())))
532a970bd74SBernhard M. Wiedemann    fd.write('\"Libraries compiled on %s on %s\\n\"\n' % (buildtime, buildhost))
53360acdfe7SSatish Balay    fd.write('\"Machine characteristics: %s\\n\"\n' % (platform.platform()))
5345188cb68SSatish Balay    fd.write('\"Using PETSc directory: %s\\n\"\n' % (escape(self.installdir.petscDir)))
5355188cb68SSatish Balay    fd.write('\"Using PETSc arch: %s\\n\"\n' % (escape(self.installdir.petscArch)))
536cdec380aSBarry Smith    fd.write('\"-----------------------------------------\\n\";\n')
5372a4161d9SMatthew G Knepley    fd.write('static const char *petsccompilerinfo = \"\\n\"\n')
5382a4161d9SMatthew G Knepley    self.setCompilers.pushLanguage(self.languages.clanguage)
5395f27b2e0SBarry Smith    fd.write('\"Using C compiler: %s %s\\n\"\n' % (escape(self.setCompilers.getCompiler()), escape(self.setCompilers.getCompilerFlags())))
5402a4161d9SMatthew G Knepley    self.setCompilers.popLanguage()
5418782282cSMatthew G Knepley    if hasattr(self.compilers, 'FC'):
5422a4161d9SMatthew G Knepley      self.setCompilers.pushLanguage('FC')
5435f27b2e0SBarry Smith      fd.write('\"Using Fortran compiler: %s %s %s\\n\"\n' % (escape(self.setCompilers.getCompiler()), escape(self.setCompilers.getCompilerFlags()), escape(self.setCompilers.CPPFLAGS)))
5442a4161d9SMatthew G Knepley      self.setCompilers.popLanguage()
545cdec380aSBarry Smith    fd.write('\"-----------------------------------------\\n\";\n')
5462a4161d9SMatthew G Knepley    fd.write('static const char *petsccompilerflagsinfo = \"\\n\"\n')
5475a21677cSJed Brown    fd.write('\"Using include paths: %s\\n\"\n' % (escape(self.PETSC_CC_INCLUDES_INSTALL.replace('${PETSC_DIR}', self.installdir.petscDir))))
548cdec380aSBarry Smith    fd.write('\"-----------------------------------------\\n\";\n')
5492a4161d9SMatthew G Knepley    fd.write('static const char *petsclinkerinfo = \"\\n\"\n')
5502a4161d9SMatthew G Knepley    self.setCompilers.pushLanguage(self.languages.clanguage)
551ca77dbeeSGeoffrey Irving    fd.write('\"Using C linker: %s\\n\"\n' % (escape(self.setCompilers.getLinker())))
5522a4161d9SMatthew G Knepley    self.setCompilers.popLanguage()
5538782282cSMatthew G Knepley    if hasattr(self.compilers, 'FC'):
5542a4161d9SMatthew G Knepley      self.setCompilers.pushLanguage('FC')
555ca77dbeeSGeoffrey Irving      fd.write('\"Using Fortran linker: %s\\n\"\n' % (escape(self.setCompilers.getLinker())))
5562a4161d9SMatthew G Knepley      self.setCompilers.popLanguage()
5575188cb68SSatish 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)))
558cdec380aSBarry Smith    fd.write('\"-----------------------------------------\\n\";\n')
5592a4161d9SMatthew G Knepley    fd.close()
5602a4161d9SMatthew G Knepley    return
561b2843cf1SBarry Smith
562b2843cf1SBarry Smith  def configurePrefetch(self):
563b2843cf1SBarry Smith    '''Sees if there are any prefetch functions supported'''
5647fca349cSMatthew G. Knepley    if config.setCompilers.Configure.isSolaris(self.log) or self.framework.argDB['with-ios'] or not self.framework.argDB['with-prefetch']:
56593f78423SSatish Balay      self.addDefine('Prefetch(a,b,c)', ' ')
56693f78423SSatish Balay      return
567ec284106SBarry Smith    self.pushLanguage(self.languages.clanguage)
56810699583SJed Brown    if self.checkLink('#include <xmmintrin.h>', 'void *v = 0;_mm_prefetch((const char*)v,_MM_HINT_NTA);\n'):
56950d8bf02SJed Brown      # The Intel Intrinsics manual [1] specifies the prototype
57050d8bf02SJed Brown      #
57150d8bf02SJed Brown      #   void _mm_prefetch(char const *a, int sel);
57250d8bf02SJed Brown      #
57350d8bf02SJed Brown      # but other vendors seem to insist on using subtly different
57450d8bf02SJed Brown      # prototypes, including void* for the pointer, and an enum for
57550d8bf02SJed Brown      # sel.  These are both reasonable changes, but negatively impact
57650d8bf02SJed Brown      # portability.
57750d8bf02SJed Brown      #
578a8d69d7bSBarry Smith      # [1] https://software.intel.com/file/6373
57950d8bf02SJed Brown      self.addDefine('HAVE_XMMINTRIN_H', 1)
58050d8bf02SJed Brown      self.addDefine('Prefetch(a,b,c)', '_mm_prefetch((const char*)(a),(c))')
58150d8bf02SJed Brown      self.addDefine('PREFETCH_HINT_NTA', '_MM_HINT_NTA')
58250d8bf02SJed Brown      self.addDefine('PREFETCH_HINT_T0',  '_MM_HINT_T0')
58350d8bf02SJed Brown      self.addDefine('PREFETCH_HINT_T1',  '_MM_HINT_T1')
58450d8bf02SJed Brown      self.addDefine('PREFETCH_HINT_T2',  '_MM_HINT_T2')
58550d8bf02SJed Brown    elif self.checkLink('#include <xmmintrin.h>', 'void *v = 0;_mm_prefetch(v,_MM_HINT_NTA);\n'):
58650d8bf02SJed Brown      self.addDefine('HAVE_XMMINTRIN_H', 1)
58750d8bf02SJed Brown      self.addDefine('Prefetch(a,b,c)', '_mm_prefetch((const void*)(a),(c))')
58850d8bf02SJed Brown      self.addDefine('PREFETCH_HINT_NTA', '_MM_HINT_NTA')
58950d8bf02SJed Brown      self.addDefine('PREFETCH_HINT_T0',  '_MM_HINT_T0')
59050d8bf02SJed Brown      self.addDefine('PREFETCH_HINT_T1',  '_MM_HINT_T1')
59150d8bf02SJed Brown      self.addDefine('PREFETCH_HINT_T2',  '_MM_HINT_T2')
59210699583SJed Brown    elif self.checkLink('', 'void *v = 0;__builtin_prefetch(v,0,0);\n'):
59310699583SJed Brown      # From GCC docs: void __builtin_prefetch(const void *addr,int rw,int locality)
59410699583SJed Brown      #
59510699583SJed Brown      #   The value of rw is a compile-time constant one or zero; one
59610699583SJed Brown      #   means that the prefetch is preparing for a write to the memory
59710699583SJed Brown      #   address and zero, the default, means that the prefetch is
59810699583SJed Brown      #   preparing for a read. The value locality must be a compile-time
59910699583SJed Brown      #   constant integer between zero and three. A value of zero means
60010699583SJed Brown      #   that the data has no temporal locality, so it need not be left
60110699583SJed Brown      #   in the cache after the access. A value of three means that the
60210699583SJed Brown      #   data has a high degree of temporal locality and should be left
60310699583SJed Brown      #   in all levels of cache possible. Values of one and two mean,
60410699583SJed Brown      #   respectively, a low or moderate degree of temporal locality.
60510699583SJed Brown      #
60610699583SJed Brown      # Here we adopt Intel's x86/x86-64 naming scheme for the locality
60710699583SJed Brown      # hints.  Using macros for these values in necessary since some
60810699583SJed Brown      # compilers require an enum.
60910699583SJed Brown      self.addDefine('Prefetch(a,b,c)', '__builtin_prefetch((a),(b),(c))')
61010699583SJed Brown      self.addDefine('PREFETCH_HINT_NTA', '0')
61110699583SJed Brown      self.addDefine('PREFETCH_HINT_T0',  '3')
61210699583SJed Brown      self.addDefine('PREFETCH_HINT_T1',  '2')
61310699583SJed Brown      self.addDefine('PREFETCH_HINT_T2',  '1')
614b2843cf1SBarry Smith    else:
615b2843cf1SBarry Smith      self.addDefine('Prefetch(a,b,c)', ' ')
6167d490b44SBarry Smith    self.popLanguage()
617b2843cf1SBarry Smith
61849fe22e6SSatish Balay  def delGenFiles(self):
61949fe22e6SSatish Balay    '''Delete generated files'''
62049fe22e6SSatish Balay    delfile = os.path.join(self.arch.arch,'lib','petsc','conf','files')
62149fe22e6SSatish Balay    try:
62249fe22e6SSatish Balay      os.unlink(delfile)
62349fe22e6SSatish Balay    except: pass
62449fe22e6SSatish Balay
62509bc878fSSatish Balay  def configureAtoll(self):
62609bc878fSSatish Balay    '''Checks if atoll exists'''
62722164b4cSPierre 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'):
62809bc878fSSatish Balay       self.addDefine('HAVE_ATOLL', '1')
62909bc878fSSatish Balay
630648c30bcSBarry Smith  def configureSanitize(self):
631648c30bcSBarry Smith    '''Checks if fsanitize is supported'''
632648c30bcSBarry Smith    if self.checkLink('#if defined(__has_feature)\n#if !__has_feature(address_sanitizer)\nGarbage\n#endif\n#else\nGarbage\n#endif\n'):
633648c30bcSBarry Smith      self.addDefine('HAVE_SANITIZER', '1')
6340ef292d3SStefano Zampini    elif self.checkLink('#if !defined(__SANITIZE_ADDRESS__)\nGarbage\n#endif\n'):
6350ef292d3SStefano Zampini      self.addDefine('HAVE_SANITIZER', '1')
636648c30bcSBarry Smith
6372400fdedSBarry Smith  def configureUnused(self):
6382400fdedSBarry Smith    '''Sees if __attribute((unused)) is supported'''
6391adaff47SSean Farley    if self.framework.argDB['with-ios']:
6402400fdedSBarry Smith      self.addDefine('UNUSED', ' ')
6412400fdedSBarry Smith      return
6422400fdedSBarry Smith    self.pushLanguage(self.languages.clanguage)
64322164b4cSPierre 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'):
6442400fdedSBarry Smith      self.addDefine('UNUSED', '__attribute((unused))')
6452400fdedSBarry Smith    else:
6462400fdedSBarry Smith      self.addDefine('UNUSED', ' ')
6472400fdedSBarry Smith    self.popLanguage()
6482400fdedSBarry Smith
64998ed35c3SBarry Smith  def configureIsatty(self):
65098ed35c3SBarry Smith    '''Check if the Unix C function isatty() works correctly
65198ed35c3SBarry Smith       Actually just assumes it does not work correctly on batch systems'''
65298ed35c3SBarry Smith    if not self.framework.argDB['with-batch']:
65398ed35c3SBarry Smith      self.addDefine('USE_ISATTY',1)
65498ed35c3SBarry Smith
6551ef8df7fSJed Brown  def configureDeprecated(self):
6561ef8df7fSJed Brown    '''Check if __attribute((deprecated)) is supported'''
6577efe37a1SJacob Faibussowitsch    def checkDeprecated(macro_base, src, is_intel):
6587efe37a1SJacob Faibussowitsch      '''
6597efe37a1SJacob Faibussowitsch      run through the various attribute deprecated combinations and define MACRO_BAS(why) to the result
6607efe37a1SJacob Faibussowitsch      it if it compiles.
6617efe37a1SJacob Faibussowitsch
6627efe37a1SJacob Faibussowitsch      If none of the combos work, defines MACRO_BASE(why) as empty
6637efe37a1SJacob Faibussowitsch      '''
6645029be03SJacob Faibussowitsch      full_macro_name = macro_base + '(string_literal_why)'
6657efe37a1SJacob Faibussowitsch      for prefix in ('__attribute__', '__attribute','__declspec'):
6667efe37a1SJacob Faibussowitsch        if prefix == '__declspec':
6677efe37a1SJacob Faibussowitsch          # declspec does not have an extra set of brackets around the arguments
6685029be03SJacob Faibussowitsch          attr_bodies = ('deprecated(string_literal_why)', 'deprecated')
6691ef8df7fSJed Brown        else:
6705029be03SJacob Faibussowitsch          attr_bodies = ('(deprecated(string_literal_why))', '(deprecated)')
6717efe37a1SJacob Faibussowitsch
6727efe37a1SJacob Faibussowitsch        for attr_body in attr_bodies:
6737efe37a1SJacob Faibussowitsch          attr_def = '{}({})'.format(prefix, attr_body)
6747efe37a1SJacob Faibussowitsch          test_src = '\n'.join((
6757efe37a1SJacob Faibussowitsch            '#define {} {}'.format(full_macro_name, attr_def),
6767efe37a1SJacob Faibussowitsch            src.format(macro_base + '("asdasdadsasd")')
6777efe37a1SJacob Faibussowitsch          ))
6787efe37a1SJacob Faibussowitsch          if self.checkCompile(test_src, ''):
6797efe37a1SJacob Faibussowitsch            self.logPrint('configureDeprecated: \'{}\' appears to work'.format(attr_def))
6807efe37a1SJacob Faibussowitsch            if is_intel and '(why)' in attr_body:
6817efe37a1SJacob 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))
6827efe37a1SJacob Faibussowitsch              self.logPrint('*** WE WILL THEREFORE REJECT \'{}\' AND CONTINUE TESTING ***'.format(attr_def))
6837efe37a1SJacob Faibussowitsch              continue
6847efe37a1SJacob Faibussowitsch            self.addDefine(full_macro_name, attr_def)
6857efe37a1SJacob Faibussowitsch            return
6867efe37a1SJacob Faibussowitsch
6877efe37a1SJacob Faibussowitsch      self.addDefine(full_macro_name, ' ')
6887efe37a1SJacob Faibussowitsch      return
6897efe37a1SJacob Faibussowitsch
6907efe37a1SJacob Faibussowitsch    lang = self.languages.clanguage
6917efe37a1SJacob Faibussowitsch    with self.Language(lang):
6927efe37a1SJacob Faibussowitsch      is_intel = self.setCompilers.isIntel(self.getCompiler(lang=lang), self.log)
6935029be03SJacob Faibussowitsch      checkDeprecated('DEPRECATED_FUNCTION_BASE', '{} int myfunc(void) {{ return 1; }}', is_intel)
6945029be03SJacob Faibussowitsch      checkDeprecated('DEPRECATED_TYPEDEF_BASE', 'typedef int my_int {};', is_intel)
6955029be03SJacob Faibussowitsch      checkDeprecated('DEPRECATED_ENUM_BASE', 'enum E {{ oldval {}, newval }};', is_intel)
696a14d4ff0SToby Isaac      checkDeprecated('DEPRECATED_OBJECT_BASE', '{} int x;', is_intel)
6977efe37a1SJacob Faibussowitsch      # I was unable to make a CPP macro that takes the old and new values as separate
6987efe37a1SJacob Faibussowitsch      # arguments and builds the message needed by _Pragma hence the deprecation message is
6997efe37a1SJacob Faibussowitsch      # handled as it is
70005de396fSBarry Smith      if self.checkCompile('#define TEST _Pragma("GCC warning \"Testing _Pragma\"") value'):
7015029be03SJacob Faibussowitsch        self.addDefine('DEPRECATED_MACRO_BASE_(why)', '_Pragma(#why)')
7025029be03SJacob Faibussowitsch        self.addDefine('DEPRECATED_MACRO_BASE(string_literal_why)', self.substPrefix + '_DEPRECATED_MACRO_BASE_(GCC warning string_literal_why)')
70305de396fSBarry Smith      else:
7045029be03SJacob Faibussowitsch        self.addDefine('DEPRECATED_MACRO_BASE(why)', ' ')
7051ef8df7fSJed Brown
70618f41590SBarry Smith  def configureAlign(self):
7077b7fc14bSLisandro Dalcin    '''Check if __attribute(aligned) is supported'''
7087b7fc14bSLisandro Dalcin    code = '''\
709752d89a4SSatish Balaystruct mystruct {int myint;} __attribute((aligned(16)));
7107b7fc14bSLisandro Dalcinchar assert_aligned[(sizeof(struct mystruct)==16)*2-1];
711752d89a4SSatish Balay'''
712752d89a4SSatish Balay    self.pushLanguage(self.languages.clanguage)
7137b7fc14bSLisandro Dalcin    if self.checkCompile(code):
714752d89a4SSatish Balay      self.addDefine('ATTRIBUTEALIGNED(size)', '__attribute((aligned(size)))')
715752d89a4SSatish Balay      self.addDefine('HAVE_ATTRIBUTEALIGNED', 1)
716752d89a4SSatish Balay    else:
7177b7fc14bSLisandro Dalcin      self.framework.logPrint('Incorrect attribute(aligned)')
718752d89a4SSatish Balay      self.addDefine('ATTRIBUTEALIGNED(size)', ' ')
7197b7fc14bSLisandro Dalcin    self.popLanguage()
720752d89a4SSatish Balay    return
72118f41590SBarry Smith
7229800092aSJed Brown  def configureExpect(self):
7239800092aSJed Brown    '''Sees if the __builtin_expect directive is supported'''
7249800092aSJed Brown    self.pushLanguage(self.languages.clanguage)
7259800092aSJed Brown    if self.checkLink('', 'if (__builtin_expect(0,1)) return 1;'):
7269800092aSJed Brown      self.addDefine('HAVE_BUILTIN_EXPECT', 1)
7279800092aSJed Brown    self.popLanguage()
7289800092aSJed Brown
72953c77d0aSJed Brown  def configureFunctionName(self):
730fbfcfee5SBarry Smith    '''Sees if the compiler supports __func__ or a variant.'''
7311ec50b02SJed Brown    def getFunctionName(lang):
732fbfcfee5SBarry Smith      name = '"unknown"'
7331ec50b02SJed Brown      self.pushLanguage(lang)
734b6ff4c76SKarl Rupp      for fname in ['__func__','__FUNCTION__','__extension__ __func__']:
7350117e5a1SSatish Balay        code = "if ("+fname+"[0] != 'm') return 1;"
7360117e5a1SSatish Balay        if self.checkCompile('',code) and self.checkLink('',code):
7370117e5a1SSatish Balay          name = fname
7380117e5a1SSatish Balay          break
7391ec50b02SJed Brown      self.popLanguage()
7401ec50b02SJed Brown      return name
7411ec50b02SJed Brown    langs = []
742628773c9SSatish Balay
743628773c9SSatish Balay    self.addDefine('FUNCTION_NAME_C', getFunctionName('C'))
7445f6e5f85SSatish Balay    if hasattr(self.compilers, 'CXX'):
745628773c9SSatish Balay      self.addDefine('FUNCTION_NAME_CXX', getFunctionName('Cxx'))
74653c77d0aSJed Brown
747753ebd1dSJed Brown  def configureIntptrt(self):
748f18a5f7eSJacob Faibussowitsch    '''Determine what to use for uintptr_t and intptr_t'''
749753ebd1dSJed Brown    def staticAssertSizeMatchesVoidStar(inc,typename):
750753ebd1dSJed Brown      # The declaration is an error if either array size is negative.
751753ebd1dSJed 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
752d26187a0SJed Brown      return self.checkCompile(inc, ('#define STATIC_ASSERT(cond) char negative_length_if_false[2*(!!(cond))-1]\n'
753979939cdSSatish Balay                                     + 'STATIC_ASSERT(sizeof(void*) == sizeof(%s));'%typename))
754f18a5f7eSJacob Faibussowitsch
755f18a5f7eSJacob Faibussowitsch    def generate_uintptr_guesses():
756f18a5f7eSJacob Faibussowitsch      for suff in ('max', '64', '32', '16'):
757f18a5f7eSJacob Faibussowitsch        yield '#include <stdint.h>', 'uint{}_t'.format(suff), 'PRIx{}'.format(suff.upper())
758f18a5f7eSJacob Faibussowitsch      yield '#include <stdlib.h>\n#include <string.h>', 'size_t', 'zx'
759f18a5f7eSJacob Faibussowitsch      yield '', 'unsigned long long', 'llx'
760f18a5f7eSJacob Faibussowitsch      yield '', 'unsigned long', 'lx'
761f18a5f7eSJacob Faibussowitsch      yield '', 'unsigned', 'x'
762f18a5f7eSJacob Faibussowitsch
763f18a5f7eSJacob Faibussowitsch    def generate_intptr_guesses():
764f18a5f7eSJacob Faibussowitsch      for suff in ('max', '64', '32', '16'):
765f18a5f7eSJacob Faibussowitsch        yield '#include <stdint.h>', 'int{}_t'.format(suff), 'PRIx{}'.format(suff.upper())
766f18a5f7eSJacob Faibussowitsch      yield '', 'long long', 'llx'
767f18a5f7eSJacob Faibussowitsch      yield '', 'long', 'lx'
768f18a5f7eSJacob Faibussowitsch      yield '', 'int', 'x'
769f18a5f7eSJacob Faibussowitsch
770f18a5f7eSJacob Faibussowitsch    def check(default_typename, generator):
771f18a5f7eSJacob Faibussowitsch      macro_name = default_typename.upper()
772f18a5f7eSJacob Faibussowitsch      with self.Language(self.languages.clanguage):
773f18a5f7eSJacob Faibussowitsch        if self.checkCompile(
774f18a5f7eSJacob Faibussowitsch            '#include <stdint.h>',
775f18a5f7eSJacob Faibussowitsch            'int x; {type_name} i = ({type_name})&x; (void)i'.format(type_name=default_typename)
776f18a5f7eSJacob Faibussowitsch        ):
777f18a5f7eSJacob Faibussowitsch          typename     = default_typename
778f18a5f7eSJacob Faibussowitsch          print_format = 'PRIxPTR'
779d26187a0SJed Brown        else:
780f18a5f7eSJacob Faibussowitsch          for include, typename, print_format in generator():
781f18a5f7eSJacob Faibussowitsch            if staticAssertSizeMatchesVoidStar(include, typename):
782f18a5f7eSJacob Faibussowitsch              break
783f18a5f7eSJacob Faibussowitsch          else:
784f18a5f7eSJacob Faibussowitsch            raise RuntimeError('Could not find any {} type matching void*'.format(macro_name))
785f18a5f7eSJacob Faibussowitsch      self.addDefine(macro_name         , typename)
786f18a5f7eSJacob Faibussowitsch      self.addDefine(macro_name + '_FMT', '\"#\" ' + print_format)
787f18a5f7eSJacob Faibussowitsch      return
788f18a5f7eSJacob Faibussowitsch
789f18a5f7eSJacob Faibussowitsch    check('uintptr_t', generate_uintptr_guesses)
790f18a5f7eSJacob Faibussowitsch    check('intptr_t', generate_intptr_guesses)
791f18a5f7eSJacob Faibussowitsch    return
792753ebd1dSJed Brown
793ed938b00SJed Brown  def configureRTLDDefault(self):
7947b65ca21SBarry Smith    '''Check for dynamic library feature'''
795bfef2c86SBarry Smith    if self.checkCompile('#include <dlfcn.h>\n void *ptr =  RTLD_DEFAULT;'):
7969fb7294dSPierre Jolivet      self.addDefine('HAVE_RTLD_DEFAULT','1')
797f8833479SBarry Smith    return
798f8833479SBarry Smith
799f8833479SBarry Smith  def configureSolaris(self):
800f8833479SBarry Smith    '''Solaris specific stuff'''
801f8833479SBarry Smith    if os.path.isdir(os.path.join('/usr','ucblib')):
802f8833479SBarry Smith      try:
803f8833479SBarry Smith        flag = getattr(self.setCompilers, self.language[-1]+'SharedLinkerFlag')
804f8833479SBarry Smith      except AttributeError:
805f8833479SBarry Smith        flag = None
806f8833479SBarry Smith      if flag is None:
807f8833479SBarry Smith        self.compilers.LIBS += ' -L/usr/ucblib'
808f8833479SBarry Smith      else:
809f8833479SBarry Smith        self.compilers.LIBS += ' '+flag+'/usr/ucblib'
810f8833479SBarry Smith    return
811f8833479SBarry Smith
8120f64ec89SBarry Smith  def configureDarwin(self):
8130f64ec89SBarry Smith    '''Log brew configuration for Apple systems'''
8140f64ec89SBarry Smith    try:
8150f64ec89SBarry Smith      self.executeShellCommand(['brew', 'config'], log = self.log)
8160f64ec89SBarry Smith      self.executeShellCommand(['brew', 'info', 'gcc'], log = self.log)
8170f64ec89SBarry Smith    except:
8180f64ec89SBarry Smith      pass
8190f64ec89SBarry Smith    return
8200f64ec89SBarry Smith
821f8833479SBarry Smith  def configureLinux(self):
822f8833479SBarry Smith    '''Linux specific stuff'''
8239f15855cSMatthew G Knepley    # TODO: Test for this by mallocing an odd number of floats and checking the address
824f8833479SBarry Smith    self.addDefine('HAVE_DOUBLE_ALIGN_MALLOC', 1)
825f8833479SBarry Smith    return
826f8833479SBarry Smith
827f8833479SBarry Smith  def configureWin32(self):
828f8833479SBarry Smith    '''Win32 non-cygwin specific stuff'''
829f8833479SBarry Smith    kernel32=0
8304e8afd12SMosè Giordano    if self.libraries.add('Kernel32.lib','GetComputerName',prototype='#include <windows.h>', call='GetComputerName(NULL,NULL);'):
831f8833479SBarry Smith      self.addDefine('HAVE_WINDOWS_H',1)
832f8833479SBarry Smith      self.addDefine('HAVE_GETCOMPUTERNAME',1)
833f8833479SBarry Smith      kernel32=1
8344e8afd12SMosè Giordano    elif self.libraries.add('kernel32','GetComputerName',prototype='#include <windows.h>', call='GetComputerName(NULL,NULL);'):
835f8833479SBarry Smith      self.addDefine('HAVE_WINDOWS_H',1)
836f8833479SBarry Smith      self.addDefine('HAVE_GETCOMPUTERNAME',1)
837f8833479SBarry Smith      kernel32=1
838f8833479SBarry Smith    if kernel32:
839eed94e11SSatish Balay      if self.framework.argDB['with-windows-graphics']:
840eed94e11SSatish Balay        self.addDefine('USE_WINDOWS_GRAPHICS',1)
8414e8afd12SMosè Giordano      if self.checkLink('#include <windows.h>','LoadLibrary(0)'):
842f8833479SBarry Smith        self.addDefine('HAVE_LOADLIBRARY',1)
8434e8afd12SMosè Giordano      if self.checkLink('#include <windows.h>','GetProcAddress(0,0)'):
844b50f6d9eSLisandro Dalcin        self.addDefine('HAVE_GETPROCADDRESS',1)
8454e8afd12SMosè Giordano      if self.checkLink('#include <windows.h>','FreeLibrary(0)'):
846b50f6d9eSLisandro Dalcin        self.addDefine('HAVE_FREELIBRARY',1)
8474e8afd12SMosè Giordano      if self.checkLink('#include <windows.h>','GetLastError()'):
848a21658a3SLisandro Dalcin        self.addDefine('HAVE_GETLASTERROR',1)
8494e8afd12SMosè Giordano      if self.checkLink('#include <windows.h>','SetLastError(0)'):
850a21658a3SLisandro Dalcin        self.addDefine('HAVE_SETLASTERROR',1)
8514e8afd12SMosè Giordano      if self.checkLink('#include <windows.h>\n','QueryPerformanceCounter(0);\n'):
852bea725cfSBarry Smith        self.addDefine('USE_MICROSOFT_TIME',1)
8534e8afd12SMosè Giordano    if self.libraries.add('Advapi32.lib','GetUserName',prototype='#include <windows.h>', call='GetUserName(NULL,NULL);'):
854f8833479SBarry Smith      self.addDefine('HAVE_GET_USER_NAME',1)
8554e8afd12SMosè Giordano    elif self.libraries.add('advapi32','GetUserName',prototype='#include <windows.h>', call='GetUserName(NULL,NULL);'):
856f8833479SBarry Smith      self.addDefine('HAVE_GET_USER_NAME',1)
857f8833479SBarry Smith
8584e8afd12SMosè Giordano    if not self.libraries.add('User32.lib','GetDC',prototype='#include <windows.h>',call='GetDC(0);'):
8594e8afd12SMosè Giordano      self.libraries.add('user32','GetDC',prototype='#include <windows.h>',call='GetDC(0);')
8604e8afd12SMosè Giordano    if not self.libraries.add('Gdi32.lib','CreateCompatibleDC',prototype='#include <windows.h>',call='CreateCompatibleDC(0);'):
8614e8afd12SMosè Giordano      self.libraries.add('gdi32','CreateCompatibleDC',prototype='#include <windows.h>',call='CreateCompatibleDC(0);')
862f8833479SBarry Smith
86322164b4cSPierre Jolivet    if not self.checkCompile('#include <sys/types.h>\n','uid_t u;\n(void)u'):
864f8833479SBarry Smith      self.addTypedef('int', 'uid_t')
865f8833479SBarry Smith      self.addTypedef('int', 'gid_t')
86622164b4cSPierre 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'):
867f8833479SBarry Smith      self.framework.addDefine('R_OK', '04')
868f8833479SBarry Smith      self.framework.addDefine('W_OK', '02')
869f8833479SBarry Smith      self.framework.addDefine('X_OK', '01')
870f8833479SBarry Smith    if not self.checkLink('#include <sys/stat.h>\n','int a=0;\nif (S_ISDIR(a)){}\n'):
871f8833479SBarry Smith      self.framework.addDefine('S_ISREG(a)', '(((a)&_S_IFMT) == _S_IFREG)')
872f8833479SBarry Smith      self.framework.addDefine('S_ISDIR(a)', '(((a)&_S_IFMT) == _S_IFDIR)')
8734e8afd12SMosè Giordano    if self.checkCompile('#include <windows.h>\n','LARGE_INTEGER a;\nDWORD b=a.u.HighPart;\n'):
874f8833479SBarry Smith      self.addDefine('HAVE_LARGE_INTEGER_U',1)
875f8833479SBarry Smith
876f8833479SBarry Smith    # Windows requires a Binary file creation flag when creating/opening binary files.  Is a better test in order?
8774e8afd12SMosè Giordano    if self.checkCompile('#include <windows.h>\n#include <fcntl.h>\n', 'int flags = O_BINARY;'):
878f8833479SBarry Smith      self.addDefine('HAVE_O_BINARY',1)
879f8833479SBarry Smith
880f8833479SBarry Smith    if self.compilers.CC.find('win32fe') >= 0:
881ad4212abSSatish Balay      self.addDefine('HAVE_WINDOWS_COMPILERS',1)
882f8833479SBarry Smith      self.addDefine('DIR_SEPARATOR','\'\\\\\'')
883f8833479SBarry Smith      self.addDefine('REPLACE_DIR_SEPARATOR','\'/\'')
884f8833479SBarry Smith      self.addDefine('CANNOT_START_DEBUGGER',1)
8855188cb68SSatish Balay      (petscdir,error,status) = self.executeShellCommand('cygpath -w '+self.installdir.petscDir, log = self.log)
88634531a4dSSatish Balay      self.addDefine('DIR','"'+petscdir.replace('\\','\\\\')+'"')
8875188cb68SSatish Balay      (petscdir,error,status) = self.executeShellCommand('cygpath -m '+self.installdir.petscDir, log = self.log)
888e433681fSSatish Balay      self.addMakeMacro('wPETSC_DIR',petscdir)
8894e00a515SSatish Balay      if self.dataFilesPath.datafilespath:
8904e00a515SSatish Balay        (datafilespath,error,status) = self.executeShellCommand('cygpath -m '+self.dataFilesPath.datafilespath, log = self.log)
8914e00a515SSatish Balay        self.addMakeMacro('DATAFILESPATH',datafilespath)
8924e00a515SSatish Balay
893f8833479SBarry Smith    else:
894f8833479SBarry Smith      self.addDefine('REPLACE_DIR_SEPARATOR','\'\\\\\'')
895f8833479SBarry Smith      self.addDefine('DIR_SEPARATOR','\'/\'')
8965188cb68SSatish Balay      self.addDefine('DIR','"'+self.installdir.petscDir+'"')
8975188cb68SSatish Balay      self.addMakeMacro('wPETSC_DIR',self.installdir.petscDir)
8984e00a515SSatish Balay      if self.dataFilesPath.datafilespath:
8994e00a515SSatish Balay        self.addMakeMacro('DATAFILESPATH',self.dataFilesPath.datafilespath)
9005188cb68SSatish Balay    self.addDefine('ARCH','"'+self.installdir.petscArch+'"')
901f8833479SBarry Smith    return
902f8833479SBarry Smith
903d3d5cfdcSJacob Faibussowitsch  def configureCoverageForLang(self, log_printer_cls, lang, extra_coverage_flags=None, extra_debug_flags=None):
9047ce81a4bSJacob Faibussowitsch    """
9057ce81a4bSJacob Faibussowitsch    Check that a compiler accepts code-coverage flags. If the compiler does accept code-coverage flags
9067ce81a4bSJacob Faibussowitsch    try to set debugging flags equivalent to -Og.
9077ce81a4bSJacob Faibussowitsch
9087ce81a4bSJacob Faibussowitsch    Arguments:
9097ce81a4bSJacob Faibussowitsch    - lang: the language to check the coverage flag for
9107ce81a4bSJacob Faibussowitsch    - extra_coverage_flags: a list of extra flags to use when checking the coverage flags
9117ce81a4bSJacob Faibussowitsch    - extra_debug_flags: a list of extra flags to try when setting debug flags
9127ce81a4bSJacob Faibussowitsch
9137ce81a4bSJacob Faibussowitsch    On success:
9147ce81a4bSJacob Faibussowitsch    - defines PETSC_USE_COVERAGE to 1
9157ce81a4bSJacob Faibussowitsch    """
916d3d5cfdcSJacob Faibussowitsch    log_print = log_printer_cls(self)
9177ce81a4bSJacob Faibussowitsch
9187ce81a4bSJacob Faibussowitsch    def quoted(string):
9197ce81a4bSJacob Faibussowitsch      return string.join(("'", "'"))
9207ce81a4bSJacob Faibussowitsch
9217ce81a4bSJacob Faibussowitsch    def make_flag_list(default, extra):
9227ce81a4bSJacob Faibussowitsch      ret = [default]
9237ce81a4bSJacob Faibussowitsch      if extra is not None:
9247ce81a4bSJacob Faibussowitsch        assert isinstance(extra, list)
9257ce81a4bSJacob Faibussowitsch        ret.extend(extra)
9267ce81a4bSJacob Faibussowitsch      return ret
9277ce81a4bSJacob Faibussowitsch
9287ce81a4bSJacob Faibussowitsch    log_print('Checking coverage flag for language {}'.format(lang))
9299b81490aSJacob Faibussowitsch
9309b81490aSJacob Faibussowitsch    compiler = self.getCompiler(lang=lang)
9319b81490aSJacob Faibussowitsch    if self.setCompilers.isGNU(compiler, self.log):
932d3d5cfdcSJacob Faibussowitsch      is_gnuish = True
9339b81490aSJacob Faibussowitsch    elif self.setCompilers.isClang(compiler, self.log):
934d3d5cfdcSJacob Faibussowitsch      is_gnuish = True
9359b81490aSJacob Faibussowitsch    else:
936d3d5cfdcSJacob Faibussowitsch      is_gnuish = False
9377ce81a4bSJacob Faibussowitsch
9387ce81a4bSJacob Faibussowitsch    # if not gnuish and we don't have a set of extra flags, bail
939d3d5cfdcSJacob Faibussowitsch    if not is_gnuish and extra_coverage_flags is None:
9407ce81a4bSJacob 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)))
9417ce81a4bSJacob Faibussowitsch      return
9427ce81a4bSJacob Faibussowitsch
9437ce81a4bSJacob Faibussowitsch    coverage_flags = make_flag_list('--coverage', extra_coverage_flags)
9447ce81a4bSJacob Faibussowitsch    log_print('Checking set of coverage flags: {}'.format(coverage_flags))
9457ce81a4bSJacob Faibussowitsch
9469b81490aSJacob Faibussowitsch    found = None
9477ce81a4bSJacob Faibussowitsch    with self.Language(lang):
9487ce81a4bSJacob Faibussowitsch      with self.setCompilers.Language(lang):
9497ce81a4bSJacob Faibussowitsch        for flag in coverage_flags:
9509b81490aSJacob Faibussowitsch          # the linker also needs to see the coverage flag
9519b81490aSJacob Faibussowitsch          with self.setCompilers.extraCompilerFlags([flag], compilerOnly=False) as skip_flags:
9529b81490aSJacob Faibussowitsch            if not skip_flags and self.checkRun():
9539b81490aSJacob Faibussowitsch              # flag was accepted
9549b81490aSJacob Faibussowitsch              found = flag
9557ce81a4bSJacob Faibussowitsch              break
9569b81490aSJacob Faibussowitsch
9577ce81a4bSJacob Faibussowitsch          log_print(
9587ce81a4bSJacob Faibussowitsch            'Compiler {} did not accept coverage flag {}'.format(quoted(compiler), quoted(flag))
9597ce81a4bSJacob Faibussowitsch          )
9607ce81a4bSJacob Faibussowitsch
9619b81490aSJacob Faibussowitsch        if found is None:
9629b81490aSJacob Faibussowitsch          log_print(
9639b81490aSJacob Faibussowitsch            'Compiler {} did not accept ANY coverage flags: {}, bailing!'.format(
9649b81490aSJacob Faibussowitsch              quoted(compiler), coverage_flags
9659b81490aSJacob Faibussowitsch            )
9669b81490aSJacob Faibussowitsch          )
9677ce81a4bSJacob Faibussowitsch          return
9687ce81a4bSJacob Faibussowitsch
9699b81490aSJacob Faibussowitsch        # must do this exactly here since:
9709b81490aSJacob Faibussowitsch        #
9719b81490aSJacob Faibussowitsch        # 1. setCompilers.extraCompilerFlags() will reset the compiler flags on __exit__()
9729b81490aSJacob Faibussowitsch        #    (so cannot do it in the loop)
9739b81490aSJacob Faibussowitsch        # 2. we need to set the compiler flag while setCompilers.Language() is still in
9749b81490aSJacob Faibussowitsch        #    effect (so cannot do it outside the with statements)
9759b81490aSJacob Faibussowitsch        self.setCompilers.insertCompilerFlag(flag, False)
9769b81490aSJacob Faibussowitsch
9777ce81a4bSJacob Faibussowitsch    if not self.functions.haveFunction('__gcov_dump'):
9787ce81a4bSJacob Faibussowitsch      self.functions.checkClassify(['__gcov_dump'])
9797ce81a4bSJacob Faibussowitsch
9807ce81a4bSJacob Faibussowitsch    # now check if we can override the optimization level. It is only kosher to do so if
9817ce81a4bSJacob Faibussowitsch    # the user did not explicitly set the optimization flags (via CFLAGS, CXXFLAGS,
9827ce81a4bSJacob Faibussowitsch    # CXXOPTFLAGS, etc). If they have done so, we sternly warn them about their lapse in
9837ce81a4bSJacob Faibussowitsch    # judgement
9847ce81a4bSJacob Faibussowitsch    with self.Language(lang):
9857ce81a4bSJacob Faibussowitsch      compiler_flags = self.getCompilerFlags()
9867ce81a4bSJacob Faibussowitsch
9877ce81a4bSJacob Faibussowitsch    user_set          = 0
9887ce81a4bSJacob Faibussowitsch    allowed_opt_flags = re.compile(r'|'.join((r'-O[01g]', r'-g[1-9]*')))
9897ce81a4bSJacob Faibussowitsch    for flagsname in [self.getCompilerFlagsName(lang), self.compilerFlags.getOptionalFlagsName(lang)]:
9907ce81a4bSJacob Faibussowitsch      if flagsname in self.argDB:
9917ce81a4bSJacob Faibussowitsch        opt_flags = [
9927ce81a4bSJacob Faibussowitsch          f for f in self.compilerFlags.findOptFlags(compiler_flags) if not allowed_opt_flags.match(f)
9937ce81a4bSJacob Faibussowitsch        ]
9947ce81a4bSJacob Faibussowitsch        if opt_flags:
9957ce81a4bSJacob 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)))
9967ce81a4bSJacob Faibussowitsch          user_set = 1
9977ce81a4bSJacob Faibussowitsch          break
9987ce81a4bSJacob Faibussowitsch
99910b490f0SJacob Faibussowitsch    # disable this for now, the warning should be sufficient. If the user still chooses to
100010b490f0SJacob Faibussowitsch    # ignore it, then that's on them
100110b490f0SJacob Faibussowitsch    if 0 and not user_set:
10027ce81a4bSJacob Faibussowitsch      debug_flags = make_flag_list('-Og', extra_debug_flags)
10037ce81a4bSJacob Faibussowitsch      with self.setCompilers.Language(lang):
10047ce81a4bSJacob Faibussowitsch        for flag in debug_flags:
10057ce81a4bSJacob Faibussowitsch          try:
10067ce81a4bSJacob Faibussowitsch            self.setCompilers.addCompilerFlag(flag)
10077ce81a4bSJacob Faibussowitsch          except RuntimeError:
10087ce81a4bSJacob Faibussowitsch            continue
10097ce81a4bSJacob Faibussowitsch          break
10107ce81a4bSJacob Faibussowitsch
10117ce81a4bSJacob Faibussowitsch    self.addDefine('USE_COVERAGE', 1)
10127ce81a4bSJacob Faibussowitsch    return
10137ce81a4bSJacob Faibussowitsch
10149b81490aSJacob Faibussowitsch  def configureCoverage(self):
1015d3d5cfdcSJacob Faibussowitsch    """
1016d3d5cfdcSJacob Faibussowitsch    Configure coverage for all available languages.
1017d3d5cfdcSJacob Faibussowitsch
1018d3d5cfdcSJacob Faibussowitsch    If user did not request coverage, this function does nothing and returns immediatel.
1019d3d5cfdcSJacob Faibussowitsch    Therefore the following only apply to the case where the user requested coverage.
1020d3d5cfdcSJacob Faibussowitsch
1021d3d5cfdcSJacob Faibussowitsch    On success:
1022d3d5cfdcSJacob Faibussowitsch    - defines PETSC_USE_COVERAGE to 1
1023d3d5cfdcSJacob Faibussowitsch
1024d3d5cfdcSJacob Faibussowitsch    On failure:
1025d3d5cfdcSJacob Faibussowitsch    - If no compilers supported the coverage flag, throws RuntimeError
1026d3d5cfdcSJacob Faibussowitsch    -
1027d3d5cfdcSJacob Faibussowitsch    """
1028d3d5cfdcSJacob Faibussowitsch    class LogPrinter:
1029d3d5cfdcSJacob Faibussowitsch      def __init__(self, cfg):
1030d3d5cfdcSJacob Faibussowitsch        self.cfg = cfg
1031d3d5cfdcSJacob Faibussowitsch        try:
1032d3d5cfdcSJacob Faibussowitsch          import inspect
1033d3d5cfdcSJacob Faibussowitsch
1034d3d5cfdcSJacob Faibussowitsch          calling_func_stack = inspect.stack()[1]
1035d3d5cfdcSJacob Faibussowitsch          if sys.version_info >= (3, 5):
1036d3d5cfdcSJacob Faibussowitsch            func_name = calling_func_stack.function
1037d3d5cfdcSJacob Faibussowitsch          else:
1038d3d5cfdcSJacob Faibussowitsch            func_name = calling_func_stack[3]
1039d3d5cfdcSJacob Faibussowitsch        except:
1040d3d5cfdcSJacob Faibussowitsch          func_name = 'Unknown'
1041d3d5cfdcSJacob Faibussowitsch        self.fmt_str = func_name + '(): {}'
1042d3d5cfdcSJacob Faibussowitsch
1043d3d5cfdcSJacob Faibussowitsch      def __call__(self, msg, *args, **kwargs):
1044d3d5cfdcSJacob Faibussowitsch        return self.cfg.logPrint(self.fmt_str.format(msg), *args, **kwargs)
1045d3d5cfdcSJacob Faibussowitsch
1046d3d5cfdcSJacob Faibussowitsch    argdb_flag = 'with-coverage'
1047d3d5cfdcSJacob Faibussowitsch    log_print  = LogPrinter(self)
1048d3d5cfdcSJacob Faibussowitsch    if not self.argDB[argdb_flag]:
1049d3d5cfdcSJacob Faibussowitsch      log_print('coverage was disabled from command line or default')
1050d3d5cfdcSJacob Faibussowitsch      return
1051d3d5cfdcSJacob Faibussowitsch
1052d3d5cfdcSJacob Faibussowitsch    tested_langs = []
10539b81490aSJacob Faibussowitsch    for LANG in ['C', 'Cxx', 'CUDA', 'HIP', 'SYCL', 'FC']:
10549b81490aSJacob Faibussowitsch      compilerName = LANG.upper() if LANG in {'Cxx', 'FC'} else LANG + 'C'
10559b81490aSJacob Faibussowitsch      if hasattr(self.setCompilers, compilerName):
10569b81490aSJacob Faibussowitsch        kwargs = {}
10579b81490aSJacob Faibussowitsch        if LANG in {'CUDA'}:
10589b81490aSJacob Faibussowitsch          # nvcc preprocesses the base file into a bunch of intermediate files, which are
10599b81490aSJacob Faibussowitsch          # then compiled by the host compiler. Why is this a problem?  Because the
10609b81490aSJacob Faibussowitsch          # generated coverage data is based on these preprocessed source files! So gcov
10619b81490aSJacob Faibussowitsch          # tries to read it later, but since its in the tmp directory it cannot. Thus we
10629b81490aSJacob Faibussowitsch          # need to keep them around (in a place we know about).
10639b81490aSJacob Faibussowitsch          nvcc_tmp_dir = os.path.join(self.petscdir.dir, self.arch.arch, 'nvcc_tmp')
10649b81490aSJacob Faibussowitsch          try:
10659b81490aSJacob Faibussowitsch            os.mkdir(nvcc_tmp_dir)
10669b81490aSJacob Faibussowitsch          except FileExistsError:
10679b81490aSJacob Faibussowitsch            pass
10689b81490aSJacob Faibussowitsch          kwargs['extra_coverage_flags'] = [
10699b81490aSJacob Faibussowitsch            '-Xcompiler --coverage -Xcompiler -fPIC --keep --keep-dir={}'.format(nvcc_tmp_dir)
10709b81490aSJacob Faibussowitsch          ]
10719b81490aSJacob Faibussowitsch          if self.kokkos.found:
10729b81490aSJacob Faibussowitsch            # yet again the kokkos nvcc_wrapper goes out of its way to be as useless as
10739b81490aSJacob Faibussowitsch            # possible. Its default arch (sm_35) is actually too low to compile kokkos,
10749b81490aSJacob Faibussowitsch            # for whatever reason this works if you dont use the --keep and --keep-dir
10759b81490aSJacob Faibussowitsch            # flags above.
10769b81490aSJacob Faibussowitsch            kwargs['extra_coverage_flags'].append('-arch=native')
10779b81490aSJacob Faibussowitsch            kwargs['extra_debug_flags'] = ['-Xcompiler -Og']
1078d3d5cfdcSJacob Faibussowitsch        tested_langs.append(LANG)
1079d3d5cfdcSJacob Faibussowitsch        self.executeTest(self.configureCoverageForLang, args=[LogPrinter, LANG], kargs=kwargs)
1080d3d5cfdcSJacob Faibussowitsch
1081d3d5cfdcSJacob Faibussowitsch    if not self.defines.get('USE_COVERAGE'):
1082d3d5cfdcSJacob Faibussowitsch      # coverage was requested but no compilers accepted it, this is an error
1083d3d5cfdcSJacob Faibussowitsch      raise RuntimeError(
1084d3d5cfdcSJacob Faibussowitsch        'Coverage was requested (--{}={}) but none of the compilers supported it:\n{}\n'.format(
1085d3d5cfdcSJacob Faibussowitsch          argdb_flag, self.argDB[argdb_flag],
1086d3d5cfdcSJacob Faibussowitsch          '\n'.join(['  - {} ({})'.format(self.getCompiler(lang=lang), lang) for lang in tested_langs])
1087d3d5cfdcSJacob Faibussowitsch        )
1088d3d5cfdcSJacob Faibussowitsch      )
1089d3d5cfdcSJacob Faibussowitsch
1090d3d5cfdcSJacob Faibussowitsch    return
1091d3d5cfdcSJacob Faibussowitsch    # Disabled for now, since this does not really work. It solves the problem of
1092d3d5cfdcSJacob Faibussowitsch    # "undefined reference to __gcov_flush()" but if we add -lgcov we get:
1093d3d5cfdcSJacob Faibussowitsch    #
1094d3d5cfdcSJacob Faibussowitsch    # duplicate symbol '___gcov_reset' in:
1095d3d5cfdcSJacob Faibussowitsch    #     /Library/.../libclang_rt.profile_osx.a(GCDAProfiling.c.o)
1096d3d5cfdcSJacob Faibussowitsch    #     /opt/.../libgcov.a(_gcov_reset.o)
1097d3d5cfdcSJacob Faibussowitsch    # duplicate symbol '___gcov_dump' in:
1098d3d5cfdcSJacob Faibussowitsch    #     /opt/.../libgcov.a(_gcov_dump.o)
1099d3d5cfdcSJacob Faibussowitsch    #     /Library/.../libclang_rt.profile_osx.a(GCDAProfiling.c.o)
1100d3d5cfdcSJacob Faibussowitsch    # duplicate symbol '___gcov_fork' in:
1101d3d5cfdcSJacob Faibussowitsch    #     /opt/.../libgcov.a(_gcov_fork.o)
1102d3d5cfdcSJacob Faibussowitsch    #     /Library/.../libclang_rt.profile_osx.a(GCDAProfiling.c.o)
1103d3d5cfdcSJacob Faibussowitsch    #
1104d3d5cfdcSJacob Faibussowitsch    # I don't know how to solve this.
1105d3d5cfdcSJacob Faibussowitsch
1106d3d5cfdcSJacob Faibussowitsch    log_print('Checking if compilers can cross-link disparate coverage libraries')
1107d3d5cfdcSJacob Faibussowitsch    # At least one of the compilers has coverage enabled. Now need to make sure multiple
1108d3d5cfdcSJacob Faibussowitsch    # code coverage impls work together, specifically when using clang C/C++ compiler with
1109d3d5cfdcSJacob Faibussowitsch    # gfortran.
1110d3d5cfdcSJacob Faibussowitsch    if not hasattr(self.setCompilers, 'FC'):
1111d3d5cfdcSJacob Faibussowitsch      log_print('No fortran compiler detected. No need to check cross-linking!')
1112d3d5cfdcSJacob Faibussowitsch      return
1113d3d5cfdcSJacob Faibussowitsch
1114d3d5cfdcSJacob Faibussowitsch    c_lang = self.languages.clanguage
1115d3d5cfdcSJacob Faibussowitsch    if not self.setCompilers.isClang(self.getCompiler(lang=c_lang), self.log):
1116d3d5cfdcSJacob Faibussowitsch      # must be GCC
1117d3d5cfdcSJacob 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')))
1118d3d5cfdcSJacob Faibussowitsch      return
1119d3d5cfdcSJacob Faibussowitsch
1120d3d5cfdcSJacob Faibussowitsch    # If we are here we:
1121d3d5cfdcSJacob Faibussowitsch    #   1. Have both C/C++ compiler and fortran compiler
1122d3d5cfdcSJacob Faibussowitsch    #   2. The C/C++ compiler is *not* the same as the fortran compiler (unless we start
1123d3d5cfdcSJacob Faibussowitsch    #      using flang)
1124d3d5cfdcSJacob Faibussowitsch    #
1125d3d5cfdcSJacob Faibussowitsch    # Now we check if we can cross-link
1126d3d5cfdcSJacob Faibussowitsch    def can_cross_link(**kwargs):
1127d3d5cfdcSJacob Faibussowitsch      f_body = "      subroutine foo()\n      print*,'testing'\n      return\n      end\n"
1128d3d5cfdcSJacob Faibussowitsch      c_body = "int main() { }"
1129d3d5cfdcSJacob Faibussowitsch
1130d3d5cfdcSJacob Faibussowitsch      return self.compilers.checkCrossLink(
1131d3d5cfdcSJacob Faibussowitsch        f_body, c_body, language1='FC', language2=c_lang, extralibs=self.compilers.flibs, **kwargs
1132d3d5cfdcSJacob Faibussowitsch      )
1133d3d5cfdcSJacob Faibussowitsch
1134d3d5cfdcSJacob Faibussowitsch    log_print('Trying to cross-link WITHOUT extra libs')
1135d3d5cfdcSJacob Faibussowitsch    if can_cross_link():
1136d3d5cfdcSJacob Faibussowitsch      log_print('Successfully cross-linked WITHOUT extra libs')
1137d3d5cfdcSJacob Faibussowitsch      # success, we already can cross-link
1138d3d5cfdcSJacob Faibussowitsch      return
1139d3d5cfdcSJacob Faibussowitsch
1140d3d5cfdcSJacob Faibussowitsch    extra_libs = ['-lgcov']
1141d3d5cfdcSJacob Faibussowitsch    log_print('Trying to cross-link with extra libs: {}'.format(extra_libs))
1142d3d5cfdcSJacob Faibussowitsch    if can_cross_link(extraObjs=extra_libs):
1143d3d5cfdcSJacob Faibussowitsch      log_print(
1144d3d5cfdcSJacob Faibussowitsch        'Successfully cross-linked using extra libs: {}, adding them to LIBS'.format(extra_libs)
1145d3d5cfdcSJacob Faibussowitsch      )
1146d3d5cfdcSJacob Faibussowitsch      self.setCompilers.LIBS += ' ' + ' '.join(extra_libs)
1147d3d5cfdcSJacob Faibussowitsch    else:
1148d3d5cfdcSJacob Faibussowitsch      # maybe should be an error?
1149d3d5cfdcSJacob 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))
11509b81490aSJacob Faibussowitsch    return
11519b81490aSJacob Faibussowitsch
11527ce81a4bSJacob Faibussowitsch  def configureCoverageExecutable(self):
11537ce81a4bSJacob Faibussowitsch    """
11547ce81a4bSJacob Faibussowitsch    Check that a code-coverage collecting tool exists and is on PATH.
11557ce81a4bSJacob Faibussowitsch
11567ce81a4bSJacob Faibussowitsch    On success:
11577ce81a4bSJacob Faibussowitsch    - Adds PETSC_COVERAGE_EXEC make macro containing the full path to the coverage tool executable.
11587ce81a4bSJacob Faibussowitsch
11597ce81a4bSJacob Faibussowitsch    Raises RuntimeError if:
11607ce81a4bSJacob Faibussowitsch    - User explicitly requests auto-detection of the coverage tool from command line, and this
11617ce81a4bSJacob Faibussowitsch      routine fails to guess the suitable tool name.
11627ce81a4bSJacob Faibussowitsch    - The routine fails to find the tool, and --with-coverage is true
11637ce81a4bSJacob Faibussowitsch    """
11647ce81a4bSJacob Faibussowitsch    def log_print(msg, *args, **kwargs):
11657ce81a4bSJacob Faibussowitsch      self.logPrint('checkCoverage: '+str(msg), *args, **kwargs)
11667ce81a4bSJacob Faibussowitsch      return
11677ce81a4bSJacob Faibussowitsch
11687ce81a4bSJacob Faibussowitsch    def quoted(string):
11697ce81a4bSJacob Faibussowitsch      return string.join(("'", "'"))
11707ce81a4bSJacob Faibussowitsch
11717ce81a4bSJacob Faibussowitsch    required         = bool(self.argDB['with-coverage'])
11727ce81a4bSJacob Faibussowitsch    arg_opt          = self.argDB['with-coverage-exec']
11737ce81a4bSJacob Faibussowitsch    use_default_path = True
11747ce81a4bSJacob Faibussowitsch    search_path      = ''
11757ce81a4bSJacob Faibussowitsch
11767ce81a4bSJacob Faibussowitsch    log_print('{} to find an executable'.format('REQUIRED' if required else 'NOT required'))
11777ce81a4bSJacob Faibussowitsch    if arg_opt in {'auto', 'default-auto', '1'}:
11787ce81a4bSJacob Faibussowitsch      # detect it based on the C language compiler, hopefully this does not clash!
1179095fb05fSJacob Faibussowitsch      lang     = self.setCompilers.languages.clanguage
1180095fb05fSJacob Faibussowitsch      compiler = self.getCompiler(lang=lang)
11817ce81a4bSJacob Faibussowitsch      log_print('User did not explicitly set coverage exec (got {}), trying to auto-detect based on compiler {}'.format(quoted(arg_opt), quoted(compiler)))
11827ce81a4bSJacob Faibussowitsch      if self.setCompilers.isGNU(compiler, self.log):
1183095fb05fSJacob Faibussowitsch        compiler_version_re = re.compile(r'[gG][cC\+\-]+[0-9]* \(.+\) (\d+)\.(\d+)\.(\d+)')
11847ce81a4bSJacob Faibussowitsch        exec_names          = ['gcov']
11857ce81a4bSJacob Faibussowitsch      elif self.setCompilers.isClang(compiler, self.log):
1186095fb05fSJacob Faibussowitsch        compiler_version_re = re.compile(r'clang version (\d+)\.(\d+)\.(\d+)')
11877ce81a4bSJacob Faibussowitsch        exec_names          = ['llvm-cov']
11887ce81a4bSJacob Faibussowitsch        if self.setCompilers.isDarwin(self.log):
11897ce81a4bSJacob Faibussowitsch          # macOS masquerades llvm-cov as just 'gcov', so we add this to the list in case
11907ce81a4bSJacob Faibussowitsch          # bare llvm-cov does not work
11917ce81a4bSJacob Faibussowitsch          exec_names.append('gcov')
11927ce81a4bSJacob Faibussowitsch      elif arg_opt == 'default-auto' and not required:
11937ce81a4bSJacob Faibussowitsch        # default-auto implies the user did not set it via command line!
11947ce81a4bSJacob 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)))
11957ce81a4bSJacob Faibussowitsch        return
11967ce81a4bSJacob Faibussowitsch      else:
11977ce81a4bSJacob Faibussowitsch        # implies 'auto' explicitly set by user, or we were required to find
11987ce81a4bSJacob Faibussowitsch        # something. either way we should error
11997ce81a4bSJacob Faibussowitsch        raise RuntimeError('Could not auto-detect coverage tool for {}, please set coverage tool name explicitly'.format(quoted(compiler)))
1200095fb05fSJacob Faibussowitsch
1201095fb05fSJacob Faibussowitsch      try:
1202095fb05fSJacob Faibussowitsch        compiler_version_str = self.compilerFlags.version[lang]
1203095fb05fSJacob Faibussowitsch      except KeyError:
1204095fb05fSJacob Faibussowitsch        compiler_version_str = 'Unknown'
1205095fb05fSJacob Faibussowitsch
1206095fb05fSJacob Faibussowitsch      log_print('Searching version string {} (for compiler {}) using pattern {}'.format(quoted(compiler_version_str), quoted(compiler), quoted(compiler_version_re.pattern)))
1207095fb05fSJacob Faibussowitsch      compiler_version = compiler_version_re.search(compiler_version_str)
1208095fb05fSJacob Faibussowitsch      if compiler_version is not None:
1209095fb05fSJacob Faibussowitsch        log_print('Found major = {}, minor = {}, patch = {}'.format(compiler_version.group(1), compiler_version.group(2), compiler_version.group(3)))
1210095fb05fSJacob Faibussowitsch        # form [llvm-cov-14, llvm-cov-14.0, llvm-cov, etc.]
1211095fb05fSJacob Faibussowitsch        cov_exec_name = exec_names[0]
1212095fb05fSJacob Faibussowitsch        exec_names    = [
1213095fb05fSJacob Faibussowitsch          # llvm-cov-14
1214095fb05fSJacob Faibussowitsch          '{}-{}'.format(cov_exec_name, compiler_version.group(1)),
1215095fb05fSJacob Faibussowitsch           # llvm-cov-14.0
1216095fb05fSJacob Faibussowitsch          '{}-{}.{}'.format(cov_exec_name, compiler_version.group(1), compiler_version.group(2))
1217095fb05fSJacob Faibussowitsch        ] + exec_names
12187ce81a4bSJacob Faibussowitsch    else:
12197ce81a4bSJacob Faibussowitsch      log_print('User explicitly set coverage exec as {}'.format(quoted(arg_opt)))
12207ce81a4bSJacob Faibussowitsch      par_dir = os.path.dirname(arg_opt)
12217ce81a4bSJacob Faibussowitsch      if os.path.exists(par_dir):
12227ce81a4bSJacob Faibussowitsch        # arg_opt is path-like, we should only search the provided directory when we go
12237ce81a4bSJacob Faibussowitsch        # looking for the tool
12247ce81a4bSJacob Faibussowitsch        use_default_path = False
12257ce81a4bSJacob Faibussowitsch        search_path      = par_dir
12267ce81a4bSJacob Faibussowitsch      exec_names = [arg_opt]
12277ce81a4bSJacob Faibussowitsch
12287ce81a4bSJacob Faibussowitsch    make_macro_name = 'PETSC_COVERAGE_EXEC'
12297ce81a4bSJacob Faibussowitsch    log_print('Checking for coverage tool(s):\n{}'.format('\n'.join('- '+t for t in exec_names)))
12307ce81a4bSJacob Faibussowitsch    found_exec = self.getExecutables(
12317ce81a4bSJacob Faibussowitsch      exec_names,
12327ce81a4bSJacob Faibussowitsch      path=search_path, getFullPath=True, useDefaultPath=use_default_path, resultName=make_macro_name
12337ce81a4bSJacob Faibussowitsch    )
12347ce81a4bSJacob Faibussowitsch
12357ce81a4bSJacob Faibussowitsch    if found_exec is None:
12367ce81a4bSJacob Faibussowitsch      # didn't find the coverage tool
12377ce81a4bSJacob Faibussowitsch      if required:
12387ce81a4bSJacob Faibussowitsch        raise RuntimeError('Coverage tool(s) {} could not be found. Please provide explicit path to coverage tool'.format(exec_names))
12397ce81a4bSJacob Faibussowitsch      return
12407ce81a4bSJacob Faibussowitsch
12417ce81a4bSJacob Faibussowitsch    found_exec_name = os.path.basename(found_exec)
12427ce81a4bSJacob Faibussowitsch    if 'llvm-cov' in found_exec_name and 'gcov' not in found_exec_name:
12437ce81a4bSJacob Faibussowitsch      # llvm-cov needs to be called as 'llvm-cov gcov' to work
12447ce81a4bSJacob Faibussowitsch      self.addMakeMacro(make_macro_name, found_exec + ' gcov')
12457ce81a4bSJacob Faibussowitsch    return
12467ce81a4bSJacob Faibussowitsch
1247689a5dfaSJacob Faibussowitsch  def configureStrictPetscErrorCode(self):
1248689a5dfaSJacob Faibussowitsch    """
1249689a5dfaSJacob Faibussowitsch    Enables or disables strict PetscErrorCode checking.
1250689a5dfaSJacob Faibussowitsch
1251689a5dfaSJacob Faibussowitsch    If --with-strict-petscerrorcode = 1:
1252689a5dfaSJacob Faibussowitsch    - defines PETSC_USE_STRICT_PETSCERRORCODE to 1
1253689a5dfaSJacob Faibussowitsch
1254689a5dfaSJacob Faibussowitsch    Else:
1255689a5dfaSJacob Faibussowitsch    - deletes any prior PETSC_USE_STRICT_PETSCERRORCODE definitions (if they exist)
1256689a5dfaSJacob Faibussowitsch    """
1257689a5dfaSJacob Faibussowitsch    define_name = 'USE_STRICT_PETSCERRORCODE'
1258689a5dfaSJacob Faibussowitsch    if self.argDB['with-strict-petscerrorcode']:
1259689a5dfaSJacob Faibussowitsch      self.addDefine(define_name, 1)
1260689a5dfaSJacob Faibussowitsch    else:
1261689a5dfaSJacob Faibussowitsch      # in case it was somehow added previously
1262689a5dfaSJacob Faibussowitsch      self.delDefine(define_name)
1263689a5dfaSJacob Faibussowitsch    return
1264689a5dfaSJacob Faibussowitsch
1265f8833479SBarry Smith#-----------------------------------------------------------------------------------------------------
1266b10d012aSSatish Balay  def configureCygwinBrokenPipe(self):
1267b10d012aSSatish Balay    '''Cygwin version <= 1.7.18 had issues with pipes and long commands invoked from gnu-make
1268b10d012aSSatish Balay    http://cygwin.com/ml/cygwin/2013-05/msg00340.html '''
12697fca349cSMatthew G. Knepley    if config.setCompilers.Configure.isCygwin(self.log):
1270b10d012aSSatish Balay      import platform
1271b10d012aSSatish Balay      import re
1272b10d012aSSatish Balay      r=re.compile("([0-9]+).([0-9]+).([0-9]+)")
1273b10d012aSSatish Balay      m=r.match(platform.release())
1274b10d012aSSatish Balay      major=int(m.group(1))
1275b10d012aSSatish Balay      minor=int(m.group(2))
1276b10d012aSSatish Balay      subminor=int(m.group(3))
1277b10d012aSSatish Balay      if ((major < 1) or (major == 1 and minor < 7) or (major == 1 and minor == 7 and subminor <= 18)):
1278b10d012aSSatish Balay        self.addMakeMacro('PETSC_CYGWIN_BROKEN_PIPE','1')
1279b10d012aSSatish Balay    return
1280b10d012aSSatish Balay
1281b10d012aSSatish Balay#-----------------------------------------------------------------------------------------------------
1282569865ddSSatish Balay  def configureDefaultArch(self):
1283af0996ceSBarry Smith    conffile = os.path.join('lib','petsc','conf', 'petscvariables')
1284569865ddSSatish Balay    if self.framework.argDB['with-default-arch']:
1285c6ef1b5bSJed Brown      fd = open(conffile, 'w')
1286569865ddSSatish Balay      fd.write('PETSC_ARCH='+self.arch.arch+'\n')
1287da93591fSBarry Smith      fd.write('PETSC_DIR='+self.petscdir.dir+'\n')
1288b9b902edSJed Brown      fd.write('include '+os.path.join('$(PETSC_DIR)','$(PETSC_ARCH)','lib','petsc','conf','petscvariables')+'\n')
1289569865ddSSatish Balay      fd.close()
1290569865ddSSatish Balay      self.framework.actions.addArgument('PETSc', 'Build', 'Set default architecture to '+self.arch.arch+' in '+conffile)
1291569865ddSSatish Balay    elif os.path.isfile(conffile):
1292569865ddSSatish Balay      try:
1293569865ddSSatish Balay        os.unlink(conffile)
1294569865ddSSatish Balay      except:
1295569865ddSSatish Balay        raise RuntimeError('Unable to remove file '+conffile+'. Did a different user create it?')
1296569865ddSSatish Balay    return
1297569865ddSSatish Balay
1298569865ddSSatish Balay#-----------------------------------------------------------------------------------------------------
1299f8833479SBarry Smith  def configureScript(self):
1300f8833479SBarry Smith    '''Output a script in the conf directory which will reproduce the configuration'''
1301f8833479SBarry Smith    import nargs
1302495bf1a9SSatish Balay    import sys
1303af0996ceSBarry Smith    scriptName = os.path.join(self.arch.arch,'lib','petsc','conf', 'reconfigure-'+self.arch.arch+'.py')
1304f8833479SBarry Smith    args = dict([(nargs.Arg.parseArgument(arg)[0], arg) for arg in self.framework.clArgs])
1305e97fc2efSSatish Balay    if 'with-clean' in args:
1306e97fc2efSSatish Balay      del args['with-clean']
1307d418e2d7SSatish Balay    if 'force' in args:
1308d418e2d7SSatish Balay      del args['force']
1309f8833479SBarry Smith    if 'configModules' in args:
13101063a081SSatish Balay      if nargs.Arg.parseArgument(args['configModules'])[1] == 'PETSc.Configure':
1311f8833479SBarry Smith        del args['configModules']
1312f8833479SBarry Smith    if 'optionsModule' in args:
131323a19ef1SSatish Balay      if nargs.Arg.parseArgument(args['optionsModule'])[1] == 'config.compilerOptions':
1314f8833479SBarry Smith        del args['optionsModule']
1315f8833479SBarry Smith    if not 'PETSC_ARCH' in args:
13161063a081SSatish Balay      args['PETSC_ARCH'] = 'PETSC_ARCH='+str(self.arch.arch)
1317c6ef1b5bSJed Brown    f = open(scriptName, 'w')
1318495bf1a9SSatish Balay    f.write('#!'+sys.executable+'\n')
1319f8833479SBarry Smith    f.write('if __name__ == \'__main__\':\n')
1320f8833479SBarry Smith    f.write('  import sys\n')
13217561c02cSSatish Balay    f.write('  import os\n')
13227561c02cSSatish Balay    f.write('  sys.path.insert(0, os.path.abspath(\'config\'))\n')
1323f8833479SBarry Smith    f.write('  import configure\n')
13241063a081SSatish Balay    # pretty print repr(args.values())
13251063a081SSatish Balay    f.write('  configure_options = [\n')
13268bec23c5SJed Brown    for itm in sorted(args.values()):
13271063a081SSatish Balay      f.write('    \''+str(itm)+'\',\n')
13281063a081SSatish Balay    f.write('  ]\n')
1329f8833479SBarry Smith    f.write('  configure.petsc_configure(configure_options)\n')
1330f8833479SBarry Smith    f.close()
1331f8833479SBarry Smith    try:
13325b6bfdb9SJed Brown      os.chmod(scriptName, 0o775)
13335b6bfdb9SJed Brown    except OSError as e:
1334f8833479SBarry Smith      self.framework.logPrint('Unable to make reconfigure script executable:\n'+str(e))
1335f8833479SBarry Smith    self.framework.actions.addArgument('PETSc', 'File creation', 'Created '+scriptName+' for automatic reconfiguration')
1336f8833479SBarry Smith    return
1337f8833479SBarry Smith
1338f8833479SBarry Smith  def configureInstall(self):
1339f8833479SBarry Smith    '''Setup the directories for installation'''
1340f8833479SBarry Smith    if self.framework.argDB['prefix']:
13415b4fc442SVaclav Hapla      self.addMakeRule('print_mesg_after_build','',
13425b4fc442SVaclav Hapla       ['-@echo "========================================="',
13435b4fc442SVaclav Hapla        '-@echo "Now to install the libraries do:"',
13445b4fc442SVaclav Hapla        '-@echo "%s${MAKE_USER} PETSC_DIR=${PETSC_DIR} PETSC_ARCH=${PETSC_ARCH} install"' % self.installdir.installSudo,
1345315b77e6SSatish Balay        '-@echo "========================================="'])
1346f8833479SBarry Smith    else:
13475b4fc442SVaclav Hapla      self.addMakeRule('print_mesg_after_build','',
13485b4fc442SVaclav Hapla       ['-@echo "========================================="',
13495b4fc442SVaclav Hapla        '-@echo "Now to check if the libraries are working do:"',
13505b4fc442SVaclav Hapla        '-@echo "${MAKE_USER} PETSC_DIR=${PETSC_DIR} PETSC_ARCH=${PETSC_ARCH} check"',
1351315b77e6SSatish Balay        '-@echo "========================================="'])
1352f8833479SBarry Smith      return
1353f8833479SBarry Smith
135428bb2e72SSatish Balay  def postProcessPackages(self):
135528bb2e72SSatish Balay    postPackages=[]
135628bb2e72SSatish Balay    for i in self.framework.packages:
135728bb2e72SSatish Balay      if hasattr(i,'postProcess'): postPackages.append(i)
135828bb2e72SSatish Balay    if postPackages:
1359f0b74427SPierre Jolivet      # ctetgen needs PETSc conf files. so attempt to create them early
1360a77eb93bSSatish Balay      self.framework.dumpConfFiles()
1361d9293e7bSBarry Smith      # tacky fix for dependency of Aluimia on Pflotran; requested via petsc-dev Matt provide a correct fix
1362d9293e7bSBarry Smith      for i in postPackages:
1363d9293e7bSBarry Smith        if i.name.upper() in ['PFLOTRAN']:
1364d9293e7bSBarry Smith          i.postProcess()
1365d9293e7bSBarry Smith          postPackages.remove(i)
136628bb2e72SSatish Balay      for i in postPackages: i.postProcess()
1367aa5c8b8eSBarry Smith      for i in postPackages:
1368aa5c8b8eSBarry Smith        if i.installedpetsc:
1369aa5c8b8eSBarry Smith          self.installed = 1
1370aa5c8b8eSBarry Smith          break
137128bb2e72SSatish Balay    return
1372f8833479SBarry Smith
1373ce78bad3SBarry Smith  def generateFortranBindings(self):
1374ce78bad3SBarry Smith    if hasattr(self.compilers, 'FC') and self.framework.argDB['with-fortran-bindings']:
1375ce78bad3SBarry Smith      self.logPrintBox('Generating Fortran binding')
1376ce78bad3SBarry Smith      try:
1377fdec6269SSatish Balay        from utils import generatefortranbindings
1378ce78bad3SBarry Smith        generatefortranbindings.main(self.petscdir.dir, self.arch.arch)
1379ce78bad3SBarry Smith      except RuntimeError as e:
1380ce78bad3SBarry Smith        raise RuntimeError('*******Error generating Fortran stubs: '+str(e)+'*******\n')
1381ce78bad3SBarry Smith
1382f8833479SBarry Smith  def configure(self):
1383bf3e94a3SBarry Smith    if 'package-prefix-hash' in self.argDB:
1384bf3e94a3SBarry Smith      # turn off prefix if it was only used to for installing external packages.
1385bf3e94a3SBarry Smith      self.framework.argDB['prefix'] = ''
1386bf3e94a3SBarry Smith      self.dir = os.path.abspath(os.path.join(self.petscdir.dir, self.arch.arch))
1387bf3e94a3SBarry Smith      self.installdir.dir = self.dir
1388bf3e94a3SBarry Smith      self.installdir.petscDir = self.petscdir.dir
1389bf3e94a3SBarry Smith      self.petscDir = self.petscdir.dir
1390bf3e94a3SBarry Smith      self.petscArch = self.arch.arch
1391bf3e94a3SBarry Smith      self.addMakeMacro('PREFIXDIR',self.dir)
1392bf3e94a3SBarry Smith      self.confDir = os.path.abspath(os.path.join(self.petscdir.dir, self.arch.arch))
1393bf3e94a3SBarry Smith
1394f8833479SBarry Smith    if not os.path.samefile(self.petscdir.dir, os.getcwd()):
1395f8833479SBarry Smith      raise RuntimeError('Wrong PETSC_DIR option specified: '+str(self.petscdir.dir) + '\n  Configure invoked in: '+os.path.realpath(os.getcwd()))
1396550489e3SMatthew 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):
13973552d8fbSSatish Balay      raise RuntimeError('Incorrect option --prefix='+self.framework.argDB['prefix']+' specified. It cannot be same as PETSC_DIR!')
13988fd0dbdbSBarry Smith    if self.framework.argDB['prefix'] and self.framework.argDB['prefix'].find(' ') > -1:
13998fd0dbdbSBarry 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')
1400c16c35a9SSatish 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)):
1401c16c35a9SSatish Balay      raise RuntimeError('Incorrect option --prefix='+self.framework.argDB['prefix']+' specified. It cannot be same as PETSC_DIR/PETSC_ARCH!')
1402f16c1317SJed Brown    self.framework.header          = os.path.join(self.arch.arch,'include','petscconf.h')
1403f16c1317SJed Brown    self.framework.cHeader         = os.path.join(self.arch.arch,'include','petscfix.h')
1404bf113f49SJacob Faibussowitsch    self.framework.poisonheader    = os.path.join(self.arch.arch,'include','petscconf_poison.h')
14059c735a01SStefano Zampini    self.framework.pkgheader       = os.path.join(self.arch.arch,'include','petscpkg_version.h')
1406af0996ceSBarry Smith    self.framework.makeMacroHeader = os.path.join(self.arch.arch,'lib','petsc','conf','petscvariables')
1407af0996ceSBarry Smith    self.framework.makeRuleHeader  = os.path.join(self.arch.arch,'lib','petsc','conf','petscrules')
1408f8833479SBarry Smith    if self.libraries.math is None:
1409f8833479SBarry Smith      raise RuntimeError('PETSc requires a functional math library. Please send configure.log to petsc-maint@mcs.anl.gov.')
1410f8833479SBarry Smith    if self.languages.clanguage == 'Cxx' and not hasattr(self.compilers, 'CXX'):
1411f8833479SBarry Smith      raise RuntimeError('Cannot set C language to C++ without a functional C++ compiler.')
1412ed938b00SJed Brown    self.executeTest(self.configureRTLDDefault)
1413b2843cf1SBarry Smith    self.executeTest(self.configurePrefetch)
14142400fdedSBarry Smith    self.executeTest(self.configureUnused)
14151ef8df7fSJed Brown    self.executeTest(self.configureDeprecated)
141698ed35c3SBarry Smith    self.executeTest(self.configureIsatty)
1417e8e972b2SVaclav Hapla    self.executeTest(self.configureExpect)
1418e8e972b2SVaclav Hapla    self.executeTest(self.configureAlign)
1419e8e972b2SVaclav Hapla    self.executeTest(self.configureFunctionName)
1420e8e972b2SVaclav Hapla    self.executeTest(self.configureIntptrt)
1421f8833479SBarry Smith    self.executeTest(self.configureSolaris)
1422f8833479SBarry Smith    self.executeTest(self.configureLinux)
14230f64ec89SBarry Smith    self.executeTest(self.configureDarwin)
1424f8833479SBarry Smith    self.executeTest(self.configureWin32)
1425b10d012aSSatish Balay    self.executeTest(self.configureCygwinBrokenPipe)
1426569865ddSSatish Balay    self.executeTest(self.configureDefaultArch)
1427f8833479SBarry Smith    self.executeTest(self.configureScript)
1428f8833479SBarry Smith    self.executeTest(self.configureInstall)
142909bc878fSSatish Balay    self.executeTest(self.configureAtoll)
14309b81490aSJacob Faibussowitsch    self.executeTest(self.configureCoverage)
14317ce81a4bSJacob Faibussowitsch    self.executeTest(self.configureCoverageExecutable)
1432689a5dfaSJacob Faibussowitsch    self.executeTest(self.configureStrictPetscErrorCode)
1433648c30bcSBarry Smith    self.executeTest(self.configureSanitize)
1434ce78bad3SBarry Smith    self.executeTest(self.generateFortranBindings)
1435f8833479SBarry Smith
1436f8833479SBarry Smith    self.Dump()
1437f8833479SBarry Smith    self.dumpConfigInfo()
14382a4161d9SMatthew G Knepley    self.dumpMachineInfo()
143949fe22e6SSatish Balay    self.delGenFiles()
144040277576SBarry Smith    # need to save the current state of BuildSystem so that postProcess() packages can read it in and perhaps run make install
144140277576SBarry Smith    self.framework.storeSubstitutions(self.framework.argDB)
1442492432c8SJed Brown    self.framework.argDB['configureCache'] = pickle.dumps(self.framework)
144340277576SBarry Smith    self.framework.argDB.save(force = True)
14448244ab14SJed Brown    self.DumpPkgconfig('PETSc.pc')
14458244ab14SJed Brown    self.DumpPkgconfig('petsc.pc')
1446351d3a41SMatthew G Knepley    self.DumpModule()
1447f7ad81e1SBarry Smith    self.postProcessPackages()
1448f8833479SBarry Smith    self.framework.log.write('================================================================================\n')
1449f8833479SBarry Smith    self.logClear()
1450f8833479SBarry Smith    return
1451