xref: /petsc/config/PETSc/Configure.py (revision 5b3958d3aca72f972445ad919a0cd58c55ecebe5)
1f8833479SBarry Smithimport config.base
2f8833479SBarry Smith
3f8833479SBarry Smithimport os
46dd73af6SBarry Smithimport sys
5f8833479SBarry Smithimport re
6492432c8SJed Brownimport pickle
7f8833479SBarry Smith
8f8833479SBarry Smithclass Configure(config.base.Configure):
9f8833479SBarry Smith  def __init__(self, framework):
10f8833479SBarry Smith    config.base.Configure.__init__(self, framework)
11f8833479SBarry Smith    self.headerPrefix = 'PETSC'
12f8833479SBarry Smith    self.substPrefix  = 'PETSC'
13aa5c8b8eSBarry Smith    self.installed    = 0 # 1 indicates that Configure itself has already compiled and installed PETSc
144161761dSBarry Smith    self.found        = 1
15f8833479SBarry Smith    return
16f8833479SBarry Smith
177c939e48SSatish Balay  def __str2__(self):
185b4fc442SVaclav Hapla    desc = ['  Using GNU make: ' + self.make.make]
19aa5c8b8eSBarry Smith    if not self.installed:
20a0022257SSatish Balay      desc.append('xxx=========================================================================xxx')
21dc0529c6SBarry Smith      desc.append(' Configure stage complete. Now build PETSc libraries with:')
225b4fc442SVaclav Hapla      desc.append('   %s PETSC_DIR=%s PETSC_ARCH=%s all' % (self.make.make_user, self.petscdir.dir, self.arch.arch))
23a0022257SSatish Balay      desc.append('xxx=========================================================================xxx')
24aa5c8b8eSBarry Smith    else:
25aa5c8b8eSBarry Smith      desc.append('xxx=========================================================================xxx')
26aa5c8b8eSBarry Smith      desc.append(' Installation complete. You do not need to run make to compile or install the software')
27aa5c8b8eSBarry Smith      desc.append('xxx=========================================================================xxx')
287c939e48SSatish Balay    return '\n'.join(desc)+'\n'
29f8833479SBarry Smith
30f8833479SBarry Smith  def setupHelp(self, help):
31f8833479SBarry Smith    import nargs
32ce0b2093SBarry Smith    help.addArgument('PETSc',  '-prefix=<dir>',                              nargs.Arg(None, '', 'Specifiy location to install PETSc (eg. /usr/local)'))
337deb5ab3SBarry Smith    help.addArgument('PETSc',  '-with-prefetch=<bool>',                      nargs.ArgBool(None, 1,'Enable checking for prefetch instructions'))
34eed94e11SSatish Balay    help.addArgument('Windows','-with-windows-graphics=<bool>',              nargs.ArgBool(None, 1,'Enable check for Windows Graphics'))
35569865ddSSatish Balay    help.addArgument('PETSc', '-with-default-arch=<bool>',                   nargs.ArgBool(None, 1, 'Allow using the last configured arch without setting PETSC_ARCH'))
3657cb31baSSatish Balay    help.addArgument('PETSc','-with-single-library=<bool>',                  nargs.ArgBool(None, 1,'Put all PETSc code into the single -lpetsc library'))
37cb297985SSatish Balay    help.addArgument('PETSc','-with-fortran-bindings=<bool>',                nargs.ArgBool(None, 1,'Build PETSc fortran bindings in the library and corresponding module files'))
38525d6f2eSBarry Smith    help.addArgument('PETSc', '-with-ios=<bool>',                            nargs.ArgBool(None, 0, 'Build an iPhone/iPad version of PETSc library'))
3913f9d092SSatish Balay    help.addArgument('PETSc', '-with-display=<x11display>',                  nargs.Arg(None, '', 'Specifiy DISPLAY env variable for use with matlab test)'))
402c30b4dfSSatish Balay    help.addArgument('PETSc', '-with-package-scripts=<pyscripts>',           nargs.ArgFileList(None,None,'Specify configure package scripts for user provided packages'))
41f8833479SBarry Smith    return
42f8833479SBarry Smith
436dd73af6SBarry Smith  def registerPythonFile(self,filename,directory):
446dd73af6SBarry Smith    ''' Add a python file to the framework and registers its headerprefix, ... externalpackagedir
456dd73af6SBarry Smith        directory is the directory where the file relative to the BuildSystem or config path in python notation with . '''
466dd73af6SBarry Smith    (utilityName, ext) = os.path.splitext(filename)
476dd73af6SBarry Smith    if not utilityName.startswith('.') and not utilityName.startswith('#') and ext == '.py' and not utilityName == '__init__':
486dd73af6SBarry Smith      if directory: directory = directory+'.'
496dd73af6SBarry Smith      utilityObj                             = self.framework.require(directory+utilityName, self)
506dd73af6SBarry Smith      utilityObj.headerPrefix                = self.headerPrefix
516dd73af6SBarry Smith      utilityObj.archProvider                = self.arch
526dd73af6SBarry Smith      utilityObj.languageProvider            = self.languages
536dd73af6SBarry Smith      utilityObj.installDirProvider          = self.installdir
546dd73af6SBarry Smith      utilityObj.externalPackagesDirProvider = self.externalpackagesdir
556dd73af6SBarry Smith      utilityObj.precisionProvider           = self.scalartypes
566dd73af6SBarry Smith      utilityObj.indexProvider               = self.indexTypes
576dd73af6SBarry Smith      setattr(self, utilityName.lower(), utilityObj)
5851294b80SMatthew G. Knepley      return utilityObj
5951294b80SMatthew G. Knepley    return None
606dd73af6SBarry Smith
61f8833479SBarry Smith  def setupDependencies(self, framework):
62f8833479SBarry Smith    config.base.Configure.setupDependencies(self, framework)
63dca78d2bSSatish Balay    self.programs      = framework.require('config.programs',           self)
64f8833479SBarry Smith    self.setCompilers  = framework.require('config.setCompilers',       self)
6530b8aa07SMatthew G. Knepley    self.compilers     = framework.require('config.compilers',          self)
669d310bb7SBarry Smith    self.arch          = framework.require('PETSc.options.arch',        self.setCompilers)
679d310bb7SBarry Smith    self.petscdir      = framework.require('PETSc.options.petscdir',    self.arch)
689d310bb7SBarry Smith    self.installdir    = framework.require('PETSc.options.installDir',  self)
694e00a515SSatish Balay    self.dataFilesPath = framework.require('PETSc.options.dataFilesPath',self)
706dd73af6SBarry Smith    self.scalartypes   = framework.require('PETSc.options.scalarTypes', self)
716dd73af6SBarry Smith    self.indexTypes    = framework.require('PETSc.options.indexTypes',  self)
729d310bb7SBarry Smith    self.languages     = framework.require('PETSc.options.languages',   self.setCompilers)
7330b8aa07SMatthew G. Knepley    self.indexTypes    = framework.require('PETSc.options.indexTypes',  self.compilers)
74f8833479SBarry Smith    self.types         = framework.require('config.types',              self)
75f8833479SBarry Smith    self.headers       = framework.require('config.headers',            self)
76f8833479SBarry Smith    self.functions     = framework.require('config.functions',          self)
77f8833479SBarry Smith    self.libraries     = framework.require('config.libraries',          self)
78cd37d877SShri Abhyankar    self.atomics       = framework.require('config.atomics',            self)
799481793eSSatish Balay    self.make          = framework.require('config.packages.make',      self)
809552296fSBarry Smith    self.blasLapack    = framework.require('config.packages.BlasLapack',self)
81e6b0c433SBarry Smith    self.mpi           = framework.require('config.packages.MPI',       self)
820542e31aSBarry Smith    self.fortran       = framework.require('config.compilersFortran',   self)
830542e31aSBarry Smith    self.externalpackagesdir = framework.require('PETSc.options.externalpackagesdir',self)
8449d43ecaSSatish Balay
8509a6cbfcSBernhard M. Wiedemann    for utility in sorted(os.listdir(os.path.join('config','PETSc','options'))):
866dd73af6SBarry Smith      self.registerPythonFile(utility,'PETSc.options')
879d310bb7SBarry Smith
8809a6cbfcSBernhard M. Wiedemann    for utility in sorted(os.listdir(os.path.join('config','BuildSystem','config','utilities'))):
896dd73af6SBarry Smith      self.registerPythonFile(utility,'config.utilities')
9006e08bc7SBarry Smith
9109a6cbfcSBernhard M. Wiedemann    for package in sorted(os.listdir(os.path.join('config', 'BuildSystem', 'config', 'packages'))):
9251294b80SMatthew G. Knepley      obj = self.registerPythonFile(package,'config.packages')
9351294b80SMatthew G. Knepley      if obj:
9451294b80SMatthew G. Knepley        obj.archProvider                = self.framework.requireModule(obj.archProvider, obj)
9551294b80SMatthew G. Knepley        obj.languageProvider            = self.framework.requireModule(obj.languageProvider, obj)
9651294b80SMatthew G. Knepley        obj.installDirProvider          = self.framework.requireModule(obj.installDirProvider, obj)
9751294b80SMatthew G. Knepley        obj.externalPackagesDirProvider = self.framework.requireModule(obj.externalPackagesDirProvider, obj)
9851294b80SMatthew G. Knepley        obj.precisionProvider           = self.framework.requireModule(obj.precisionProvider, obj)
9951294b80SMatthew G. Knepley        obj.indexProvider               = self.framework.requireModule(obj.indexProvider, obj)
1006dd73af6SBarry Smith
1015faf1eacSMatthew G. Knepley    # Force blaslapack and opencl to depend on scalarType so precision is set before BlasLapack is built
1029d310bb7SBarry Smith    framework.require('PETSc.options.scalarTypes', self.f2cblaslapack)
1039d310bb7SBarry Smith    framework.require('PETSc.options.scalarTypes', self.fblaslapack)
1049d310bb7SBarry Smith    framework.require('PETSc.options.scalarTypes', self.blaslapack)
1055faf1eacSMatthew G. Knepley    framework.require('PETSc.options.scalarTypes', self.opencl)
106f8833479SBarry Smith
107dca78d2bSSatish Balay    self.programs.headerPrefix     = self.headerPrefix
108*5b3958d3SJacob Faibussowitsch    self.setCompilers.headerPrefix = self.headerPrefix
109f8833479SBarry Smith    self.compilers.headerPrefix    = self.headerPrefix
1100542e31aSBarry Smith    self.fortran.headerPrefix      = self.headerPrefix
111f8833479SBarry Smith    self.types.headerPrefix        = self.headerPrefix
112f8833479SBarry Smith    self.headers.headerPrefix      = self.headerPrefix
113f8833479SBarry Smith    self.functions.headerPrefix    = self.headerPrefix
114f8833479SBarry Smith    self.libraries.headerPrefix    = self.headerPrefix
1156dd73af6SBarry Smith
1162c30b4dfSSatish Balay    # Register user provided package scripts
1172c30b4dfSSatish Balay    if 'with-package-scripts' in self.framework.argDB:
1182c30b4dfSSatish Balay      for script in self.framework.argDB['with-package-scripts']:
1192c30b4dfSSatish Balay        if os.path.splitext(script)[1] != '.py':
1202c30b4dfSSatish Balay          raise RuntimeError('Only python scripts compatible with configure package script format should be specified! Invalid option -with-package-scripts='+script)
1212c30b4dfSSatish Balay        self.framework.logPrint('User is registering a new package script: '+script)
1222c30b4dfSSatish Balay        dname,fname = os.path.split(script)
1232c30b4dfSSatish Balay        if dname: sys.path.append(dname)
1242c30b4dfSSatish Balay        self.registerPythonFile(fname,'')
1256dd73af6SBarry Smith
1266dd73af6SBarry Smith    # test for a variety of basic headers and functions
1274211eb48SBarry Smith    headersC = map(lambda name: name+'.h',['setjmp','dos','fcntl','float','io','malloc','pwd','strings',
128ba61063dSBarry Smith                                            'unistd','sys/sysinfo','machine/endian','sys/param','sys/procfs','sys/resource',
1292475b7caSBarry Smith                                            'sys/systeminfo','sys/times','sys/utsname',
1307e4f0192SMosè Giordano                                            'sys/socket','sys/wait','netinet/in','netdb','direct','time','Ws2tcpip','sys/types',
1312475b7caSBarry Smith                                            'WindowsX','float','ieeefp','stdint','pthread','inttypes','immintrin','zmmintrin'])
13245082d64SJed Brown    functions = ['access','_access','clock','drand48','getcwd','_getcwd','getdomainname','gethostname',
1334211eb48SBarry Smith                 'getwd','memalign','popen','PXFGETARG','rand','getpagesize',
1344211eb48SBarry Smith                 'readlink','realpath','usleep','sleep','_sleep',
1352475b7caSBarry Smith                 'uname','snprintf','_snprintf','lseek','_lseek','time','fork','stricmp',
1362475b7caSBarry Smith                 'strcasecmp','bzero','dlopen','dlsym','dlclose','dlerror',
137cc9df77eSBarry Smith                 '_set_output_format','_mkdir','socket','gethostbyname','_pipe','fpresetsticky','fpsetsticky']
138b0651e32SBarry Smith    libraries = [(['fpe'],'handle_sigfpes')]
139b0651e32SBarry Smith    librariessock = [(['socket','nsl'],'socket')]
140f8833479SBarry Smith    self.headers.headers.extend(headersC)
141f8833479SBarry Smith    self.functions.functions.extend(functions)
142b0651e32SBarry Smith    self.libraries.libraries.extend(libraries)
143b0651e32SBarry Smith    if not hasattr(self,'socket'):
144b0651e32SBarry Smith      self.libraries.libraries.extend(librariessock)
145f8833479SBarry Smith    return
146f8833479SBarry Smith
1478244ab14SJed Brown  def DumpPkgconfig(self, petsc_pc):
148262119f8SBarry Smith    ''' Create a pkg-config file '''
149262119f8SBarry Smith    if not os.path.exists(os.path.join(self.petscdir.dir,self.arch.arch,'lib','pkgconfig')):
150262119f8SBarry Smith      os.makedirs(os.path.join(self.petscdir.dir,self.arch.arch,'lib','pkgconfig'))
1512eefe1c6SJed Brown    with open(os.path.join(self.petscdir.dir,self.arch.arch,'lib','pkgconfig',petsc_pc),'w') as fd:
1525e3311eeSJed Brown      cflags_inc = ['-I${includedir}']
153262119f8SBarry Smith      if self.framework.argDB['prefix']:
1545bb5b263SMatthew G. Knepley        fd.write('prefix='+self.installdir.dir+'\n')
155262119f8SBarry Smith      else:
156e1e675deSJed Brown        fd.write('prefix='+os.path.join(self.petscdir.dir, self.arch.arch)+'\n')
157e1e675deSJed Brown        cflags_inc.append('-I' + os.path.join(self.petscdir.dir, 'include'))
158262119f8SBarry Smith      fd.write('exec_prefix=${prefix}\n')
159262119f8SBarry Smith      fd.write('includedir=${prefix}/include\n')
1605e3311eeSJed Brown      fd.write('libdir=${prefix}/lib\n')
161262119f8SBarry Smith
1622eefe1c6SJed Brown      with self.setCompilers.Language('C'):
163262119f8SBarry Smith        fd.write('ccompiler='+self.setCompilers.getCompiler()+'\n')
164756c7f9fSJed Brown        fd.write('cflags_extra='+self.setCompilers.getCompilerFlags().strip()+'\n')
165756c7f9fSJed Brown        fd.write('cflags_dep='+self.compilers.dependenciesGenerationFlag.get('C','')+'\n')
166756c7f9fSJed Brown        fd.write('ldflag_rpath='+self.setCompilers.CSharedLinkerFlag+'\n')
16703e383c8SJed Brown      if hasattr(self.compilers, 'CXX'):
1682eefe1c6SJed Brown        with self.setCompilers.Language('C++'):
169262119f8SBarry Smith          fd.write('cxxcompiler='+self.setCompilers.getCompiler()+'\n')
170756c7f9fSJed Brown          fd.write('cxxflags_extra='+self.setCompilers.getCompilerFlags().strip()+'\n')
171262119f8SBarry Smith      if hasattr(self.compilers, 'FC'):
1722eefe1c6SJed Brown        with self.setCompilers.Language('FC'):
173262119f8SBarry Smith          fd.write('fcompiler='+self.setCompilers.getCompiler()+'\n')
174756c7f9fSJed Brown          fd.write('fflags_extra='+self.setCompilers.getCompilerFlags().strip()+'\n')
17550520af6SPatrick Sanan      if hasattr(self.compilers, 'CUDAC'):
17650520af6SPatrick Sanan        with self.setCompilers.Language('CUDA'):
17750520af6SPatrick Sanan          fd.write('cudacompiler='+self.setCompilers.getCompiler()+'\n')
17850520af6SPatrick Sanan          fd.write('cudaflags_extra='+self.setCompilers.getCompilerFlags().strip()+'\n')
1798f561fa3SPatrick Sanan          p = self.framework.require('config.packages.cuda')
18050520af6SPatrick Sanan          fd.write('cudalib='+self.libraries.toStringNoDupes(p.lib)+'\n')
18150520af6SPatrick Sanan          fd.write('cudainclude='+self.headers.toStringNoDupes(p.include)+'\n')
1827ba7a817SBarry Smith          if hasattr(self.setCompilers,'CUDA_CXX'):
1837ba7a817SBarry Smith            fd.write('cuda_cxx='+self.setCompilers.CUDA_CXX+'\n')
1847ba7a817SBarry Smith            fd.write('cuda_cxxflags='+self.setCompilers.CUDA_CXXFLAGS+'\n')
185262119f8SBarry Smith
186262119f8SBarry Smith      fd.write('\n')
187262119f8SBarry Smith      fd.write('Name: PETSc\n')
188262119f8SBarry Smith      fd.write('Description: Library to solve ODEs and algebraic equations\n')
189351d3a41SMatthew G Knepley      fd.write('Version: %s\n' % self.petscdir.version)
1905e3311eeSJed Brown      fd.write('Cflags: ' + ' '.join([self.setCompilers.CPPFLAGS] + cflags_inc) + '\n')
19137371b91SJed Brown      fd.write('Libs: '+self.libraries.toStringNoDupes(['-L${libdir}', self.petsclib], with_rpath=False)+'\n')
1928ebf8858SJed Brown      # Remove RPATH flags from library list.  User can add them using
1938ebf8858SJed Brown      # pkg-config --variable=ldflag_rpath and pkg-config --libs-only-L
194de8f682fSSatish 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')
195262119f8SBarry Smith    return
196262119f8SBarry Smith
197351d3a41SMatthew G Knepley  def DumpModule(self):
198351d3a41SMatthew G Knepley    ''' Create a module file '''
199af0996ceSBarry Smith    if not os.path.exists(os.path.join(self.petscdir.dir,self.arch.arch,'lib','petsc','conf','modules')):
200af0996ceSBarry Smith      os.makedirs(os.path.join(self.petscdir.dir,self.arch.arch,'lib','petsc','conf','modules'))
201af0996ceSBarry Smith    if not os.path.exists(os.path.join(self.petscdir.dir,self.arch.arch,'lib','petsc','conf','modules','petsc')):
202af0996ceSBarry Smith      os.makedirs(os.path.join(self.petscdir.dir,self.arch.arch,'lib','petsc','conf','modules','petsc'))
203351d3a41SMatthew G Knepley    if self.framework.argDB['prefix']:
2045bb5b263SMatthew G. Knepley      installdir  = self.installdir.dir
20555d606a3SSatish Balay      installarch = ''
20655d606a3SSatish Balay      installpath = os.path.join(installdir,'bin')
207351d3a41SMatthew G Knepley    else:
208351d3a41SMatthew G Knepley      installdir  = self.petscdir.dir
20955d606a3SSatish Balay      installarch = self.arch.arch
21055d606a3SSatish Balay      installpath = os.path.join(installdir,installarch,'bin')+':'+os.path.join(installdir,'bin')
211af0996ceSBarry Smith    fd = open(os.path.join(self.petscdir.dir,self.arch.arch,'lib','petsc','conf','modules','petsc',self.petscdir.version),'w')
212351d3a41SMatthew G Knepley    fd.write('''\
213351d3a41SMatthew G Knepley#%%Module
214351d3a41SMatthew G Knepley
215351d3a41SMatthew G Knepleyproc ModulesHelp { } {
216351d3a41SMatthew G Knepley    puts stderr "This module sets the path and environment variables for petsc-%s"
217a17b96a8SKyle Gerard Felker    puts stderr "     see https://petsc.org/ for more information      "
218351d3a41SMatthew G Knepley    puts stderr ""
219351d3a41SMatthew G Knepley}
220351d3a41SMatthew G Knepleymodule-whatis "PETSc - Portable, Extensible Toolkit for Scientific Computation"
221351d3a41SMatthew G Knepley
222dd486775SJed Brownset petsc_dir   "%s"
223dd486775SJed Brownset petsc_arch  "%s"
224351d3a41SMatthew G Knepley
225dd486775SJed Brownsetenv PETSC_ARCH "$petsc_arch"
226dd486775SJed Brownsetenv PETSC_DIR "$petsc_dir"
227dd486775SJed Brownprepend-path PATH "%s"
22855d606a3SSatish Balay''' % (self.petscdir.version, installdir, installarch, installpath))
229351d3a41SMatthew G Knepley    fd.close()
230351d3a41SMatthew G Knepley    return
231351d3a41SMatthew G Knepley
232f8833479SBarry Smith  def Dump(self):
233f8833479SBarry Smith    ''' Actually put the values into the configuration files '''
234f8833479SBarry Smith    # eventually everything between -- should be gone
23517f368bcSBarry Smith    if self.mpi.usingMPIUni:
23617f368bcSBarry Smith      #
23717f368bcSBarry Smith      # Remove any MPI/MPICH include files that may have been put here by previous runs of ./configure
2387908f030SMatthew 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)
23917f368bcSBarry Smith
240b5f71184SBarry Smith    self.logPrintDivider()
241b5f71184SBarry Smith    # Test for compiler-specific macros that need to be defined.
242b5f71184SBarry Smith    if self.setCompilers.isCrayVector('CC', self.log):
243b5f71184SBarry Smith      self.addDefine('HAVE_CRAY_VECTOR','1')
244b5f71184SBarry Smith
245b5f71184SBarry Smith    if self.functions.haveFunction('gethostbyname') and self.functions.haveFunction('socket') and self.headers.haveHeader('netinet/in.h'):
246b5f71184SBarry Smith      self.addDefine('USE_SOCKET_VIEWER','1')
247b5f71184SBarry Smith      if self.checkCompile('#include <sys/socket.h>','setsockopt(0,SOL_SOCKET,SO_REUSEADDR,0,0)'):
248b5f71184SBarry Smith        self.addDefine('HAVE_SO_REUSEADDR','1')
249b5f71184SBarry Smith
250b5f71184SBarry Smith    self.logPrintDivider()
2515f27b2e0SBarry Smith    self.setCompilers.pushLanguage('C')
2525f27b2e0SBarry Smith    compiler = self.setCompilers.getCompiler()
253217fe27eSSatish Balay    if [s for s in ['mpicc','mpiicc'] if os.path.basename(compiler).find(s)>=0]:
2545f27b2e0SBarry Smith      try:
2555f27b2e0SBarry Smith        output   = self.executeShellCommand(compiler + ' -show', log = self.log)[0]
2565f27b2e0SBarry Smith        compiler = output.split(' ')[0]
257f424265bSStefano Zampini        self.addDefine('MPICC_SHOW','"'+output.strip().replace('\n','\\\\n').replace('"','')+'"')
2585f27b2e0SBarry Smith      except:
259c9872b61SBarry Smith        self.addDefine('MPICC_SHOW','"Unavailable"')
260c9872b61SBarry Smith    else:
261c9872b61SBarry Smith      self.addDefine('MPICC_SHOW','"Unavailable"')
2625f27b2e0SBarry Smith    self.setCompilers.popLanguage()
263f8833479SBarry Smith#-----------------------------------------------------------------------------------------------------
264f8833479SBarry Smith
265f8833479SBarry Smith    # Sometimes we need C compiler, even if built with C++
266f8833479SBarry Smith    self.setCompilers.pushLanguage('C')
267f8833479SBarry Smith    self.addMakeMacro('CC_FLAGS',self.setCompilers.getCompilerFlags())
268f8833479SBarry Smith    self.setCompilers.popLanguage()
269f8833479SBarry Smith
27034f774f6SJed Brown    # And sometimes we need a C++ compiler even when PETSc is built with C
27134f774f6SJed Brown    if hasattr(self.compilers, 'CXX'):
27234f774f6SJed Brown      self.setCompilers.pushLanguage('Cxx')
27329921a8fSScott Kruger      self.addDefine('HAVE_CXX','1')
2740b119762SSatish Balay      self.addMakeMacro('CXXPP_FLAGS',self.setCompilers.CXXPPFLAGS)
27534f774f6SJed Brown      self.addMakeMacro('CXX_FLAGS',self.setCompilers.getCompilerFlags())
2762f4326f3SSatish Balay      cxx_linker = self.setCompilers.getLinker()
2772f4326f3SSatish Balay      self.addMakeMacro('CXX_LINKER',cxx_linker)
2782f4326f3SSatish Balay      self.addMakeMacro('CXX_LINKER_FLAGS',self.setCompilers.getLinkerFlags())
27934f774f6SJed Brown      self.setCompilers.popLanguage()
28034f774f6SJed Brown
281f8833479SBarry Smith    # C preprocessor values
2821315f054SBarry Smith    self.addMakeMacro('CPP_FLAGS',self.setCompilers.CPPFLAGS)
283f8833479SBarry Smith
284f8833479SBarry Smith    # compiler values
285f8833479SBarry Smith    self.setCompilers.pushLanguage(self.languages.clanguage)
286f8833479SBarry Smith    self.addMakeMacro('PCC',self.setCompilers.getCompiler())
287f8833479SBarry Smith    self.addMakeMacro('PCC_FLAGS',self.setCompilers.getCompilerFlags())
288f8833479SBarry Smith    self.setCompilers.popLanguage()
289f8833479SBarry Smith    # .o or .obj
290f8833479SBarry Smith    self.addMakeMacro('CC_SUFFIX','o')
291f8833479SBarry Smith
292f8833479SBarry Smith    # executable linker values
293f8833479SBarry Smith    self.setCompilers.pushLanguage(self.languages.clanguage)
294f8833479SBarry Smith    pcc_linker = self.setCompilers.getLinker()
295f8833479SBarry Smith    self.addMakeMacro('PCC_LINKER',pcc_linker)
296c84a332bSSatish Balay    self.addMakeMacro('PCC_LINKER_FLAGS',self.setCompilers.getLinkerFlags())
297f8833479SBarry Smith    self.setCompilers.popLanguage()
298f8833479SBarry Smith    # '' for Unix, .exe for Windows
299f8833479SBarry Smith    self.addMakeMacro('CC_LINKER_SUFFIX','')
300f8833479SBarry Smith
301f8833479SBarry Smith    if hasattr(self.compilers, 'FC'):
302cb297985SSatish Balay      if self.framework.argDB['with-fortran-bindings']:
303257f4e5aSSatish Balay        if not self.fortran.fortranIsF90:
304257f4e5aSSatish Balay          raise RuntimeError('Error! Fortran compiler "'+self.compilers.FC+'" does not support F90! PETSc fortran bindings require a F90 compiler')
305cb297985SSatish Balay        self.addDefine('HAVE_FORTRAN','1')
306f8833479SBarry Smith      self.setCompilers.pushLanguage('FC')
307f8833479SBarry Smith      # need FPPFLAGS in config/setCompilers
3080b119762SSatish Balay      self.addMakeMacro('FPP_FLAGS',self.setCompilers.FPPFLAGS)
309f8833479SBarry Smith
310f8833479SBarry Smith      # compiler values
311f8833479SBarry Smith      self.addMakeMacro('FC_FLAGS',self.setCompilers.getCompilerFlags())
312f8833479SBarry Smith      self.setCompilers.popLanguage()
313f8833479SBarry Smith      # .o or .obj
314f8833479SBarry Smith      self.addMakeMacro('FC_SUFFIX','o')
315f8833479SBarry Smith
316f8833479SBarry Smith      # executable linker values
317f8833479SBarry Smith      self.setCompilers.pushLanguage('FC')
318a9acdec7SBarry Smith      self.addMakeMacro('FC_LINKER',self.setCompilers.getLinker())
3196d53d35eSSatish Balay      self.addMakeMacro('FC_LINKER_FLAGS',self.setCompilers.getLinkerFlags())
3203feacd00SBarry Smith      # apple requires this shared library linker flag on SOME versions of the os
3213feacd00SBarry Smith      if self.setCompilers.getLinkerFlags().find('-Wl,-commons,use_dylibs') > -1:
3223feacd00SBarry Smith        self.addMakeMacro('DARWIN_COMMONS_USE_DYLIBS',' -Wl,-commons,use_dylibs ')
323bb82cf9cSSatish Balay      self.setCompilers.popLanguage()
3245d631499SMatthew Knepley
3255d631499SMatthew Knepley      # F90 Modules
3265d631499SMatthew Knepley      if self.setCompilers.fortranModuleIncludeFlag:
3275d631499SMatthew Knepley        self.addMakeMacro('FC_MODULE_FLAG', self.setCompilers.fortranModuleIncludeFlag)
3286ddd6694SSatish Balay      else: # for non-f90 compilers like g77
3296ddd6694SSatish Balay        self.addMakeMacro('FC_MODULE_FLAG', '-I')
330a324c51cSMatthew G Knepley      if self.setCompilers.fortranModuleIncludeFlag:
331a324c51cSMatthew G Knepley        self.addMakeMacro('FC_MODULE_OUTPUT_FLAG', self.setCompilers.fortranModuleOutputFlag)
332f8833479SBarry Smith    else:
333f8833479SBarry Smith      self.addMakeMacro('FC','')
334f8833479SBarry Smith
33546a3958fSBarry Smith    if hasattr(self.compilers, 'CUDAC'):
3367ff2890cSSatish Balay      self.setCompilers.pushLanguage('CUDA')
337d93a25ecSSatish Balay      self.addMakeMacro('CUDAC_FLAGS',self.setCompilers.getCompilerFlags())
33850dcbc5aSJunchao Zhang      self.addMakeMacro('CUDAPP_FLAGS',self.setCompilers.CUDAPPFLAGS)
3397ff2890cSSatish Balay      self.setCompilers.popLanguage()
3407ff2890cSSatish Balay
341694a2f0eSJunchao Zhang    if hasattr(self.compilers, 'HIPC'):
34228f796eaSScott Kruger      self.setCompilers.pushLanguage('HIP')
343694a2f0eSJunchao Zhang      self.addMakeMacro('HIPC_FLAGS',self.setCompilers.getCompilerFlags())
3447fb1458fSStefano Zampini      self.addMakeMacro('HIPPP_FLAGS',self.setCompilers.HIPPPFLAGS)
34528f796eaSScott Kruger      self.setCompilers.popLanguage()
34628f796eaSScott Kruger
34750dcbc5aSJunchao Zhang    if hasattr(self.compilers, 'SYCLC'):
34828f796eaSScott Kruger      self.setCompilers.pushLanguage('SYCL')
34950dcbc5aSJunchao Zhang      self.addMakeMacro('SYCLC_FLAGS',self.setCompilers.getCompilerFlags())
35050dcbc5aSJunchao Zhang      self.addMakeMacro('SYCLPP_FLAGS',self.setCompilers.SYCLPPFLAGS)
35128f796eaSScott Kruger      self.setCompilers.popLanguage()
35228f796eaSScott Kruger
353f8833479SBarry Smith    # shared library linker values
354f8833479SBarry Smith    self.setCompilers.pushLanguage(self.languages.clanguage)
355f8833479SBarry Smith    # need to fix BuildSystem to collect these separately
356f8833479SBarry Smith    self.addMakeMacro('SL_LINKER',self.setCompilers.getLinker())
35770db8aa6SSatish Balay    self.addMakeMacro('SL_LINKER_FLAGS','${PCC_LINKER_FLAGS}')
358f8833479SBarry Smith    self.setCompilers.popLanguage()
359f8833479SBarry Smith    # One of 'a', 'so', 'lib', 'dll', 'dylib' (perhaps others also?) depending on the library generator and architecture
360f8833479SBarry Smith    # Note: . is not included in this macro, consistent with AR_LIB_SUFFIX
361f8833479SBarry Smith    if self.setCompilers.sharedLibraryExt == self.setCompilers.AR_LIB_SUFFIX:
362f8833479SBarry Smith      self.addMakeMacro('SL_LINKER_SUFFIX', '')
36346bc77b6SBarry Smith      self.addDefine('SLSUFFIX','""')
364f8833479SBarry Smith    else:
365f8833479SBarry Smith      self.addMakeMacro('SL_LINKER_SUFFIX', self.setCompilers.sharedLibraryExt)
36646bc77b6SBarry Smith      self.addDefine('SLSUFFIX','"'+self.setCompilers.sharedLibraryExt+'"')
367bb82cf9cSSatish Balay
36823e93537SBarry Smith    self.addMakeMacro('SL_LINKER_LIBS','${PETSC_EXTERNAL_LIB_BASIC}')
369bb82cf9cSSatish Balay
370f8833479SBarry Smith#-----------------------------------------------------------------------------------------------------
371f8833479SBarry Smith
372f8833479SBarry Smith    # CONLY or CPP. We should change the PETSc makefiles to do this better
373f8833479SBarry Smith    if self.languages.clanguage == 'C': lang = 'CONLY'
374f8833479SBarry Smith    else: lang = 'CXXONLY'
375f8833479SBarry Smith    self.addMakeMacro('PETSC_LANGUAGE',lang)
376f8833479SBarry Smith
377f8833479SBarry Smith    # real or complex
378f8833479SBarry Smith    self.addMakeMacro('PETSC_SCALAR',self.scalartypes.scalartype)
379f8833479SBarry Smith    # double or float
380f8833479SBarry Smith    self.addMakeMacro('PETSC_PRECISION',self.scalartypes.precision)
381f8833479SBarry Smith
382f8833479SBarry Smith    if self.framework.argDB['with-batch']:
383f8833479SBarry Smith      self.addMakeMacro('PETSC_WITH_BATCH','1')
384f8833479SBarry Smith
385f8833479SBarry Smith#-----------------------------------------------------------------------------------------------------
386a6cc6bb1SBarry Smith    # print include and lib for makefiles
387b5f71184SBarry Smith    self.logPrintDivider()
388f8833479SBarry Smith    self.framework.packages.reverse()
3895a21677cSJed Brown    petscincludes = [os.path.join(self.petscdir.dir,'include'),os.path.join(self.petscdir.dir,self.arch.arch,'include')]
3908749e224SJunchao Zhang    petscincludes_install = [os.path.join(self.installdir.dir, 'include')] if self.framework.argDB['prefix'] else petscincludes
3915a21677cSJed Brown    includes = []
392de8f682fSSatish Balay    self.packagelibs = []
393f8833479SBarry Smith    for i in self.framework.packages:
3947f0ff1afSBarry Smith      if not i.required:
3953972cb20SJacob Faibussowitsch        if i.devicePackage:
3963972cb20SJacob Faibussowitsch          self.addDefine('HAVE_DEVICE',1)
397eeb16384SBarry Smith        self.addDefine('HAVE_'+i.PACKAGE.replace('-','_'), 1)  # ONLY list package if it is used directly by PETSc (and not only by another package)
398f8833479SBarry Smith      if not isinstance(i.lib, list):
399f8833479SBarry Smith        i.lib = [i.lib]
400de8f682fSSatish Balay      if i.linkedbypetsc: self.packagelibs.extend(i.lib)
401eeb16384SBarry Smith      self.addMakeMacro(i.PACKAGE.replace('-','_')+'_LIB', self.libraries.toStringNoDupes(i.lib))
402f8833479SBarry Smith      if hasattr(i,'include'):
403f8833479SBarry Smith        if not isinstance(i.include,list):
404f8833479SBarry Smith          i.include = [i.include]
405ac9e4c42SSatish Balay        includes.extend(i.include)
406eeb16384SBarry Smith        self.addMakeMacro(i.PACKAGE.replace('-','_')+'_INCLUDE',self.headers.toStringNoDupes(i.include))
4072df986feSBarry Smith    if self.framework.argDB['with-single-library']:
408e282ce78SJed Brown      self.petsclib = '-lpetsc'
40991bb3077SSatish Balay    else:
410e282ce78SJed Brown      self.petsclib = '-lpetscts -lpetscsnes -lpetscksp -lpetscdm -lpetscmat -lpetscvec -lpetscsys'
411de8f682fSSatish Balay    self.complibs = self.compilers.flibs+self.compilers.cxxlibs+self.compilers.LIBS.split()
4125a21677cSJed Brown    self.PETSC_WITH_EXTERNAL_LIB = self.libraries.toStringNoDupes(['-L${PETSC_DIR}/${PETSC_ARCH}/lib', self.petsclib]+self.packagelibs+self.complibs)
413de8f682fSSatish Balay    self.PETSC_EXTERNAL_LIB_BASIC = self.libraries.toStringNoDupes(self.packagelibs+self.complibs)
414de8f682fSSatish Balay
415de8f682fSSatish Balay    self.addMakeMacro('PETSC_EXTERNAL_LIB_BASIC',self.PETSC_EXTERNAL_LIB_BASIC)
4165a21677cSJed Brown    allincludes = petscincludes + includes
4175a21677cSJed Brown    allincludes_install = petscincludes_install + includes
4185a21677cSJed Brown    self.PETSC_CC_INCLUDES = self.headers.toStringNoDupes(allincludes)
4195a21677cSJed Brown    self.PETSC_CC_INCLUDES_INSTALL = self.headers.toStringNoDupes(allincludes_install)
4205a21677cSJed Brown    self.addMakeMacro('PETSC_CC_INCLUDES',self.PETSC_CC_INCLUDES)
4215a21677cSJed Brown    self.addMakeMacro('PETSC_CC_INCLUDES_INSTALL', self.PETSC_CC_INCLUDES_INSTALL)
422cbd5cc15SBarry Smith    if hasattr(self.compilers, 'FC'):
4235a21677cSJed Brown      def modinc(includes):
4240542e31aSBarry Smith        return includes if self.fortran.fortranIsF90 else []
4255a21677cSJed Brown      self.addMakeMacro('PETSC_FC_INCLUDES',self.headers.toStringNoDupes(allincludes,modinc(allincludes)))
4265a21677cSJed Brown      self.addMakeMacro('PETSC_FC_INCLUDES_INSTALL',self.headers.toStringNoDupes(allincludes_install,modinc(allincludes_install)))
427f8833479SBarry Smith
4285bb5b263SMatthew G. Knepley    self.addDefine('LIB_DIR','"'+os.path.join(self.installdir.dir,'lib')+'"')
429f8833479SBarry Smith
4300f3b21c2SBarry Smith    if self.framework.argDB['with-single-library']:
4310f3b21c2SBarry Smith      # overrides the values set in conf/variables
4320f3b21c2SBarry Smith      self.addMakeMacro('LIBNAME','${INSTALL_LIB_DIR}/libpetsc.${AR_LIB_SUFFIX}')
43357cb31baSSatish Balay      self.addMakeMacro('SHLIBS','libpetsc')
434bccf1c12SBarry Smith      self.addMakeMacro('PETSC_LIB_BASIC','-lpetsc')
435797063a9SSatish Balay      self.addMakeMacro('PETSC_KSP_LIB_BASIC','-lpetsc')
436797063a9SSatish Balay      self.addMakeMacro('PETSC_TS_LIB_BASIC','-lpetsc')
437b0a7d7e7SSatish Balay      self.addMakeMacro('PETSC_TAO_LIB_BASIC','-lpetsc')
438de8f682fSSatish Balay      self.addMakeMacro('PETSC_WITH_EXTERNAL_LIB',self.PETSC_WITH_EXTERNAL_LIB)
439bb84e0fdSBarry Smith      self.addDefine('USE_SINGLE_LIBRARY', '1')
4402df986feSBarry Smith      if self.sharedlibraries.useShared:
441ea820d49SSatish Balay        self.addMakeMacro('PETSC_SYS_LIB','${C_SH_LIB_PATH} ${PETSC_WITH_EXTERNAL_LIB}')
442ea820d49SSatish Balay        self.addMakeMacro('PETSC_VEC_LIB','${C_SH_LIB_PATH} ${PETSC_WITH_EXTERNAL_LIB}')
443ea820d49SSatish Balay        self.addMakeMacro('PETSC_MAT_LIB','${C_SH_LIB_PATH} ${PETSC_WITH_EXTERNAL_LIB}')
444ea820d49SSatish Balay        self.addMakeMacro('PETSC_DM_LIB','${C_SH_LIB_PATH} ${PETSC_WITH_EXTERNAL_LIB}')
445ea820d49SSatish Balay        self.addMakeMacro('PETSC_KSP_LIB','${C_SH_LIB_PATH} ${PETSC_WITH_EXTERNAL_LIB}')
446ea820d49SSatish Balay        self.addMakeMacro('PETSC_SNES_LIB','${C_SH_LIB_PATH} ${PETSC_WITH_EXTERNAL_LIB}')
447ea820d49SSatish Balay        self.addMakeMacro('PETSC_TS_LIB','${C_SH_LIB_PATH} ${PETSC_WITH_EXTERNAL_LIB}')
448b0a7d7e7SSatish Balay        self.addMakeMacro('PETSC_TAO_LIB','${C_SH_LIB_PATH} ${PETSC_WITH_EXTERNAL_LIB}')
449fdb87e33SJed Brown        self.addMakeMacro('PETSC_CHARACTERISTIC_LIB','${C_SH_LIB_PATH} ${PETSC_WITH_EXTERNAL_LIB}')
450ea820d49SSatish Balay        self.addMakeMacro('PETSC_LIB','${C_SH_LIB_PATH} ${PETSC_WITH_EXTERNAL_LIB}')
451ea820d49SSatish Balay        self.addMakeMacro('PETSC_CONTRIB','${C_SH_LIB_PATH} ${PETSC_WITH_EXTERNAL_LIB}')
4522df986feSBarry Smith      else:
453ea820d49SSatish Balay        self.addMakeMacro('PETSC_SYS_LIB','${PETSC_WITH_EXTERNAL_LIB}')
454ea820d49SSatish Balay        self.addMakeMacro('PETSC_VEC_LIB','${PETSC_WITH_EXTERNAL_LIB}')
455ea820d49SSatish Balay        self.addMakeMacro('PETSC_MAT_LIB','${PETSC_WITH_EXTERNAL_LIB}')
456ea820d49SSatish Balay        self.addMakeMacro('PETSC_DM_LIB','${PETSC_WITH_EXTERNAL_LIB}')
457ea820d49SSatish Balay        self.addMakeMacro('PETSC_KSP_LIB','${PETSC_WITH_EXTERNAL_LIB}')
458ea820d49SSatish Balay        self.addMakeMacro('PETSC_SNES_LIB','${PETSC_WITH_EXTERNAL_LIB}')
459ea820d49SSatish Balay        self.addMakeMacro('PETSC_TS_LIB','${PETSC_WITH_EXTERNAL_LIB}')
460b0a7d7e7SSatish Balay        self.addMakeMacro('PETSC_TAO_LIB','${PETSC_WITH_EXTERNAL_LIB}')
461fdb87e33SJed Brown        self.addMakeMacro('PETSC_CHARACTERISTIC_LIB','${PETSC_WITH_EXTERNAL_LIB}')
462ea820d49SSatish Balay        self.addMakeMacro('PETSC_LIB','${PETSC_WITH_EXTERNAL_LIB}')
463ea820d49SSatish Balay        self.addMakeMacro('PETSC_CONTRIB','${PETSC_WITH_EXTERNAL_LIB}')
4640f3b21c2SBarry Smith
465f8833479SBarry Smith    if not os.path.exists(os.path.join(self.petscdir.dir,self.arch.arch,'lib')):
466f8833479SBarry Smith      os.makedirs(os.path.join(self.petscdir.dir,self.arch.arch,'lib'))
467f8833479SBarry Smith
46813f9d092SSatish Balay# add a makefile endtry for display
46913f9d092SSatish Balay    if self.framework.argDB['with-display']:
47013f9d092SSatish Balay      self.addMakeMacro('DISPLAY',self.framework.argDB['with-display'])
47113f9d092SSatish Balay
472f8833479SBarry Smith    # add a makefile entry for configure options
473f8833479SBarry Smith    self.addMakeMacro('CONFIGURE_OPTIONS', self.framework.getOptionsString(['configModules', 'optionsModule']).replace('\"','\\"'))
474f8833479SBarry Smith    return
475f8833479SBarry Smith
476f8833479SBarry Smith  def dumpConfigInfo(self):
477f8833479SBarry Smith    import time
478c6ef1b5bSJed Brown    fd = open(os.path.join(self.arch.arch,'include','petscconfiginfo.h'),'w')
479dc25a686SPierre Jolivet    fd.write('static const char *petscconfigureoptions = "'+self.framework.getOptionsString(['configModules', 'optionsModule']).replace('\"','\\"').replace('\\ ','\\\\ ')+'";\n')
480f8833479SBarry Smith    fd.close()
481f8833479SBarry Smith    return
482f8833479SBarry Smith
4832a4161d9SMatthew G Knepley  def dumpMachineInfo(self):
4842a4161d9SMatthew G Knepley    import platform
485a970bd74SBernhard M. Wiedemann    import datetime
4862a4161d9SMatthew G Knepley    import time
48740373944SSatish Balay    import script
488ca77dbeeSGeoffrey Irving    def escape(s):
489e08ecd42SSatish Balay      return s.replace('"',r'\"').replace(r'\ ',r'\\ ') # novermin
490c6ef1b5bSJed Brown    fd = open(os.path.join(self.arch.arch,'include','petscmachineinfo.h'),'w')
4912a4161d9SMatthew G Knepley    fd.write('static const char *petscmachineinfo = \"\\n\"\n')
4922a4161d9SMatthew G Knepley    fd.write('\"-----------------------------------------\\n\"\n')
493a970bd74SBernhard M. Wiedemann    buildhost = platform.node()
494a970bd74SBernhard M. Wiedemann    if os.environ.get('SOURCE_DATE_EPOCH'):
495a970bd74SBernhard M. Wiedemann      buildhost = "reproducible"
496a970bd74SBernhard M. Wiedemann    buildtime = datetime.datetime.utcfromtimestamp(int(os.environ.get('SOURCE_DATE_EPOCH', time.time())))
497a970bd74SBernhard M. Wiedemann    fd.write('\"Libraries compiled on %s on %s \\n\"\n' % (buildtime, buildhost))
49860acdfe7SSatish Balay    fd.write('\"Machine characteristics: %s\\n\"\n' % (platform.platform()))
4995188cb68SSatish Balay    fd.write('\"Using PETSc directory: %s\\n\"\n' % (escape(self.installdir.petscDir)))
5005188cb68SSatish Balay    fd.write('\"Using PETSc arch: %s\\n\"\n' % (escape(self.installdir.petscArch)))
501cdec380aSBarry Smith    fd.write('\"-----------------------------------------\\n\";\n')
5022a4161d9SMatthew G Knepley    fd.write('static const char *petsccompilerinfo = \"\\n\"\n')
5032a4161d9SMatthew G Knepley    self.setCompilers.pushLanguage(self.languages.clanguage)
5045f27b2e0SBarry Smith    fd.write('\"Using C compiler: %s %s \\n\"\n' % (escape(self.setCompilers.getCompiler()), escape(self.setCompilers.getCompilerFlags())))
5052a4161d9SMatthew G Knepley    self.setCompilers.popLanguage()
5068782282cSMatthew G Knepley    if hasattr(self.compilers, 'FC'):
5072a4161d9SMatthew G Knepley      self.setCompilers.pushLanguage('FC')
5085f27b2e0SBarry Smith      fd.write('\"Using Fortran compiler: %s %s  %s\\n\"\n' % (escape(self.setCompilers.getCompiler()), escape(self.setCompilers.getCompilerFlags()), escape(self.setCompilers.CPPFLAGS)))
5092a4161d9SMatthew G Knepley      self.setCompilers.popLanguage()
510cdec380aSBarry Smith    fd.write('\"-----------------------------------------\\n\";\n')
5112a4161d9SMatthew G Knepley    fd.write('static const char *petsccompilerflagsinfo = \"\\n\"\n')
5125a21677cSJed Brown    fd.write('\"Using include paths: %s\\n\"\n' % (escape(self.PETSC_CC_INCLUDES_INSTALL.replace('${PETSC_DIR}', self.installdir.petscDir))))
513cdec380aSBarry Smith    fd.write('\"-----------------------------------------\\n\";\n')
5142a4161d9SMatthew G Knepley    fd.write('static const char *petsclinkerinfo = \"\\n\"\n')
5152a4161d9SMatthew G Knepley    self.setCompilers.pushLanguage(self.languages.clanguage)
516ca77dbeeSGeoffrey Irving    fd.write('\"Using C linker: %s\\n\"\n' % (escape(self.setCompilers.getLinker())))
5172a4161d9SMatthew G Knepley    self.setCompilers.popLanguage()
5188782282cSMatthew G Knepley    if hasattr(self.compilers, 'FC'):
5192a4161d9SMatthew G Knepley      self.setCompilers.pushLanguage('FC')
520ca77dbeeSGeoffrey Irving      fd.write('\"Using Fortran linker: %s\\n\"\n' % (escape(self.setCompilers.getLinker())))
5212a4161d9SMatthew G Knepley      self.setCompilers.popLanguage()
5225188cb68SSatish 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)))
523cdec380aSBarry Smith    fd.write('\"-----------------------------------------\\n\";\n')
5242a4161d9SMatthew G Knepley    fd.close()
5252a4161d9SMatthew G Knepley    return
526b2843cf1SBarry Smith
527b2843cf1SBarry Smith  def configurePrefetch(self):
528b2843cf1SBarry Smith    '''Sees if there are any prefetch functions supported'''
5297fca349cSMatthew G. Knepley    if config.setCompilers.Configure.isSolaris(self.log) or self.framework.argDB['with-ios'] or not self.framework.argDB['with-prefetch']:
53093f78423SSatish Balay      self.addDefine('Prefetch(a,b,c)', ' ')
53193f78423SSatish Balay      return
532ec284106SBarry Smith    self.pushLanguage(self.languages.clanguage)
53310699583SJed Brown    if self.checkLink('#include <xmmintrin.h>', 'void *v = 0;_mm_prefetch((const char*)v,_MM_HINT_NTA);\n'):
53450d8bf02SJed Brown      # The Intel Intrinsics manual [1] specifies the prototype
53550d8bf02SJed Brown      #
53650d8bf02SJed Brown      #   void _mm_prefetch(char const *a, int sel);
53750d8bf02SJed Brown      #
53850d8bf02SJed Brown      # but other vendors seem to insist on using subtly different
53950d8bf02SJed Brown      # prototypes, including void* for the pointer, and an enum for
54050d8bf02SJed Brown      # sel.  These are both reasonable changes, but negatively impact
54150d8bf02SJed Brown      # portability.
54250d8bf02SJed Brown      #
543a8d69d7bSBarry Smith      # [1] https://software.intel.com/file/6373
54450d8bf02SJed Brown      self.addDefine('HAVE_XMMINTRIN_H', 1)
54550d8bf02SJed Brown      self.addDefine('Prefetch(a,b,c)', '_mm_prefetch((const char*)(a),(c))')
54650d8bf02SJed Brown      self.addDefine('PREFETCH_HINT_NTA', '_MM_HINT_NTA')
54750d8bf02SJed Brown      self.addDefine('PREFETCH_HINT_T0',  '_MM_HINT_T0')
54850d8bf02SJed Brown      self.addDefine('PREFETCH_HINT_T1',  '_MM_HINT_T1')
54950d8bf02SJed Brown      self.addDefine('PREFETCH_HINT_T2',  '_MM_HINT_T2')
55050d8bf02SJed Brown    elif self.checkLink('#include <xmmintrin.h>', 'void *v = 0;_mm_prefetch(v,_MM_HINT_NTA);\n'):
55150d8bf02SJed Brown      self.addDefine('HAVE_XMMINTRIN_H', 1)
55250d8bf02SJed Brown      self.addDefine('Prefetch(a,b,c)', '_mm_prefetch((const void*)(a),(c))')
55350d8bf02SJed Brown      self.addDefine('PREFETCH_HINT_NTA', '_MM_HINT_NTA')
55450d8bf02SJed Brown      self.addDefine('PREFETCH_HINT_T0',  '_MM_HINT_T0')
55550d8bf02SJed Brown      self.addDefine('PREFETCH_HINT_T1',  '_MM_HINT_T1')
55650d8bf02SJed Brown      self.addDefine('PREFETCH_HINT_T2',  '_MM_HINT_T2')
55710699583SJed Brown    elif self.checkLink('', 'void *v = 0;__builtin_prefetch(v,0,0);\n'):
55810699583SJed Brown      # From GCC docs: void __builtin_prefetch(const void *addr,int rw,int locality)
55910699583SJed Brown      #
56010699583SJed Brown      #   The value of rw is a compile-time constant one or zero; one
56110699583SJed Brown      #   means that the prefetch is preparing for a write to the memory
56210699583SJed Brown      #   address and zero, the default, means that the prefetch is
56310699583SJed Brown      #   preparing for a read. The value locality must be a compile-time
56410699583SJed Brown      #   constant integer between zero and three. A value of zero means
56510699583SJed Brown      #   that the data has no temporal locality, so it need not be left
56610699583SJed Brown      #   in the cache after the access. A value of three means that the
56710699583SJed Brown      #   data has a high degree of temporal locality and should be left
56810699583SJed Brown      #   in all levels of cache possible. Values of one and two mean,
56910699583SJed Brown      #   respectively, a low or moderate degree of temporal locality.
57010699583SJed Brown      #
57110699583SJed Brown      # Here we adopt Intel's x86/x86-64 naming scheme for the locality
57210699583SJed Brown      # hints.  Using macros for these values in necessary since some
57310699583SJed Brown      # compilers require an enum.
57410699583SJed Brown      self.addDefine('Prefetch(a,b,c)', '__builtin_prefetch((a),(b),(c))')
57510699583SJed Brown      self.addDefine('PREFETCH_HINT_NTA', '0')
57610699583SJed Brown      self.addDefine('PREFETCH_HINT_T0',  '3')
57710699583SJed Brown      self.addDefine('PREFETCH_HINT_T1',  '2')
57810699583SJed Brown      self.addDefine('PREFETCH_HINT_T2',  '1')
579b2843cf1SBarry Smith    else:
580b2843cf1SBarry Smith      self.addDefine('Prefetch(a,b,c)', ' ')
5817d490b44SBarry Smith    self.popLanguage()
582b2843cf1SBarry Smith
58349fe22e6SSatish Balay  def delGenFiles(self):
58449fe22e6SSatish Balay    '''Delete generated files'''
58549fe22e6SSatish Balay    delfile = os.path.join(self.arch.arch,'lib','petsc','conf','files')
58649fe22e6SSatish Balay    try:
58749fe22e6SSatish Balay      os.unlink(delfile)
58849fe22e6SSatish Balay    except: pass
58949fe22e6SSatish Balay
59009bc878fSSatish Balay  def configureAtoll(self):
59109bc878fSSatish Balay    '''Checks if atoll exists'''
592436b02dcSSatish Balay    if self.checkLink('#define _POSIX_C_SOURCE 200112L\n#include <stdlib.h>','long v = atoll("25")') or self.checkLink ('#include <stdlib.h>','long v = atoll("25")'):
59309bc878fSSatish Balay       self.addDefine('HAVE_ATOLL', '1')
59409bc878fSSatish Balay
5952400fdedSBarry Smith  def configureUnused(self):
5962400fdedSBarry Smith    '''Sees if __attribute((unused)) is supported'''
5971adaff47SSean Farley    if self.framework.argDB['with-ios']:
5982400fdedSBarry Smith      self.addDefine('UNUSED', ' ')
5992400fdedSBarry Smith      return
6002400fdedSBarry Smith    self.pushLanguage(self.languages.clanguage)
601edf21b64SSatish Balay    if self.checkLink('__attribute((unused)) static int myfunc(__attribute((unused)) void *name){ return 1;}', 'int i = 0;\nint j = myfunc(&i);\ntypedef void* atype;\n__attribute((unused))  atype a;\n'):
6022400fdedSBarry Smith      self.addDefine('UNUSED', '__attribute((unused))')
6032400fdedSBarry Smith    else:
6042400fdedSBarry Smith      self.addDefine('UNUSED', ' ')
6052400fdedSBarry Smith    self.popLanguage()
6062400fdedSBarry Smith
60798ed35c3SBarry Smith  def configureIsatty(self):
60898ed35c3SBarry Smith    '''Check if the Unix C function isatty() works correctly
60998ed35c3SBarry Smith       Actually just assumes it does not work correctly on batch systems'''
61098ed35c3SBarry Smith    if not self.framework.argDB['with-batch']:
61198ed35c3SBarry Smith      self.addDefine('USE_ISATTY',1)
61298ed35c3SBarry Smith
6131ef8df7fSJed Brown  def configureDeprecated(self):
6141ef8df7fSJed Brown    '''Check if __attribute((deprecated)) is supported'''
6151ef8df7fSJed Brown    self.pushLanguage(self.languages.clanguage)
61659a26b54SJed Brown    ## Recent versions of gcc and clang support __attribute((deprecated("string argument"))), which is very useful, but
61759a26b54SJed Brown    ## Intel has conspired to make a supremely environment-sensitive compiler.  The Intel compiler looks at the gcc
61859a26b54SJed Brown    ## executable in the environment to determine the language compatibility that it should attempt to emulate.  Some
61959a26b54SJed Brown    ## important Cray installations have built PETSc using the Intel compiler, but with a newer gcc module loaded (e.g.,
620df3898eeSBarry Smith    ## 4.7).  Thus at PETSc configure time, the Intel compiler decides to support the string argument, but the gcc
62159a26b54SJed Brown    ## found in the default user environment is older and does not support the argument.  If GCC and Intel were cool
62259a26b54SJed Brown    ## like Clang and supported __has_attribute, we could avoid configure tests entirely, but they don't.  And that is
62359a26b54SJed Brown    ## why we can't have nice things.
62459a26b54SJed Brown    #
62559a26b54SJed Brown    # if self.checkCompile("""__attribute((deprecated("Why you shouldn't use myfunc"))) static int myfunc(void) { return 1;}""", ''):
62625ef9dfeSBarry Smith    #   self.addDefine('DEPRECATED_FUNCTION(why)', '__attribute((deprecated(why)))')
62725ef9dfeSBarry Smith    #   self.addDefine('DEPRECATED_TYPEDEF(why)', '__attribute((deprecated(why)))')
62859a26b54SJed Brown    if self.checkCompile("""__attribute((deprecated)) static int myfunc(void) { return 1;}""", ''):
62925ef9dfeSBarry Smith      self.addDefine('DEPRECATED_FUNCTION(why)', '__attribute((deprecated))')
63025ef9dfeSBarry Smith      self.addDefine('DEPRECATED_TYPEDEF(why)', '__attribute((deprecated))')
6311ef8df7fSJed Brown    else:
63225ef9dfeSBarry Smith      self.addDefine('DEPRECATED_FUNCTION(why)', ' ')
63325ef9dfeSBarry Smith      self.addDefine('DEPRECATED_TYPEDEF(why)', ' ')
634a8678870SSatish Balay    if self.checkCompile("""enum E {oldval __attribute((deprecated)), newval };""", ''):
635933a48e8SSatish Balay      self.addDefine('DEPRECATED_ENUM(why)', '__attribute((deprecated))')
636933a48e8SSatish Balay    else:
637933a48e8SSatish Balay      self.addDefine('DEPRECATED_ENUM(why)', ' ')
6388e5aa403SBarry Smith    # I was unable to make a CPP macro that takes the old and new values as separate arguments and builds the message needed by _Pragma
63905de396fSBarry Smith    # hence the deprecation message is handled as it is
64005de396fSBarry Smith    if self.checkCompile('#define TEST _Pragma("GCC warning \"Testing _Pragma\"") value'):
64105de396fSBarry Smith      self.addDefine('DEPRECATED_MACRO(why)', '_Pragma(why)')
64205de396fSBarry Smith    else:
64305de396fSBarry Smith      self.addDefine('DEPRECATED_MACRO(why)', ' ')
6441ef8df7fSJed Brown    self.popLanguage()
6451ef8df7fSJed Brown
64618f41590SBarry Smith  def configureAlign(self):
6477b7fc14bSLisandro Dalcin    '''Check if __attribute(aligned) is supported'''
6487b7fc14bSLisandro Dalcin    code = '''\
649752d89a4SSatish Balaystruct mystruct {int myint;} __attribute((aligned(16)));
6507b7fc14bSLisandro Dalcinchar assert_aligned[(sizeof(struct mystruct)==16)*2-1];
651752d89a4SSatish Balay'''
652752d89a4SSatish Balay    self.pushLanguage(self.languages.clanguage)
6537b7fc14bSLisandro Dalcin    if self.checkCompile(code):
654752d89a4SSatish Balay      self.addDefine('ATTRIBUTEALIGNED(size)', '__attribute((aligned(size)))')
655752d89a4SSatish Balay      self.addDefine('HAVE_ATTRIBUTEALIGNED', 1)
656752d89a4SSatish Balay    else:
6577b7fc14bSLisandro Dalcin      self.framework.logPrint('Incorrect attribute(aligned)')
658752d89a4SSatish Balay      self.addDefine('ATTRIBUTEALIGNED(size)', ' ')
6597b7fc14bSLisandro Dalcin    self.popLanguage()
660752d89a4SSatish Balay    return
66118f41590SBarry Smith
6629800092aSJed Brown  def configureExpect(self):
6639800092aSJed Brown    '''Sees if the __builtin_expect directive is supported'''
6649800092aSJed Brown    self.pushLanguage(self.languages.clanguage)
6659800092aSJed Brown    if self.checkLink('', 'if (__builtin_expect(0,1)) return 1;'):
6669800092aSJed Brown      self.addDefine('HAVE_BUILTIN_EXPECT', 1)
6679800092aSJed Brown    self.popLanguage()
6689800092aSJed Brown
66953c77d0aSJed Brown  def configureFunctionName(self):
670fbfcfee5SBarry Smith    '''Sees if the compiler supports __func__ or a variant.'''
6711ec50b02SJed Brown    def getFunctionName(lang):
672fbfcfee5SBarry Smith      name = '"unknown"'
6731ec50b02SJed Brown      self.pushLanguage(lang)
674b6ff4c76SKarl Rupp      for fname in ['__func__','__FUNCTION__','__extension__ __func__']:
6750117e5a1SSatish Balay        code = "if ("+fname+"[0] != 'm') return 1;"
6760117e5a1SSatish Balay        if self.checkCompile('',code) and self.checkLink('',code):
6770117e5a1SSatish Balay          name = fname
6780117e5a1SSatish Balay          break
6791ec50b02SJed Brown      self.popLanguage()
6801ec50b02SJed Brown      return name
6811ec50b02SJed Brown    langs = []
682628773c9SSatish Balay
683628773c9SSatish Balay    self.addDefine('FUNCTION_NAME_C', getFunctionName('C'))
6845f6e5f85SSatish Balay    if hasattr(self.compilers, 'CXX'):
685628773c9SSatish Balay      self.addDefine('FUNCTION_NAME_CXX', getFunctionName('Cxx'))
68653c77d0aSJed Brown
687753ebd1dSJed Brown  def configureIntptrt(self):
688753ebd1dSJed Brown    '''Determine what to use for uintptr_t'''
689753ebd1dSJed Brown    def staticAssertSizeMatchesVoidStar(inc,typename):
690753ebd1dSJed Brown      # The declaration is an error if either array size is negative.
691753ebd1dSJed 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
692d26187a0SJed Brown      return self.checkCompile(inc, ('#define STATIC_ASSERT(cond) char negative_length_if_false[2*(!!(cond))-1]\n'
693979939cdSSatish Balay                                     + 'STATIC_ASSERT(sizeof(void*) == sizeof(%s));'%typename))
694753ebd1dSJed Brown    self.pushLanguage(self.languages.clanguage)
695753ebd1dSJed Brown    if self.checkCompile('#include <stdint.h>', 'int x; uintptr_t i = (uintptr_t)&x;'):
696753ebd1dSJed Brown      self.addDefine('UINTPTR_T', 'uintptr_t')
697753ebd1dSJed Brown    elif staticAssertSizeMatchesVoidStar('','unsigned long long'):
698753ebd1dSJed Brown      self.addDefine('UINTPTR_T', 'unsigned long long')
699753ebd1dSJed Brown    elif staticAssertSizeMatchesVoidStar('#include <stdlib.h>','size_t') or staticAssertSizeMatchesVoidStar('#include <string.h>', 'size_t'):
700753ebd1dSJed Brown      self.addDefine('UINTPTR_T', 'size_t')
701c82284b1SJed Brown    elif staticAssertSizeMatchesVoidStar('','unsigned long'):
702c82284b1SJed Brown      self.addDefine('UINTPTR_T', 'unsigned long')
7032d1b7972SSatish Balay    elif staticAssertSizeMatchesVoidStar('','unsigned'):
704753ebd1dSJed Brown      self.addDefine('UINTPTR_T', 'unsigned')
705d26187a0SJed Brown    else:
706d26187a0SJed Brown      raise RuntimeError('Could not find any unsigned integer type matching void*')
707753ebd1dSJed Brown    self.popLanguage()
708753ebd1dSJed Brown
709ed938b00SJed Brown  def configureRTLDDefault(self):
710bfef2c86SBarry Smith    if self.checkCompile('#include <dlfcn.h>\n void *ptr =  RTLD_DEFAULT;'):
711bfef2c86SBarry Smith      self.addDefine('RTLD_DEFAULT','1')
712f8833479SBarry Smith    return
713f8833479SBarry Smith
714f8833479SBarry Smith  def configureSolaris(self):
715f8833479SBarry Smith    '''Solaris specific stuff'''
716f8833479SBarry Smith    if os.path.isdir(os.path.join('/usr','ucblib')):
717f8833479SBarry Smith      try:
718f8833479SBarry Smith        flag = getattr(self.setCompilers, self.language[-1]+'SharedLinkerFlag')
719f8833479SBarry Smith      except AttributeError:
720f8833479SBarry Smith        flag = None
721f8833479SBarry Smith      if flag is None:
722f8833479SBarry Smith        self.compilers.LIBS += ' -L/usr/ucblib'
723f8833479SBarry Smith      else:
724f8833479SBarry Smith        self.compilers.LIBS += ' '+flag+'/usr/ucblib'
725f8833479SBarry Smith    return
726f8833479SBarry Smith
7270f64ec89SBarry Smith  def configureDarwin(self):
7280f64ec89SBarry Smith    '''Log brew configuration for Apple systems'''
7290f64ec89SBarry Smith    try:
7300f64ec89SBarry Smith      self.executeShellCommand(['brew', 'config'], log = self.log)
7310f64ec89SBarry Smith      self.executeShellCommand(['brew', 'info', 'gcc'], log = self.log)
7320f64ec89SBarry Smith    except:
7330f64ec89SBarry Smith      pass
7340f64ec89SBarry Smith    return
7350f64ec89SBarry Smith
736f8833479SBarry Smith  def configureLinux(self):
737f8833479SBarry Smith    '''Linux specific stuff'''
7389f15855cSMatthew G Knepley    # TODO: Test for this by mallocing an odd number of floats and checking the address
739f8833479SBarry Smith    self.addDefine('HAVE_DOUBLE_ALIGN_MALLOC', 1)
740f8833479SBarry Smith    return
741f8833479SBarry Smith
742f8833479SBarry Smith  def configureWin32(self):
743f8833479SBarry Smith    '''Win32 non-cygwin specific stuff'''
744f8833479SBarry Smith    kernel32=0
7454e8afd12SMosè Giordano    if self.libraries.add('Kernel32.lib','GetComputerName',prototype='#include <windows.h>', call='GetComputerName(NULL,NULL);'):
746f8833479SBarry Smith      self.addDefine('HAVE_WINDOWS_H',1)
747f8833479SBarry Smith      self.addDefine('HAVE_GETCOMPUTERNAME',1)
748f8833479SBarry Smith      kernel32=1
7494e8afd12SMosè Giordano    elif self.libraries.add('kernel32','GetComputerName',prototype='#include <windows.h>', call='GetComputerName(NULL,NULL);'):
750f8833479SBarry Smith      self.addDefine('HAVE_WINDOWS_H',1)
751f8833479SBarry Smith      self.addDefine('HAVE_GETCOMPUTERNAME',1)
752f8833479SBarry Smith      kernel32=1
753f8833479SBarry Smith    if kernel32:
754eed94e11SSatish Balay      if self.framework.argDB['with-windows-graphics']:
755eed94e11SSatish Balay        self.addDefine('USE_WINDOWS_GRAPHICS',1)
7564e8afd12SMosè Giordano      if self.checkLink('#include <windows.h>','LoadLibrary(0)'):
757f8833479SBarry Smith        self.addDefine('HAVE_LOADLIBRARY',1)
7584e8afd12SMosè Giordano      if self.checkLink('#include <windows.h>','GetProcAddress(0,0)'):
759b50f6d9eSLisandro Dalcin        self.addDefine('HAVE_GETPROCADDRESS',1)
7604e8afd12SMosè Giordano      if self.checkLink('#include <windows.h>','FreeLibrary(0)'):
761b50f6d9eSLisandro Dalcin        self.addDefine('HAVE_FREELIBRARY',1)
7624e8afd12SMosè Giordano      if self.checkLink('#include <windows.h>','GetLastError()'):
763a21658a3SLisandro Dalcin        self.addDefine('HAVE_GETLASTERROR',1)
7644e8afd12SMosè Giordano      if self.checkLink('#include <windows.h>','SetLastError(0)'):
765a21658a3SLisandro Dalcin        self.addDefine('HAVE_SETLASTERROR',1)
7664e8afd12SMosè Giordano      if self.checkLink('#include <windows.h>\n','QueryPerformanceCounter(0);\n'):
767bea725cfSBarry Smith        self.addDefine('USE_MICROSOFT_TIME',1)
7684e8afd12SMosè Giordano    if self.libraries.add('Advapi32.lib','GetUserName',prototype='#include <windows.h>', call='GetUserName(NULL,NULL);'):
769f8833479SBarry Smith      self.addDefine('HAVE_GET_USER_NAME',1)
7704e8afd12SMosè Giordano    elif self.libraries.add('advapi32','GetUserName',prototype='#include <windows.h>', call='GetUserName(NULL,NULL);'):
771f8833479SBarry Smith      self.addDefine('HAVE_GET_USER_NAME',1)
772f8833479SBarry Smith
7734e8afd12SMosè Giordano    if not self.libraries.add('User32.lib','GetDC',prototype='#include <windows.h>',call='GetDC(0);'):
7744e8afd12SMosè Giordano      self.libraries.add('user32','GetDC',prototype='#include <windows.h>',call='GetDC(0);')
7754e8afd12SMosè Giordano    if not self.libraries.add('Gdi32.lib','CreateCompatibleDC',prototype='#include <windows.h>',call='CreateCompatibleDC(0);'):
7764e8afd12SMosè Giordano      self.libraries.add('gdi32','CreateCompatibleDC',prototype='#include <windows.h>',call='CreateCompatibleDC(0);')
777f8833479SBarry Smith
778f8833479SBarry Smith    self.types.check('int32_t', 'int')
779f8833479SBarry Smith    if not self.checkCompile('#include <sys/types.h>\n','uid_t u;\n'):
780f8833479SBarry Smith      self.addTypedef('int', 'uid_t')
781f8833479SBarry Smith      self.addTypedef('int', 'gid_t')
782f8833479SBarry Smith    if not self.checkLink('#if defined(PETSC_HAVE_UNISTD_H)\n#include <unistd.h>\n#endif\n','int a=R_OK;\n'):
783f8833479SBarry Smith      self.framework.addDefine('R_OK', '04')
784f8833479SBarry Smith      self.framework.addDefine('W_OK', '02')
785f8833479SBarry Smith      self.framework.addDefine('X_OK', '01')
786f8833479SBarry Smith    if not self.checkLink('#include <sys/stat.h>\n','int a=0;\nif (S_ISDIR(a)){}\n'):
787f8833479SBarry Smith      self.framework.addDefine('S_ISREG(a)', '(((a)&_S_IFMT) == _S_IFREG)')
788f8833479SBarry Smith      self.framework.addDefine('S_ISDIR(a)', '(((a)&_S_IFMT) == _S_IFDIR)')
7894e8afd12SMosè Giordano    if self.checkCompile('#include <windows.h>\n','LARGE_INTEGER a;\nDWORD b=a.u.HighPart;\n'):
790f8833479SBarry Smith      self.addDefine('HAVE_LARGE_INTEGER_U',1)
791f8833479SBarry Smith
792f8833479SBarry Smith    # Windows requires a Binary file creation flag when creating/opening binary files.  Is a better test in order?
7934e8afd12SMosè Giordano    if self.checkCompile('#include <windows.h>\n#include <fcntl.h>\n', 'int flags = O_BINARY;'):
794f8833479SBarry Smith      self.addDefine('HAVE_O_BINARY',1)
795f8833479SBarry Smith
796f8833479SBarry Smith    if self.compilers.CC.find('win32fe') >= 0:
797ad4212abSSatish Balay      self.addDefine('HAVE_WINDOWS_COMPILERS',1)
798f8833479SBarry Smith      self.addDefine('DIR_SEPARATOR','\'\\\\\'')
799f8833479SBarry Smith      self.addDefine('REPLACE_DIR_SEPARATOR','\'/\'')
800f8833479SBarry Smith      self.addDefine('CANNOT_START_DEBUGGER',1)
8015188cb68SSatish Balay      (petscdir,error,status) = self.executeShellCommand('cygpath -w '+self.installdir.petscDir, log = self.log)
80234531a4dSSatish Balay      self.addDefine('DIR','"'+petscdir.replace('\\','\\\\')+'"')
8035188cb68SSatish Balay      (petscdir,error,status) = self.executeShellCommand('cygpath -m '+self.installdir.petscDir, log = self.log)
804e433681fSSatish Balay      self.addMakeMacro('wPETSC_DIR',petscdir)
8054e00a515SSatish Balay      if self.dataFilesPath.datafilespath:
8064e00a515SSatish Balay        (datafilespath,error,status) = self.executeShellCommand('cygpath -m '+self.dataFilesPath.datafilespath, log = self.log)
8074e00a515SSatish Balay        self.addMakeMacro('DATAFILESPATH',datafilespath)
8084e00a515SSatish Balay
809f8833479SBarry Smith    else:
810f8833479SBarry Smith      self.addDefine('REPLACE_DIR_SEPARATOR','\'\\\\\'')
811f8833479SBarry Smith      self.addDefine('DIR_SEPARATOR','\'/\'')
8125188cb68SSatish Balay      self.addDefine('DIR','"'+self.installdir.petscDir+'"')
8135188cb68SSatish Balay      self.addMakeMacro('wPETSC_DIR',self.installdir.petscDir)
8144e00a515SSatish Balay      if self.dataFilesPath.datafilespath:
8154e00a515SSatish Balay        self.addMakeMacro('DATAFILESPATH',self.dataFilesPath.datafilespath)
8165188cb68SSatish Balay    self.addDefine('ARCH','"'+self.installdir.petscArch+'"')
817f8833479SBarry Smith    return
818f8833479SBarry Smith
819f8833479SBarry Smith#-----------------------------------------------------------------------------------------------------
820b10d012aSSatish Balay  def configureCygwinBrokenPipe(self):
821b10d012aSSatish Balay    '''Cygwin version <= 1.7.18 had issues with pipes and long commands invoked from gnu-make
822b10d012aSSatish Balay    http://cygwin.com/ml/cygwin/2013-05/msg00340.html '''
8237fca349cSMatthew G. Knepley    if config.setCompilers.Configure.isCygwin(self.log):
824b10d012aSSatish Balay      import platform
825b10d012aSSatish Balay      import re
826b10d012aSSatish Balay      r=re.compile("([0-9]+).([0-9]+).([0-9]+)")
827b10d012aSSatish Balay      m=r.match(platform.release())
828b10d012aSSatish Balay      major=int(m.group(1))
829b10d012aSSatish Balay      minor=int(m.group(2))
830b10d012aSSatish Balay      subminor=int(m.group(3))
831b10d012aSSatish Balay      if ((major < 1) or (major == 1 and minor < 7) or (major == 1 and minor == 7 and subminor <= 18)):
832b10d012aSSatish Balay        self.addMakeMacro('PETSC_CYGWIN_BROKEN_PIPE','1')
833b10d012aSSatish Balay    return
834b10d012aSSatish Balay
835b10d012aSSatish Balay#-----------------------------------------------------------------------------------------------------
836569865ddSSatish Balay  def configureDefaultArch(self):
837af0996ceSBarry Smith    conffile = os.path.join('lib','petsc','conf', 'petscvariables')
838569865ddSSatish Balay    if self.framework.argDB['with-default-arch']:
839c6ef1b5bSJed Brown      fd = open(conffile, 'w')
840569865ddSSatish Balay      fd.write('PETSC_ARCH='+self.arch.arch+'\n')
841da93591fSBarry Smith      fd.write('PETSC_DIR='+self.petscdir.dir+'\n')
842b9b902edSJed Brown      fd.write('include '+os.path.join('$(PETSC_DIR)','$(PETSC_ARCH)','lib','petsc','conf','petscvariables')+'\n')
843569865ddSSatish Balay      fd.close()
844569865ddSSatish Balay      self.framework.actions.addArgument('PETSc', 'Build', 'Set default architecture to '+self.arch.arch+' in '+conffile)
845569865ddSSatish Balay    elif os.path.isfile(conffile):
846569865ddSSatish Balay      try:
847569865ddSSatish Balay        os.unlink(conffile)
848569865ddSSatish Balay      except:
849569865ddSSatish Balay        raise RuntimeError('Unable to remove file '+conffile+'. Did a different user create it?')
850569865ddSSatish Balay    return
851569865ddSSatish Balay
852569865ddSSatish Balay#-----------------------------------------------------------------------------------------------------
853f8833479SBarry Smith  def configureScript(self):
854f8833479SBarry Smith    '''Output a script in the conf directory which will reproduce the configuration'''
855f8833479SBarry Smith    import nargs
856495bf1a9SSatish Balay    import sys
857af0996ceSBarry Smith    scriptName = os.path.join(self.arch.arch,'lib','petsc','conf', 'reconfigure-'+self.arch.arch+'.py')
858f8833479SBarry Smith    args = dict([(nargs.Arg.parseArgument(arg)[0], arg) for arg in self.framework.clArgs])
859e97fc2efSSatish Balay    if 'with-clean' in args:
860e97fc2efSSatish Balay      del args['with-clean']
861d418e2d7SSatish Balay    if 'force' in args:
862d418e2d7SSatish Balay      del args['force']
863f8833479SBarry Smith    if 'configModules' in args:
8641063a081SSatish Balay      if nargs.Arg.parseArgument(args['configModules'])[1] == 'PETSc.Configure':
865f8833479SBarry Smith        del args['configModules']
866f8833479SBarry Smith    if 'optionsModule' in args:
86723a19ef1SSatish Balay      if nargs.Arg.parseArgument(args['optionsModule'])[1] == 'config.compilerOptions':
868f8833479SBarry Smith        del args['optionsModule']
869f8833479SBarry Smith    if not 'PETSC_ARCH' in args:
8701063a081SSatish Balay      args['PETSC_ARCH'] = 'PETSC_ARCH='+str(self.arch.arch)
871c6ef1b5bSJed Brown    f = open(scriptName, 'w')
872495bf1a9SSatish Balay    f.write('#!'+sys.executable+'\n')
873f8833479SBarry Smith    f.write('if __name__ == \'__main__\':\n')
874f8833479SBarry Smith    f.write('  import sys\n')
8757561c02cSSatish Balay    f.write('  import os\n')
8767561c02cSSatish Balay    f.write('  sys.path.insert(0, os.path.abspath(\'config\'))\n')
877f8833479SBarry Smith    f.write('  import configure\n')
8781063a081SSatish Balay    # pretty print repr(args.values())
8791063a081SSatish Balay    f.write('  configure_options = [\n')
8808bec23c5SJed Brown    for itm in sorted(args.values()):
8811063a081SSatish Balay      f.write('    \''+str(itm)+'\',\n')
8821063a081SSatish Balay    f.write('  ]\n')
883f8833479SBarry Smith    f.write('  configure.petsc_configure(configure_options)\n')
884f8833479SBarry Smith    f.close()
885f8833479SBarry Smith    try:
8865b6bfdb9SJed Brown      os.chmod(scriptName, 0o775)
8875b6bfdb9SJed Brown    except OSError as e:
888f8833479SBarry Smith      self.framework.logPrint('Unable to make reconfigure script executable:\n'+str(e))
889f8833479SBarry Smith    self.framework.actions.addArgument('PETSc', 'File creation', 'Created '+scriptName+' for automatic reconfiguration')
890f8833479SBarry Smith    return
891f8833479SBarry Smith
892f8833479SBarry Smith  def configureInstall(self):
893f8833479SBarry Smith    '''Setup the directories for installation'''
894f8833479SBarry Smith    if self.framework.argDB['prefix']:
8955b4fc442SVaclav Hapla      self.addMakeRule('print_mesg_after_build','',
8965b4fc442SVaclav Hapla       ['-@echo "========================================="',
8975b4fc442SVaclav Hapla        '-@echo "Now to install the libraries do:"',
8985b4fc442SVaclav Hapla        '-@echo "%s${MAKE_USER} PETSC_DIR=${PETSC_DIR} PETSC_ARCH=${PETSC_ARCH} install"' % self.installdir.installSudo,
899315b77e6SSatish Balay        '-@echo "========================================="'])
900f8833479SBarry Smith    else:
9015b4fc442SVaclav Hapla      self.addMakeRule('print_mesg_after_build','',
9025b4fc442SVaclav Hapla       ['-@echo "========================================="',
9035b4fc442SVaclav Hapla        '-@echo "Now to check if the libraries are working do:"',
9045b4fc442SVaclav Hapla        '-@echo "${MAKE_USER} PETSC_DIR=${PETSC_DIR} PETSC_ARCH=${PETSC_ARCH} check"',
905315b77e6SSatish Balay        '-@echo "========================================="'])
906f8833479SBarry Smith      return
907f8833479SBarry Smith
908f8833479SBarry Smith  def configureGCOV(self):
909f8833479SBarry Smith    if self.framework.argDB['with-gcov']:
910f8833479SBarry Smith      self.addDefine('USE_GCOV','1')
911f8833479SBarry Smith    return
912f8833479SBarry Smith
91328bb2e72SSatish Balay  def postProcessPackages(self):
91428bb2e72SSatish Balay    postPackages=[]
91528bb2e72SSatish Balay    for i in self.framework.packages:
91628bb2e72SSatish Balay      if hasattr(i,'postProcess'): postPackages.append(i)
91728bb2e72SSatish Balay    if postPackages:
918e64d19dfSSatish Balay      # ctetgen needs petsc conf files. so attempt to create them early
919a77eb93bSSatish Balay      self.framework.dumpConfFiles()
920d9293e7bSBarry Smith      # tacky fix for dependency of Aluimia on Pflotran; requested via petsc-dev Matt provide a correct fix
921d9293e7bSBarry Smith      for i in postPackages:
922d9293e7bSBarry Smith        if i.name.upper() in ['PFLOTRAN']:
923d9293e7bSBarry Smith          i.postProcess()
924d9293e7bSBarry Smith          postPackages.remove(i)
92528bb2e72SSatish Balay      for i in postPackages: i.postProcess()
926aa5c8b8eSBarry Smith      for i in postPackages:
927aa5c8b8eSBarry Smith        if i.installedpetsc:
928aa5c8b8eSBarry Smith          self.installed = 1
929aa5c8b8eSBarry Smith          break
93028bb2e72SSatish Balay    return
931f8833479SBarry Smith
932f8833479SBarry Smith  def configure(self):
933bf3e94a3SBarry Smith    if 'package-prefix-hash' in self.argDB:
934bf3e94a3SBarry Smith      # turn off prefix if it was only used to for installing external packages.
935bf3e94a3SBarry Smith      self.framework.argDB['prefix'] = ''
936bf3e94a3SBarry Smith      self.dir = os.path.abspath(os.path.join(self.petscdir.dir, self.arch.arch))
937bf3e94a3SBarry Smith      self.installdir.dir = self.dir
938bf3e94a3SBarry Smith      self.installdir.petscDir = self.petscdir.dir
939bf3e94a3SBarry Smith      self.petscDir = self.petscdir.dir
940bf3e94a3SBarry Smith      self.petscArch = self.arch.arch
941bf3e94a3SBarry Smith      self.addMakeMacro('PREFIXDIR',self.dir)
942bf3e94a3SBarry Smith      self.confDir = os.path.abspath(os.path.join(self.petscdir.dir, self.arch.arch))
943bf3e94a3SBarry Smith
944f8833479SBarry Smith    if not os.path.samefile(self.petscdir.dir, os.getcwd()):
945f8833479SBarry Smith      raise RuntimeError('Wrong PETSC_DIR option specified: '+str(self.petscdir.dir) + '\n  Configure invoked in: '+os.path.realpath(os.getcwd()))
946550489e3SMatthew 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):
9473552d8fbSSatish Balay      raise RuntimeError('Incorrect option --prefix='+self.framework.argDB['prefix']+' specified. It cannot be same as PETSC_DIR!')
9488fd0dbdbSBarry Smith    if self.framework.argDB['prefix'] and self.framework.argDB['prefix'].find(' ') > -1:
9498fd0dbdbSBarry 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')
950c16c35a9SSatish 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)):
951c16c35a9SSatish Balay      raise RuntimeError('Incorrect option --prefix='+self.framework.argDB['prefix']+' specified. It cannot be same as PETSC_DIR/PETSC_ARCH!')
952f16c1317SJed Brown    self.framework.header          = os.path.join(self.arch.arch,'include','petscconf.h')
953f16c1317SJed Brown    self.framework.cHeader         = os.path.join(self.arch.arch,'include','petscfix.h')
954bf113f49SJacob Faibussowitsch    self.framework.poisonheader    = os.path.join(self.arch.arch,'include','petscconf_poison.h')
9559c735a01SStefano Zampini    self.framework.pkgheader       = os.path.join(self.arch.arch,'include','petscpkg_version.h')
956af0996ceSBarry Smith    self.framework.makeMacroHeader = os.path.join(self.arch.arch,'lib','petsc','conf','petscvariables')
957af0996ceSBarry Smith    self.framework.makeRuleHeader  = os.path.join(self.arch.arch,'lib','petsc','conf','petscrules')
958f8833479SBarry Smith    if self.libraries.math is None:
959f8833479SBarry Smith      raise RuntimeError('PETSc requires a functional math library. Please send configure.log to petsc-maint@mcs.anl.gov.')
960f8833479SBarry Smith    if self.languages.clanguage == 'Cxx' and not hasattr(self.compilers, 'CXX'):
961f8833479SBarry Smith      raise RuntimeError('Cannot set C language to C++ without a functional C++ compiler.')
962ed938b00SJed Brown    self.executeTest(self.configureRTLDDefault)
963b2843cf1SBarry Smith    self.executeTest(self.configurePrefetch)
9642400fdedSBarry Smith    self.executeTest(self.configureUnused)
9651ef8df7fSJed Brown    self.executeTest(self.configureDeprecated)
96698ed35c3SBarry Smith    self.executeTest(self.configureIsatty)
967e8e972b2SVaclav Hapla    self.executeTest(self.configureExpect)
968e8e972b2SVaclav Hapla    self.executeTest(self.configureAlign)
969e8e972b2SVaclav Hapla    self.executeTest(self.configureFunctionName)
970e8e972b2SVaclav Hapla    self.executeTest(self.configureIntptrt)
971f8833479SBarry Smith    self.executeTest(self.configureSolaris)
972f8833479SBarry Smith    self.executeTest(self.configureLinux)
9730f64ec89SBarry Smith    self.executeTest(self.configureDarwin)
974f8833479SBarry Smith    self.executeTest(self.configureWin32)
975b10d012aSSatish Balay    self.executeTest(self.configureCygwinBrokenPipe)
976569865ddSSatish Balay    self.executeTest(self.configureDefaultArch)
977f8833479SBarry Smith    self.executeTest(self.configureScript)
978f8833479SBarry Smith    self.executeTest(self.configureInstall)
979f8833479SBarry Smith    self.executeTest(self.configureGCOV)
98009bc878fSSatish Balay    self.executeTest(self.configureAtoll)
981f8833479SBarry Smith
982f8833479SBarry Smith    self.Dump()
983f8833479SBarry Smith    self.dumpConfigInfo()
9842a4161d9SMatthew G Knepley    self.dumpMachineInfo()
98549fe22e6SSatish Balay    self.delGenFiles()
98640277576SBarry Smith    # need to save the current state of BuildSystem so that postProcess() packages can read it in and perhaps run make install
98740277576SBarry Smith    self.framework.storeSubstitutions(self.framework.argDB)
988492432c8SJed Brown    self.framework.argDB['configureCache'] = pickle.dumps(self.framework)
98940277576SBarry Smith    self.framework.argDB.save(force = True)
9908244ab14SJed Brown    self.DumpPkgconfig('PETSc.pc')
9918244ab14SJed Brown    self.DumpPkgconfig('petsc.pc')
992351d3a41SMatthew G Knepley    self.DumpModule()
993f7ad81e1SBarry Smith    self.postProcessPackages()
994f8833479SBarry Smith    self.framework.log.write('================================================================================\n')
995f8833479SBarry Smith    self.logClear()
996f8833479SBarry Smith    return
997