xref: /petsc/config/PETSc/Configure.py (revision 7bf3d34b3ee2f90bf8e9d9687cef35fcc008df20)
1f8833479SBarry Smithimport config.base
2f8833479SBarry Smith
3f8833479SBarry Smithimport os
4f8833479SBarry Smithimport re
5f8833479SBarry Smith
68bec23c5SJed Brown# The sorted() builtin is not available with python-2.3
78bec23c5SJed Browntry: sorted
88bec23c5SJed Brownexcept NameError:
98bec23c5SJed Brown  def sorted(lst):
108bec23c5SJed Brown    lst.sort()
118bec23c5SJed Brown    return lst
128bec23c5SJed Brown
13f8833479SBarry Smithclass Configure(config.base.Configure):
14f8833479SBarry Smith  def __init__(self, framework):
15f8833479SBarry Smith    config.base.Configure.__init__(self, framework)
16f8833479SBarry Smith    self.headerPrefix = 'PETSC'
17f8833479SBarry Smith    self.substPrefix  = 'PETSC'
18f8833479SBarry Smith    return
19f8833479SBarry Smith
207c939e48SSatish Balay  def __str2__(self):
217c939e48SSatish Balay    desc = []
22a0022257SSatish Balay    desc.append('xxx=========================================================================xxx')
23022a4980SJose Roman    if self.cmakeboot_success and not hasattr(self.compilers, 'CUDAC'):
24b3618d6dSSatish Balay      build_type = 'cmake build'
25b3618d6dSSatish Balay    else:
26b3618d6dSSatish Balay      build_type = 'legacy build'
27b3618d6dSSatish Balay    desc.append(' Configure stage complete. Now build PETSc libraries with (%s):' % build_type)
28b3618d6dSSatish Balay    desc.append('   make PETSC_DIR='+self.petscdir.dir+' PETSC_ARCH='+self.arch.arch+' all')
29a8a678f1SSatish Balay    desc.append(' or (experimental with python):')
30c77b0880SMatthew G Knepley    desc.append('   PETSC_DIR='+self.petscdir.dir+' PETSC_ARCH='+self.arch.arch+' ./config/builder.py')
31a0022257SSatish Balay    desc.append('xxx=========================================================================xxx')
327c939e48SSatish Balay    return '\n'.join(desc)+'\n'
33f8833479SBarry Smith
34f8833479SBarry Smith  def setupHelp(self, help):
35f8833479SBarry Smith    import nargs
36ce0b2093SBarry Smith    help.addArgument('PETSc',  '-prefix=<dir>',                  nargs.Arg(None, '', 'Specifiy location to install PETSc (eg. /usr/local)'))
37eed94e11SSatish Balay    help.addArgument('Windows','-with-windows-graphics=<bool>',   nargs.ArgBool(None, 1,'Enable check for Windows Graphics'))
38569865ddSSatish Balay    help.addArgument('PETSc', '-with-default-arch=<bool>',        nargs.ArgBool(None, 1, 'Allow using the last configured arch without setting PETSC_ARCH'))
3957cb31baSSatish Balay    help.addArgument('PETSc','-with-single-library=<bool>',       nargs.ArgBool(None, 1,'Put all PETSc code into the single -lpetsc library'))
40ce0b2093SBarry Smith    help.addArgument('PETSc', '-with-iphone=<bool>',              nargs.ArgBool(None, 0, 'Build an iPhone version of PETSc'))
41f8833479SBarry Smith    return
42f8833479SBarry Smith
43f8833479SBarry Smith  def setupDependencies(self, framework):
44f8833479SBarry Smith    config.base.Configure.setupDependencies(self, framework)
45f8833479SBarry Smith    self.setCompilers  = framework.require('config.setCompilers',      self)
46f8833479SBarry Smith    self.arch          = framework.require('PETSc.utilities.arch',     self.setCompilers)
47f8833479SBarry Smith    self.petscdir      = framework.require('PETSc.utilities.petscdir', self.setCompilers)
48f8833479SBarry Smith    self.languages     = framework.require('PETSc.utilities.languages',self.setCompilers)
49f8833479SBarry Smith    self.debugging     = framework.require('PETSc.utilities.debugging',self.setCompilers)
50a2b1cf82SBarry Smith    self.CHUD          = framework.require('PETSc.utilities.CHUD',     self)
51f8833479SBarry Smith    self.compilers     = framework.require('config.compilers',         self)
52f8833479SBarry Smith    self.types         = framework.require('config.types',             self)
53f8833479SBarry Smith    self.headers       = framework.require('config.headers',           self)
54f8833479SBarry Smith    self.functions     = framework.require('config.functions',         self)
55f8833479SBarry Smith    self.libraries     = framework.require('config.libraries',         self)
56f8833479SBarry Smith    if os.path.isdir(os.path.join('config', 'PETSc')):
57f8833479SBarry Smith      for d in ['utilities', 'packages']:
58f8833479SBarry Smith        for utility in os.listdir(os.path.join('config', 'PETSc', d)):
59f8833479SBarry Smith          (utilityName, ext) = os.path.splitext(utility)
60f8833479SBarry Smith          if not utilityName.startswith('.') and not utilityName.startswith('#') and ext == '.py' and not utilityName == '__init__':
61f8833479SBarry Smith            utilityObj                    = self.framework.require('PETSc.'+d+'.'+utilityName, self)
62f8833479SBarry Smith            utilityObj.headerPrefix       = self.headerPrefix
63f1ecedd2SMatthew Knepley            utilityObj.archProvider       = self.arch
64fdfda77fSMatthew Knepley            utilityObj.languageProvider   = self.languages
65fdfda77fSMatthew Knepley            utilityObj.installDirProvider = self.petscdir
66f8833479SBarry Smith            setattr(self, utilityName.lower(), utilityObj)
678cf378d1SBarry Smith
68d37554e4SMatthew G Knepley    for package in config.packages.all:
69d37554e4SMatthew G Knepley      if not package == 'PETSc':
70d37554e4SMatthew G Knepley        packageObj                    = framework.require('config.packages.'+package, self)
71d37554e4SMatthew G Knepley        packageObj.archProvider       = self.arch
72d37554e4SMatthew G Knepley        packageObj.languageProvider   = self.languages
73d37554e4SMatthew G Knepley        packageObj.installDirProvider = self.petscdir
74d37554e4SMatthew G Knepley        setattr(self, package.lower(), packageObj)
75d37554e4SMatthew G Knepley    # Force blaslapack to depend on scalarType so precision is set before BlasLapack is built
766d2d6fe1SBarry Smith    framework.require('PETSc.utilities.scalarTypes', self.f2cblaslapack)
776d2d6fe1SBarry Smith    self.f2cblaslapack.precisionProvider = self.scalartypes
788cf378d1SBarry Smith    framework.require('PETSc.utilities.scalarTypes', self.blaslapack)
79f8833479SBarry Smith    self.blaslapack.precisionProvider = self.scalartypes
80f8833479SBarry Smith
81f8833479SBarry Smith    self.compilers.headerPrefix  = self.headerPrefix
82f8833479SBarry Smith    self.types.headerPrefix      = self.headerPrefix
83f8833479SBarry Smith    self.headers.headerPrefix    = self.headerPrefix
84f8833479SBarry Smith    self.functions.headerPrefix  = self.headerPrefix
85f8833479SBarry Smith    self.libraries.headerPrefix  = self.headerPrefix
86f8833479SBarry Smith    self.blaslapack.headerPrefix = self.headerPrefix
87f8833479SBarry Smith    self.mpi.headerPrefix        = self.headerPrefix
88f8833479SBarry Smith    headersC = map(lambda name: name+'.h', ['dos', 'endian', 'fcntl', 'float', 'io', 'limits', 'malloc', 'pwd', 'search', 'strings',
89ba61063dSBarry Smith                                            'unistd', 'sys/sysinfo', 'machine/endian', 'sys/param', 'sys/procfs', 'sys/resource',
90b4bb646cSSatish Balay                                            'sys/systeminfo', 'sys/times', 'sys/utsname','string', 'stdlib','memory',
91f8833479SBarry Smith                                            'sys/socket','sys/wait','netinet/in','netdb','Direct','time','Ws2tcpip','sys/types',
928f54b378SBarry Smith                                            'WindowsX', 'cxxabi','float','ieeefp','stdint','fenv','sched'])
93f8833479SBarry Smith    functions = ['access', '_access', 'clock', 'drand48', 'getcwd', '_getcwd', 'getdomainname', 'gethostname', 'getpwuid',
94f8833479SBarry Smith                 'gettimeofday', 'getwd', 'memalign', 'memmove', 'mkstemp', 'popen', 'PXFGETARG', 'rand', 'getpagesize',
95a6d0e24fSJed Brown                 'readlink', 'realpath',  'sigaction', 'signal', 'sigset', 'nanosleep', 'usleep', 'sleep', '_sleep', 'socket',
96a6d0e24fSJed Brown                 'times', 'gethostbyname', 'uname','snprintf','_snprintf','_fullpath','lseek','_lseek','time','fork','stricmp',
97a6d0e24fSJed Brown                 'strcasecmp', 'bzero', 'dlopen', 'dlsym', 'dlclose', 'dlerror',
9864ffe704SBarry Smith                 '_intel_fast_memcpy','_intel_fast_memset']
99f8833479SBarry Smith    libraries1 = [(['socket', 'nsl'], 'socket'), (['fpe'], 'handle_sigfpes')]
100f8833479SBarry Smith    self.headers.headers.extend(headersC)
101f8833479SBarry Smith    self.functions.functions.extend(functions)
102f8833479SBarry Smith    self.libraries.libraries.extend(libraries1)
1037d421530SBarry Smith
104f8833479SBarry Smith    return
105f8833479SBarry Smith
106f8833479SBarry Smith  def Dump(self):
107f8833479SBarry Smith    ''' Actually put the values into the configuration files '''
108f8833479SBarry Smith    # eventually everything between -- should be gone
109f8833479SBarry Smith#-----------------------------------------------------------------------------------------------------
110f8833479SBarry Smith
111f8833479SBarry Smith    # Sometimes we need C compiler, even if built with C++
112f8833479SBarry Smith    self.setCompilers.pushLanguage('C')
113f8833479SBarry Smith    self.addMakeMacro('CC_FLAGS',self.setCompilers.getCompilerFlags())
114f8833479SBarry Smith    self.setCompilers.popLanguage()
115f8833479SBarry Smith
116f8833479SBarry Smith    # C preprocessor values
117a2b1cf82SBarry Smith    self.addMakeMacro('CPP_FLAGS',self.setCompilers.CPPFLAGS+self.CHUD.CPPFLAGS)
118f8833479SBarry Smith
119f8833479SBarry Smith    # compiler values
120f8833479SBarry Smith    self.setCompilers.pushLanguage(self.languages.clanguage)
121f8833479SBarry Smith    self.addMakeMacro('PCC',self.setCompilers.getCompiler())
122f8833479SBarry Smith    self.addMakeMacro('PCC_FLAGS',self.setCompilers.getCompilerFlags())
123f8833479SBarry Smith    self.setCompilers.popLanguage()
124f8833479SBarry Smith    # .o or .obj
125f8833479SBarry Smith    self.addMakeMacro('CC_SUFFIX','o')
126f8833479SBarry Smith
127f8833479SBarry Smith    # executable linker values
128f8833479SBarry Smith    self.setCompilers.pushLanguage(self.languages.clanguage)
129f8833479SBarry Smith    pcc_linker = self.setCompilers.getLinker()
130f8833479SBarry Smith    self.addMakeMacro('PCC_LINKER',pcc_linker)
131c84a332bSSatish Balay    self.addMakeMacro('PCC_LINKER_FLAGS',self.setCompilers.getLinkerFlags())
132f8833479SBarry Smith    self.setCompilers.popLanguage()
133f8833479SBarry Smith    # '' for Unix, .exe for Windows
134f8833479SBarry Smith    self.addMakeMacro('CC_LINKER_SUFFIX','')
135f8833479SBarry Smith
136f8833479SBarry Smith    if hasattr(self.compilers, 'FC'):
137f8833479SBarry Smith      self.setCompilers.pushLanguage('FC')
138f8833479SBarry Smith      # need FPPFLAGS in config/setCompilers
139f8833479SBarry Smith      self.addDefine('HAVE_FORTRAN','1')
140f8833479SBarry Smith      self.addMakeMacro('FPP_FLAGS',self.setCompilers.CPPFLAGS)
141f8833479SBarry Smith
142f8833479SBarry Smith      # compiler values
143f8833479SBarry Smith      self.addMakeMacro('FC_FLAGS',self.setCompilers.getCompilerFlags())
144f8833479SBarry Smith      self.setCompilers.popLanguage()
145f8833479SBarry Smith      # .o or .obj
146f8833479SBarry Smith      self.addMakeMacro('FC_SUFFIX','o')
147f8833479SBarry Smith
148f8833479SBarry Smith      # executable linker values
149f8833479SBarry Smith      self.setCompilers.pushLanguage('FC')
150f8833479SBarry Smith      # Cannot have NAG f90 as the linker - so use pcc_linker as fc_linker
151f8833479SBarry Smith      fc_linker = self.setCompilers.getLinker()
152f8833479SBarry Smith      if config.setCompilers.Configure.isNAG(fc_linker):
153f8833479SBarry Smith        self.addMakeMacro('FC_LINKER',pcc_linker)
154f8833479SBarry Smith      else:
155f8833479SBarry Smith        self.addMakeMacro('FC_LINKER',fc_linker)
1566d53d35eSSatish Balay      self.addMakeMacro('FC_LINKER_FLAGS',self.setCompilers.getLinkerFlags())
1573feacd00SBarry Smith      # apple requires this shared library linker flag on SOME versions of the os
1583feacd00SBarry Smith      if self.setCompilers.getLinkerFlags().find('-Wl,-commons,use_dylibs') > -1:
1593feacd00SBarry Smith        self.addMakeMacro('DARWIN_COMMONS_USE_DYLIBS',' -Wl,-commons,use_dylibs ')
160bb82cf9cSSatish Balay      self.setCompilers.popLanguage()
1615d631499SMatthew Knepley
1625d631499SMatthew Knepley      # F90 Modules
1635d631499SMatthew Knepley      if self.setCompilers.fortranModuleIncludeFlag:
1645d631499SMatthew Knepley        self.addMakeMacro('FC_MODULE_FLAG', self.setCompilers.fortranModuleIncludeFlag)
1656ddd6694SSatish Balay      else: # for non-f90 compilers like g77
1666ddd6694SSatish Balay        self.addMakeMacro('FC_MODULE_FLAG', '-I')
167a324c51cSMatthew G Knepley      if self.setCompilers.fortranModuleIncludeFlag:
168a324c51cSMatthew G Knepley        self.addMakeMacro('FC_MODULE_OUTPUT_FLAG', self.setCompilers.fortranModuleOutputFlag)
169f8833479SBarry Smith    else:
170f8833479SBarry Smith      self.addMakeMacro('FC','')
171f8833479SBarry Smith
17246a3958fSBarry Smith    if hasattr(self.compilers, 'CUDAC'):
1737ff2890cSSatish Balay      self.setCompilers.pushLanguage('CUDA')
174d93a25ecSSatish Balay      self.addMakeMacro('CUDAC_FLAGS',self.setCompilers.getCompilerFlags())
1757ff2890cSSatish Balay      self.setCompilers.popLanguage()
1767ff2890cSSatish Balay
177f8833479SBarry Smith    # shared library linker values
178f8833479SBarry Smith    self.setCompilers.pushLanguage(self.languages.clanguage)
179f8833479SBarry Smith    # need to fix BuildSystem to collect these separately
180f8833479SBarry Smith    self.addMakeMacro('SL_LINKER',self.setCompilers.getLinker())
18170db8aa6SSatish Balay    self.addMakeMacro('SL_LINKER_FLAGS','${PCC_LINKER_FLAGS}')
182f8833479SBarry Smith    self.setCompilers.popLanguage()
183f8833479SBarry Smith    # One of 'a', 'so', 'lib', 'dll', 'dylib' (perhaps others also?) depending on the library generator and architecture
184f8833479SBarry Smith    # Note: . is not included in this macro, consistent with AR_LIB_SUFFIX
185f8833479SBarry Smith    if self.setCompilers.sharedLibraryExt == self.setCompilers.AR_LIB_SUFFIX:
186f8833479SBarry Smith      self.addMakeMacro('SL_LINKER_SUFFIX', '')
18746bc77b6SBarry Smith      self.addDefine('SLSUFFIX','""')
188f8833479SBarry Smith    else:
189f8833479SBarry Smith      self.addMakeMacro('SL_LINKER_SUFFIX', self.setCompilers.sharedLibraryExt)
19046bc77b6SBarry Smith      self.addDefine('SLSUFFIX','"'+self.setCompilers.sharedLibraryExt+'"')
191bb82cf9cSSatish Balay
19223e93537SBarry Smith    self.addMakeMacro('SL_LINKER_LIBS','${PETSC_EXTERNAL_LIB_BASIC}')
193bb82cf9cSSatish Balay
194f8833479SBarry Smith#-----------------------------------------------------------------------------------------------------
195f8833479SBarry Smith
196f8833479SBarry Smith    # CONLY or CPP. We should change the PETSc makefiles to do this better
197f8833479SBarry Smith    if self.languages.clanguage == 'C': lang = 'CONLY'
198f8833479SBarry Smith    else: lang = 'CXXONLY'
199f8833479SBarry Smith    self.addMakeMacro('PETSC_LANGUAGE',lang)
200f8833479SBarry Smith
201f8833479SBarry Smith    # real or complex
202f8833479SBarry Smith    self.addMakeMacro('PETSC_SCALAR',self.scalartypes.scalartype)
203f8833479SBarry Smith    # double or float
204f8833479SBarry Smith    self.addMakeMacro('PETSC_PRECISION',self.scalartypes.precision)
205f8833479SBarry Smith
206f8833479SBarry Smith    if self.framework.argDB['with-batch']:
207f8833479SBarry Smith      self.addMakeMacro('PETSC_WITH_BATCH','1')
208f8833479SBarry Smith
209f8833479SBarry Smith    # Test for compiler-specific macros that need to be defined.
210b409243cSBarry Smith    if self.setCompilers.isCrayVector('CC'):
211b409243cSBarry Smith      self.addDefine('HAVE_CRAY_VECTOR','1')
212f8833479SBarry Smith
213f8833479SBarry Smith#-----------------------------------------------------------------------------------------------------
214df1a78b3SMatthew G Knepley    if self.functions.haveFunction('gethostbyname') and self.functions.haveFunction('socket') and self.headers.haveHeader('netinet/in.h'):
215f8833479SBarry Smith      self.addDefine('USE_SOCKET_VIEWER','1')
216f8833479SBarry Smith
217f8833479SBarry Smith#-----------------------------------------------------------------------------------------------------
218a6cc6bb1SBarry Smith    # print include and lib for makefiles
219f8833479SBarry Smith    self.framework.packages.reverse()
220a6cc6bb1SBarry Smith    includes = [os.path.join(self.petscdir.dir,'include'),os.path.join(self.petscdir.dir,self.arch.arch,'include')]
221996b3231SBarry Smith    libs = []
222f8833479SBarry Smith    for i in self.framework.packages:
223898a086dSBarry Smith      if i.useddirectly:
224898a086dSBarry Smith        self.addDefine('HAVE_'+i.PACKAGE, 1)  # ONLY list package if it is used directly by PETSc (and not only by another package)
225f8833479SBarry Smith      if not isinstance(i.lib, list):
226f8833479SBarry Smith        i.lib = [i.lib]
227ac9e4c42SSatish Balay      libs.extend(i.lib)
22834cdeb2aSSatish Balay      self.addMakeMacro(i.PACKAGE+'_LIB', self.libraries.toStringNoDupes(i.lib))
229f8833479SBarry Smith      if hasattr(i,'include'):
230f8833479SBarry Smith        if not isinstance(i.include,list):
231f8833479SBarry Smith          i.include = [i.include]
232ac9e4c42SSatish Balay        includes.extend(i.include)
23334cdeb2aSSatish Balay        self.addMakeMacro(i.PACKAGE+'_INCLUDE',self.headers.toStringNoDupes(i.include))
2342df986feSBarry Smith    if self.framework.argDB['with-single-library']:
235ea820d49SSatish Balay      self.addMakeMacro('PETSC_WITH_EXTERNAL_LIB',self.libraries.toStringNoDupes(['-L'+os.path.join(self.petscdir.dir,self.arch.arch,'lib'),' -lpetsc']+libs+self.libraries.math+self.compilers.flibs+self.compilers.cxxlibs+self.compilers.LIBS.split(' '))+self.CHUD.LIBS)
23623e93537SBarry Smith    self.addMakeMacro('PETSC_EXTERNAL_LIB_BASIC',self.libraries.toStringNoDupes(libs+self.libraries.math+self.compilers.flibs+self.compilers.cxxlibs+self.compilers.LIBS.split(' '))+self.CHUD.LIBS)
23723e93537SBarry Smith    self.PETSC_EXTERNAL_LIB_BASIC = self.libraries.toStringNoDupes(libs+self.libraries.math+self.compilers.flibs+self.compilers.cxxlibs+self.compilers.LIBS.split(' '))+self.CHUD.LIBS
238a6cc6bb1SBarry Smith    self.addMakeMacro('PETSC_CC_INCLUDES',self.headers.toStringNoDupes(includes))
239a6cc6bb1SBarry Smith    self.PETSC_CC_INCLUDES = self.headers.toStringNoDupes(includes)
240cbd5cc15SBarry Smith    if hasattr(self.compilers, 'FC'):
241208c3fd5SBarry Smith      if self.compilers.fortranIsF90:
24243a63bfbSSatish Balay        self.addMakeMacro('PETSC_FC_INCLUDES',self.headers.toStringNoDupes(includes,includes))
24330d43657SSatish Balay      else:
24430d43657SSatish Balay        self.addMakeMacro('PETSC_FC_INCLUDES',self.headers.toStringNoDupes(includes))
245f8833479SBarry Smith
2461b1e03dfSSatish Balay    self.addMakeMacro('DESTDIR',self.installdir)
247c6f99f23SBarry Smith    self.addDefine('LIB_DIR','"'+os.path.join(self.installdir,'lib')+'"')
248f8833479SBarry Smith
2490f3b21c2SBarry Smith    if self.framework.argDB['with-single-library']:
2500f3b21c2SBarry Smith      # overrides the values set in conf/variables
2510f3b21c2SBarry Smith      self.addMakeMacro('LIBNAME','${INSTALL_LIB_DIR}/libpetsc.${AR_LIB_SUFFIX}')
25257cb31baSSatish Balay      self.addMakeMacro('SHLIBS','libpetsc')
253bccf1c12SBarry Smith      self.addMakeMacro('PETSC_LIB_BASIC','-lpetsc')
254797063a9SSatish Balay      self.addMakeMacro('PETSC_KSP_LIB_BASIC','-lpetsc')
255797063a9SSatish Balay      self.addMakeMacro('PETSC_TS_LIB_BASIC','-lpetsc')
256bb84e0fdSBarry Smith      self.addDefine('USE_SINGLE_LIBRARY', '1')
2572df986feSBarry Smith      if self.sharedlibraries.useShared:
258ea820d49SSatish Balay        self.addMakeMacro('PETSC_SYS_LIB','${C_SH_LIB_PATH} ${PETSC_WITH_EXTERNAL_LIB}')
259ea820d49SSatish Balay        self.addMakeMacro('PETSC_VEC_LIB','${C_SH_LIB_PATH} ${PETSC_WITH_EXTERNAL_LIB}')
260ea820d49SSatish Balay        self.addMakeMacro('PETSC_MAT_LIB','${C_SH_LIB_PATH} ${PETSC_WITH_EXTERNAL_LIB}')
261ea820d49SSatish Balay        self.addMakeMacro('PETSC_DM_LIB','${C_SH_LIB_PATH} ${PETSC_WITH_EXTERNAL_LIB}')
262ea820d49SSatish Balay        self.addMakeMacro('PETSC_KSP_LIB','${C_SH_LIB_PATH} ${PETSC_WITH_EXTERNAL_LIB}')
263ea820d49SSatish Balay        self.addMakeMacro('PETSC_SNES_LIB','${C_SH_LIB_PATH} ${PETSC_WITH_EXTERNAL_LIB}')
264ea820d49SSatish Balay        self.addMakeMacro('PETSC_TS_LIB','${C_SH_LIB_PATH} ${PETSC_WITH_EXTERNAL_LIB}')
265fdb87e33SJed Brown        self.addMakeMacro('PETSC_CHARACTERISTIC_LIB','${C_SH_LIB_PATH} ${PETSC_WITH_EXTERNAL_LIB}')
266ea820d49SSatish Balay        self.addMakeMacro('PETSC_LIB','${C_SH_LIB_PATH} ${PETSC_WITH_EXTERNAL_LIB}')
267ea820d49SSatish Balay        self.addMakeMacro('PETSC_CONTRIB','${C_SH_LIB_PATH} ${PETSC_WITH_EXTERNAL_LIB}')
2682df986feSBarry Smith      else:
269ea820d49SSatish Balay        self.addMakeMacro('PETSC_SYS_LIB','${PETSC_WITH_EXTERNAL_LIB}')
270ea820d49SSatish Balay        self.addMakeMacro('PETSC_VEC_LIB','${PETSC_WITH_EXTERNAL_LIB}')
271ea820d49SSatish Balay        self.addMakeMacro('PETSC_MAT_LIB','${PETSC_WITH_EXTERNAL_LIB}')
272ea820d49SSatish Balay        self.addMakeMacro('PETSC_DM_LIB','${PETSC_WITH_EXTERNAL_LIB}')
273ea820d49SSatish Balay        self.addMakeMacro('PETSC_KSP_LIB','${PETSC_WITH_EXTERNAL_LIB}')
274ea820d49SSatish Balay        self.addMakeMacro('PETSC_SNES_LIB','${PETSC_WITH_EXTERNAL_LIB}')
275ea820d49SSatish Balay        self.addMakeMacro('PETSC_TS_LIB','${PETSC_WITH_EXTERNAL_LIB}')
276fdb87e33SJed Brown        self.addMakeMacro('PETSC_CHARACTERISTIC_LIB','${PETSC_WITH_EXTERNAL_LIB}')
277ea820d49SSatish Balay        self.addMakeMacro('PETSC_LIB','${PETSC_WITH_EXTERNAL_LIB}')
278ea820d49SSatish Balay        self.addMakeMacro('PETSC_CONTRIB','${PETSC_WITH_EXTERNAL_LIB}')
2790f3b21c2SBarry Smith
280f8833479SBarry Smith    if not os.path.exists(os.path.join(self.petscdir.dir,self.arch.arch,'lib')):
281f8833479SBarry Smith      os.makedirs(os.path.join(self.petscdir.dir,self.arch.arch,'lib'))
282f8833479SBarry Smith
283f8833479SBarry Smith    # add a makefile entry for configure options
284f8833479SBarry Smith    self.addMakeMacro('CONFIGURE_OPTIONS', self.framework.getOptionsString(['configModules', 'optionsModule']).replace('\"','\\"'))
285f8833479SBarry Smith    return
286f8833479SBarry Smith
287f8833479SBarry Smith  def dumpConfigInfo(self):
288f8833479SBarry Smith    import time
289f8833479SBarry Smith    fd = file(os.path.join(self.arch.arch,'include','petscconfiginfo.h'),'w')
290f8833479SBarry Smith    fd.write('static const char *petscconfigureruntime = "'+time.ctime(time.time())+'";\n')
291f8833479SBarry Smith    fd.write('static const char *petscconfigureoptions = "'+self.framework.getOptionsString(['configModules', 'optionsModule']).replace('\"','\\"')+'";\n')
292f8833479SBarry Smith    fd.close()
293f8833479SBarry Smith    return
294f8833479SBarry Smith
2952a4161d9SMatthew G Knepley  def dumpMachineInfo(self):
2962a4161d9SMatthew G Knepley    import platform
2972a4161d9SMatthew G Knepley    import time
29840373944SSatish Balay    import script
2992a4161d9SMatthew G Knepley    fd = file(os.path.join(self.arch.arch,'include','petscmachineinfo.h'),'w')
3002a4161d9SMatthew G Knepley    fd.write('static const char *petscmachineinfo = \"\\n\"\n')
3012a4161d9SMatthew G Knepley    fd.write('\"-----------------------------------------\\n\"\n')
3022a4161d9SMatthew G Knepley    fd.write('\"Libraries compiled on %s on %s \\n\"\n' % (time.ctime(time.time()), platform.node()))
30360acdfe7SSatish Balay    fd.write('\"Machine characteristics: %s\\n\"\n' % (platform.platform()))
30460acdfe7SSatish Balay    fd.write('\"Using PETSc directory: %s\\n\"\n' % (self.petscdir.dir))
30560acdfe7SSatish Balay    fd.write('\"Using PETSc arch: %s\\n\"\n' % (self.arch.arch))
306cdec380aSBarry Smith    fd.write('\"-----------------------------------------\\n\";\n')
3072a4161d9SMatthew G Knepley    fd.write('static const char *petsccompilerinfo = \"\\n\"\n')
3082a4161d9SMatthew G Knepley    self.setCompilers.pushLanguage(self.languages.clanguage)
30960acdfe7SSatish Balay    fd.write('\"Using C compiler: %s %s ${COPTFLAGS} ${CFLAGS}\\n\"\n' % (self.setCompilers.getCompiler(), self.setCompilers.getCompilerFlags()))
3102a4161d9SMatthew G Knepley    self.setCompilers.popLanguage()
3118782282cSMatthew G Knepley    if hasattr(self.compilers, 'FC'):
3122a4161d9SMatthew G Knepley      self.setCompilers.pushLanguage('FC')
31360acdfe7SSatish Balay      fd.write('\"Using Fortran compiler: %s %s ${FOPTFLAGS} ${FFLAGS} %s\\n\"\n' % (self.setCompilers.getCompiler(), self.setCompilers.getCompilerFlags(), self.setCompilers.CPPFLAGS))
3142a4161d9SMatthew G Knepley      self.setCompilers.popLanguage()
315cdec380aSBarry Smith    fd.write('\"-----------------------------------------\\n\";\n')
3162a4161d9SMatthew G Knepley    fd.write('static const char *petsccompilerflagsinfo = \"\\n\"\n')
317ac448642SSatish Balay    fd.write('\"Using include paths: %s %s %s\\n\"\n' % ('-I'+os.path.join(self.petscdir.dir, self.arch.arch, 'include'), '-I'+os.path.join(self.petscdir.dir, 'include'), self.PETSC_CC_INCLUDES.replace('\\ ','\\\\ ')))
318cdec380aSBarry Smith    fd.write('\"-----------------------------------------\\n\";\n')
3192a4161d9SMatthew G Knepley    fd.write('static const char *petsclinkerinfo = \"\\n\"\n')
3202a4161d9SMatthew G Knepley    self.setCompilers.pushLanguage(self.languages.clanguage)
32160acdfe7SSatish Balay    fd.write('\"Using C linker: %s\\n\"\n' % (self.setCompilers.getLinker()))
3222a4161d9SMatthew G Knepley    self.setCompilers.popLanguage()
3238782282cSMatthew G Knepley    if hasattr(self.compilers, 'FC'):
3242a4161d9SMatthew G Knepley      self.setCompilers.pushLanguage('FC')
32560acdfe7SSatish Balay      fd.write('\"Using Fortran linker: %s\\n\"\n' % (self.setCompilers.getLinker()))
3262a4161d9SMatthew G Knepley      self.setCompilers.popLanguage()
327ad782ac6SSatish Balay    if self.framework.argDB['with-single-library']:
328ad782ac6SSatish Balay      petsclib = '-lpetsc'
329ad782ac6SSatish Balay    else:
330ad782ac6SSatish Balay      petsclib = '-lpetscts -lpetscsnes -lpetscksp -lpetscdm -lpetscmat -lpetscvec -lpetscsys'
331ac448642SSatish Balay    fd.write('\"Using libraries: %s%s -L%s %s %s\\n\"\n' % (self.setCompilers.CSharedLinkerFlag, os.path.join(self.petscdir.dir, self.arch.arch, 'lib'), os.path.join(self.petscdir.dir, self.arch.arch, 'lib'), petsclib, self.PETSC_EXTERNAL_LIB_BASIC.replace('\\ ','\\\\ ')))
332cdec380aSBarry Smith    fd.write('\"-----------------------------------------\\n\";\n')
3332a4161d9SMatthew G Knepley    fd.close()
3342a4161d9SMatthew G Knepley    return
335b2843cf1SBarry Smith
336511a6afcSJed Brown  def dumpCMakeConfig(self):
337511a6afcSJed Brown    '''
33834ed7027SBarry Smith    Writes configuration-specific values to ${PETSC_ARCH}/conf/PETScConfig.cmake.
339511a6afcSJed Brown    This file is private to PETSc and should not be included by third parties
340511a6afcSJed Brown    (a suitable file can be produced later by CMake, but this is not it).
341511a6afcSJed Brown    '''
342511a6afcSJed Brown    def cmakeset(fd,key,val=True):
343511a6afcSJed Brown      if val == True: val = 'YES'
344511a6afcSJed Brown      if val == False: val = 'NO'
345511a6afcSJed Brown      fd.write('set (' + key + ' ' + val + ')\n')
346511a6afcSJed Brown    def ensurelist(a):
347826d9344SJed Brown      if isinstance(a,list):
348826d9344SJed Brown        return a
349826d9344SJed Brown      else:
350826d9344SJed Brown        return [a]
351511a6afcSJed Brown    def libpath(lib):
352511a6afcSJed Brown      'Returns a search path if that is what this item provides, else "" which will be cleaned out later'
3531b1c0b30SJed Brown      if not isinstance(lib,str): return ''
354511a6afcSJed Brown      if lib.startswith('-L'): return lib[2:]
355511a6afcSJed Brown      if lib.startswith('-R'): return lib[2:]
356511a6afcSJed Brown      if lib.startswith('-Wl,-rpath,'):
357511a6afcSJed Brown        # This case occurs when an external package needs a specific system library that is normally provided by the compiler.
358511a6afcSJed Brown        # In other words, the -L path is builtin to the wrapper or compiler, here we provide it so that CMake can locate the
359511a6afcSJed Brown        # corresponding library.
360511a6afcSJed Brown        return lib[len('-Wl,-rpath,'):]
361511a6afcSJed Brown      if lib.startswith('-'): return ''
362511a6afcSJed Brown      return os.path.dirname(lib)
363511a6afcSJed Brown    def cleanlib(lib):
364511a6afcSJed Brown      'Returns a library name if that is what this item provides, else "" which will be cleaned out later'
36542e8629dSMatthew G Knepley      if not isinstance(lib,str): return ''
366511a6afcSJed Brown      if lib.startswith('-l'):  return lib[2:]
367511a6afcSJed Brown      if lib.startswith('-Wl') or lib.startswith('-L'): return ''
368511a6afcSJed Brown      lib = os.path.splitext(os.path.basename(lib))[0]
369511a6afcSJed Brown      if lib.startswith('lib'): return lib[3:]
370511a6afcSJed Brown      return lib
371511a6afcSJed Brown    def nub(lst):
37206e8c1ddSJed Brown      'Return a list containing the first occurrence of each unique element'
373511a6afcSJed Brown      unique = []
374511a6afcSJed Brown      for elem in lst:
375511a6afcSJed Brown        if elem not in unique and elem != '':
376511a6afcSJed Brown          unique.append(elem)
377511a6afcSJed Brown      return unique
37850937898SJed Brown    try: reversed # reversed was added in Python-2.4
37950937898SJed Brown    except NameError:
38050937898SJed Brown      def reversed(lst): return lst[::-1]
38106e8c1ddSJed Brown    def nublast(lst):
38206e8c1ddSJed Brown      'Return a list containing the last occurrence of each unique entry in a list'
38350937898SJed Brown      return reversed(nub(reversed(lst)))
384511a6afcSJed Brown    def cmakeexpand(varname):
385511a6afcSJed Brown      return r'"${' + varname + r'}"'
386582751aaSJed Brown    def uniqextend(lst,new):
387511a6afcSJed Brown      for x in ensurelist(new):
388582751aaSJed Brown        if x not in lst:
389582751aaSJed Brown          lst.append(x)
390511a6afcSJed Brown    def notstandardinclude(path):
391511a6afcSJed Brown      return path not in '/usr/include /usr/local/include'.split()
392511a6afcSJed Brown    def writeMacroDefinitions(fd):
393511a6afcSJed Brown      if self.mpi.usingMPIUni:
394511a6afcSJed Brown        cmakeset(fd,'PETSC_HAVE_MPIUNI')
395511a6afcSJed Brown      for pkg in self.framework.packages:
396511a6afcSJed Brown        if pkg.useddirectly:
397511a6afcSJed Brown          cmakeset(fd,'PETSC_HAVE_' + pkg.PACKAGE)
398511a6afcSJed Brown      for name,val in self.functions.defines.items():
399511a6afcSJed Brown        cmakeset(fd,'PETSC_'+name,val)
400511a6afcSJed Brown      for dct in [self.defines, self.libraryoptions.defines]:
401511a6afcSJed Brown        for k,v in dct.items():
402511a6afcSJed Brown          if k.startswith('USE_'):
403511a6afcSJed Brown            cmakeset(fd,'PETSC_' + k, v)
404511a6afcSJed Brown      cmakeset(fd,'PETSC_USE_COMPLEX', self.scalartypes.scalartype == 'complex')
405ce63c4c1SBarry Smith      cmakeset(fd,'PETSC_USE_REAL_' + self.scalartypes.precision.upper())
406511a6afcSJed Brown      cmakeset(fd,'PETSC_CLANGUAGE_'+self.languages.clanguage)
407511a6afcSJed Brown      if hasattr(self.compilers, 'FC'):
408511a6afcSJed Brown        cmakeset(fd,'PETSC_HAVE_FORTRAN')
409511a6afcSJed Brown        if self.compilers.fortranIsF90:
410511a6afcSJed Brown          cmakeset(fd,'PETSC_USING_F90')
411511a6afcSJed Brown      if self.sharedlibraries.useShared:
412511a6afcSJed Brown        cmakeset(fd,'BUILD_SHARED_LIBS')
413511a6afcSJed Brown    def writeBuildFlags(fd):
41406e8c1ddSJed Brown      def extendby(lib):
41506e8c1ddSJed Brown        libs = ensurelist(lib)
41606e8c1ddSJed Brown        lib_paths.extend(map(libpath,libs))
41706e8c1ddSJed Brown        lib_libs.extend(map(cleanlib,libs))
41806e8c1ddSJed Brown        uniqextend(includes,pkg.include)
419511a6afcSJed Brown      lib_paths = []
420511a6afcSJed Brown      lib_libs  = []
421511a6afcSJed Brown      includes  = []
422511a6afcSJed Brown      libvars   = []
423511a6afcSJed Brown      for pkg in self.framework.packages:
42406e8c1ddSJed Brown        extendby(pkg.lib)
42506e8c1ddSJed Brown      extendby(self.libraries.math)
42606e8c1ddSJed Brown      extendby(self.libraries.rt)
42706e8c1ddSJed Brown      extendby(self.compilers.flibs)
42806e8c1ddSJed Brown      extendby(self.compilers.cxxlibs)
42906e8c1ddSJed Brown      extendby(self.compilers.LIBS.split())
43006e8c1ddSJed Brown      for libname in nublast(lib_libs):
431511a6afcSJed Brown        libvar = 'PETSC_' + libname.upper() + '_LIB'
4324c0032a9SSatish Balay        addpath = ''
43306e8c1ddSJed Brown        for lpath in nublast(lib_paths):
4344c0032a9SSatish Balay          addpath += '"' + str(lpath) + '" '
4354c0032a9SSatish Balay        fd.write('find_library (' + libvar + ' ' + libname + ' HINTS ' + addpath + ')\n')
436511a6afcSJed Brown        libvars.append(libvar)
437511a6afcSJed Brown      fd.write('mark_as_advanced (' + ' '.join(libvars) + ')\n')
438511a6afcSJed Brown      fd.write('set (PETSC_PACKAGE_LIBS ' + ' '.join(map(cmakeexpand,libvars)) + ')\n')
439511a6afcSJed Brown      fd.write('set (PETSC_PACKAGE_INCLUDES ' + ' '.join(map(lambda i: '"'+i+'"',filter(notstandardinclude,includes))) + ')\n')
440511a6afcSJed Brown    fd = open(os.path.join(self.arch.arch,'conf','PETScConfig.cmake'), 'w')
441511a6afcSJed Brown    writeMacroDefinitions(fd)
442511a6afcSJed Brown    writeBuildFlags(fd)
443511a6afcSJed Brown    fd.close()
444511a6afcSJed Brown    return
445511a6afcSJed Brown
4468b0282a9SJed Brown  def dumpCMakeLists(self):
4478b0282a9SJed Brown    import sys
4488b0282a9SJed Brown    if sys.version_info >= (2,5):
4498b0282a9SJed Brown      import cmakegen
4508b0282a9SJed Brown      try:
451a98e69d2SJed Brown        cmakegen.main(self.petscdir.dir, log=self.framework.log)
4528b0282a9SJed Brown      except (OSError), e:
4538b0282a9SJed Brown        self.framework.logPrint('Generating CMakeLists.txt failed:\n' + str(e))
454aac20692SSatish Balay    else:
455aac20692SSatish Balay      self.framework.logPrint('Skipping cmakegen due to old python version: ' +str(sys.version_info) )
4568b0282a9SJed Brown
4578b0282a9SJed Brown  def cmakeBoot(self):
4588b0282a9SJed Brown    import sys
459ae937f1dSJed Brown    self.cmakeboot_success = False
4600b7111d2SJed Brown    if sys.version_info >= (2,5) and hasattr(self.cmake,'cmake'):
461356464bcSMatthew G Knepley      try:
4628b0282a9SJed Brown        import cmakeboot
463ae937f1dSJed Brown        self.cmakeboot_success = cmakeboot.main(petscdir=self.petscdir.dir,petscarch=self.arch.arch,argDB=self.argDB,framework=self.framework,log=self.framework.log)
4648b0282a9SJed Brown      except (OSError), e:
4658b0282a9SJed Brown        self.framework.logPrint('Booting CMake in PETSC_ARCH failed:\n' + str(e))
466356464bcSMatthew G Knepley      except (ImportError, KeyError), e:
467356464bcSMatthew G Knepley        self.framework.logPrint('Importing cmakeboot failed:\n' + str(e))
468f7b66a64SJed Brown      if self.cmakeboot_success and not hasattr(self.compilers, 'CUDAC'): # Our CMake build does not support CUDA at this time
469f7b66a64SJed Brown        self.addMakeMacro('PETSC_BUILD_USING_CMAKE',1)
470aac20692SSatish Balay    else:
471aac20692SSatish Balay      self.framework.logPrint('Skipping cmakeboot due to old python version: ' +str(sys.version_info) )
472356464bcSMatthew G Knepley    return
4738b0282a9SJed Brown
474b2843cf1SBarry Smith  def configurePrefetch(self):
475b2843cf1SBarry Smith    '''Sees if there are any prefetch functions supported'''
4763649974fSBarry Smith    if config.setCompilers.Configure.isSolaris() or self.framework.argDB['with-iphone'] or self.framework.argDB['with-cuda']:
47793f78423SSatish Balay      self.addDefine('Prefetch(a,b,c)', ' ')
47893f78423SSatish Balay      return
479ec284106SBarry Smith    self.pushLanguage(self.languages.clanguage)
48010699583SJed Brown    if self.checkLink('#include <xmmintrin.h>', 'void *v = 0;_mm_prefetch((const char*)v,_MM_HINT_NTA);\n'):
48150d8bf02SJed Brown      # The Intel Intrinsics manual [1] specifies the prototype
48250d8bf02SJed Brown      #
48350d8bf02SJed Brown      #   void _mm_prefetch(char const *a, int sel);
48450d8bf02SJed Brown      #
48550d8bf02SJed Brown      # but other vendors seem to insist on using subtly different
48650d8bf02SJed Brown      # prototypes, including void* for the pointer, and an enum for
48750d8bf02SJed Brown      # sel.  These are both reasonable changes, but negatively impact
48850d8bf02SJed Brown      # portability.
48950d8bf02SJed Brown      #
49050d8bf02SJed Brown      # [1] http://software.intel.com/file/6373
49150d8bf02SJed Brown      self.addDefine('HAVE_XMMINTRIN_H', 1)
49250d8bf02SJed Brown      self.addDefine('Prefetch(a,b,c)', '_mm_prefetch((const char*)(a),(c))')
49350d8bf02SJed Brown      self.addDefine('PREFETCH_HINT_NTA', '_MM_HINT_NTA')
49450d8bf02SJed Brown      self.addDefine('PREFETCH_HINT_T0',  '_MM_HINT_T0')
49550d8bf02SJed Brown      self.addDefine('PREFETCH_HINT_T1',  '_MM_HINT_T1')
49650d8bf02SJed Brown      self.addDefine('PREFETCH_HINT_T2',  '_MM_HINT_T2')
49750d8bf02SJed Brown    elif self.checkLink('#include <xmmintrin.h>', 'void *v = 0;_mm_prefetch(v,_MM_HINT_NTA);\n'):
49850d8bf02SJed Brown      self.addDefine('HAVE_XMMINTRIN_H', 1)
49950d8bf02SJed Brown      self.addDefine('Prefetch(a,b,c)', '_mm_prefetch((const void*)(a),(c))')
50050d8bf02SJed Brown      self.addDefine('PREFETCH_HINT_NTA', '_MM_HINT_NTA')
50150d8bf02SJed Brown      self.addDefine('PREFETCH_HINT_T0',  '_MM_HINT_T0')
50250d8bf02SJed Brown      self.addDefine('PREFETCH_HINT_T1',  '_MM_HINT_T1')
50350d8bf02SJed Brown      self.addDefine('PREFETCH_HINT_T2',  '_MM_HINT_T2')
50410699583SJed Brown    elif self.checkLink('', 'void *v = 0;__builtin_prefetch(v,0,0);\n'):
50510699583SJed Brown      # From GCC docs: void __builtin_prefetch(const void *addr,int rw,int locality)
50610699583SJed Brown      #
50710699583SJed Brown      #   The value of rw is a compile-time constant one or zero; one
50810699583SJed Brown      #   means that the prefetch is preparing for a write to the memory
50910699583SJed Brown      #   address and zero, the default, means that the prefetch is
51010699583SJed Brown      #   preparing for a read. The value locality must be a compile-time
51110699583SJed Brown      #   constant integer between zero and three. A value of zero means
51210699583SJed Brown      #   that the data has no temporal locality, so it need not be left
51310699583SJed Brown      #   in the cache after the access. A value of three means that the
51410699583SJed Brown      #   data has a high degree of temporal locality and should be left
51510699583SJed Brown      #   in all levels of cache possible. Values of one and two mean,
51610699583SJed Brown      #   respectively, a low or moderate degree of temporal locality.
51710699583SJed Brown      #
51810699583SJed Brown      # Here we adopt Intel's x86/x86-64 naming scheme for the locality
51910699583SJed Brown      # hints.  Using macros for these values in necessary since some
52010699583SJed Brown      # compilers require an enum.
52110699583SJed Brown      self.addDefine('Prefetch(a,b,c)', '__builtin_prefetch((a),(b),(c))')
52210699583SJed Brown      self.addDefine('PREFETCH_HINT_NTA', '0')
52310699583SJed Brown      self.addDefine('PREFETCH_HINT_T0',  '3')
52410699583SJed Brown      self.addDefine('PREFETCH_HINT_T1',  '2')
52510699583SJed Brown      self.addDefine('PREFETCH_HINT_T2',  '1')
526b2843cf1SBarry Smith    else:
527b2843cf1SBarry Smith      self.addDefine('Prefetch(a,b,c)', ' ')
5287d490b44SBarry Smith    self.popLanguage()
529b2843cf1SBarry Smith
53073fca5a0SBarry Smith  def configureFeatureTestMacros(self):
53173fca5a0SBarry Smith    '''Checks if certain feature test macros are support'''
53209bc878fSSatish Balay    if self.checkCompile('#define _POSIX_C_SOURCE 200112L\n#include <stdlib.h>',''):
53373fca5a0SBarry Smith       self.addDefine('_POSIX_C_SOURCE_200112L', '1')
53409bc878fSSatish Balay    if self.checkCompile('#define _BSD_SOURCE\n#include<stdlib.h>',''):
53573fca5a0SBarry Smith       self.addDefine('_BSD_SOURCE', '1')
53673fca5a0SBarry Smith
53709bc878fSSatish Balay  def configureAtoll(self):
53809bc878fSSatish Balay    '''Checks if atoll exists'''
53909bc878fSSatish Balay    if self.checkCompile('#define _POSIX_C_SOURCE 200112L\n#include <stdlib.h>','long v = atoll("25")') or self.checkCompile ('#include <stdlib.h>','long v = atoll("25")'):
54009bc878fSSatish Balay       self.addDefine('HAVE_ATOLL', '1')
54109bc878fSSatish Balay
5422400fdedSBarry Smith  def configureUnused(self):
5432400fdedSBarry Smith    '''Sees if __attribute((unused)) is supported'''
5442400fdedSBarry Smith    if self.framework.argDB['with-iphone'] or self.framework.argDB['with-cuda']:
5452400fdedSBarry Smith      self.addDefine('UNUSED', ' ')
5462400fdedSBarry Smith      return
5472400fdedSBarry Smith    self.pushLanguage(self.languages.clanguage)
548*7bf3d34bSSatish Balay    if self.checkLink('__attribute((unused)) static int myfunc(void){ return 1;}', 'int i = myfunc();\ntypedef void* atype;\n__attribute((unused))  atype a;\n'):
5492400fdedSBarry Smith      self.addDefine('UNUSED', '__attribute((unused))')
5502400fdedSBarry Smith    else:
5512400fdedSBarry Smith      self.addDefine('UNUSED', ' ')
5522400fdedSBarry Smith    self.popLanguage()
5532400fdedSBarry Smith
5549800092aSJed Brown  def configureExpect(self):
5559800092aSJed Brown    '''Sees if the __builtin_expect directive is supported'''
5569800092aSJed Brown    self.pushLanguage(self.languages.clanguage)
5579800092aSJed Brown    if self.checkLink('', 'if (__builtin_expect(0,1)) return 1;'):
5589800092aSJed Brown      self.addDefine('HAVE_BUILTIN_EXPECT', 1)
5599800092aSJed Brown    self.popLanguage()
5609800092aSJed Brown
56153c77d0aSJed Brown  def configureFunctionName(self):
5621ec50b02SJed Brown    '''Sees if the compiler supports __func__ or a variant.  Falls back
5631ec50b02SJed Brown    on __FUNCT__ which PETSc source defines, but most users do not, thus
5641ec50b02SJed Brown    stack traces through user code are better when the compiler's
5651ec50b02SJed Brown    variant is used.'''
5661ec50b02SJed Brown    def getFunctionName(lang):
5671ec50b02SJed Brown      name = '__FUNCT__'
5681ec50b02SJed Brown      self.pushLanguage(lang)
56953c77d0aSJed Brown      if self.checkLink('', "if (__func__[0] != 'm') return 1;"):
5701ec50b02SJed Brown        name = '__func__'
57153c77d0aSJed Brown      elif self.checkLink('', "if (__FUNCTION__[0] != 'm') return 1;"):
5721ec50b02SJed Brown        name = '__FUNCTION__'
5731ec50b02SJed Brown      self.popLanguage()
5741ec50b02SJed Brown      return name
5751ec50b02SJed Brown    langs = []
576628773c9SSatish Balay
577628773c9SSatish Balay    self.addDefine('FUNCTION_NAME_C', getFunctionName('C'))
578628773c9SSatish Balay    if hasattr(self.compilers, 'CXX'):
579628773c9SSatish Balay      self.addDefine('FUNCTION_NAME_CXX', getFunctionName('Cxx'))
58012607bf0SSatish Balay    else:
58112607bf0SSatish Balay      self.addDefine('FUNCTION_NAME_CXX', '__FUNCT__')
58253c77d0aSJed Brown
583753ebd1dSJed Brown  def configureIntptrt(self):
584753ebd1dSJed Brown    '''Determine what to use for uintptr_t'''
585753ebd1dSJed Brown    def staticAssertSizeMatchesVoidStar(inc,typename):
586753ebd1dSJed Brown      # The declaration is an error if either array size is negative.
587753ebd1dSJed 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
588d26187a0SJed Brown      return self.checkCompile(inc, ('#define STATIC_ASSERT(cond) char negative_length_if_false[2*(!!(cond))-1]\n'
589979939cdSSatish Balay                                     + 'STATIC_ASSERT(sizeof(void*) == sizeof(%s));'%typename))
590753ebd1dSJed Brown    self.pushLanguage(self.languages.clanguage)
591753ebd1dSJed Brown    if self.checkCompile('#include <stdint.h>', 'int x; uintptr_t i = (uintptr_t)&x;'):
592753ebd1dSJed Brown      self.addDefine('UINTPTR_T', 'uintptr_t')
593753ebd1dSJed Brown    elif staticAssertSizeMatchesVoidStar('','unsigned long long'):
594753ebd1dSJed Brown      self.addDefine('UINTPTR_T', 'unsigned long long')
595753ebd1dSJed Brown    elif staticAssertSizeMatchesVoidStar('#include <stdlib.h>','size_t') or staticAssertSizeMatchesVoidStar('#include <string.h>', 'size_t'):
596753ebd1dSJed Brown      self.addDefine('UINTPTR_T', 'size_t')
597c82284b1SJed Brown    elif staticAssertSizeMatchesVoidStar('','unsigned long'):
598c82284b1SJed Brown      self.addDefine('UINTPTR_T', 'unsigned long')
5992d1b7972SSatish Balay    elif staticAssertSizeMatchesVoidStar('','unsigned'):
600753ebd1dSJed Brown      self.addDefine('UINTPTR_T', 'unsigned')
601d26187a0SJed Brown    else:
602d26187a0SJed Brown      raise RuntimeError('Could not find any unsigned integer type matching void*')
603753ebd1dSJed Brown    self.popLanguage()
604753ebd1dSJed Brown
605f8833479SBarry Smith  def configureInline(self):
606f8833479SBarry Smith    '''Get a generic inline keyword, depending on the language'''
607f8833479SBarry Smith    if self.languages.clanguage == 'C':
608f8833479SBarry Smith      self.addDefine('STATIC_INLINE', self.compilers.cStaticInlineKeyword)
609f8833479SBarry Smith      self.addDefine('RESTRICT', self.compilers.cRestrict)
610f8833479SBarry Smith    elif self.languages.clanguage == 'Cxx':
611f8833479SBarry Smith      self.addDefine('STATIC_INLINE', self.compilers.cxxStaticInlineKeyword)
612f8833479SBarry Smith      self.addDefine('RESTRICT', self.compilers.cxxRestrict)
613bfef2c86SBarry Smith
614bfef2c86SBarry Smith    if self.checkCompile('#include <dlfcn.h>\n void *ptr =  RTLD_DEFAULT;'):
615bfef2c86SBarry Smith      self.addDefine('RTLD_DEFAULT','1')
616f8833479SBarry Smith    return
617f8833479SBarry Smith
618f8833479SBarry Smith  def configureSolaris(self):
619f8833479SBarry Smith    '''Solaris specific stuff'''
620f8833479SBarry Smith    if os.path.isdir(os.path.join('/usr','ucblib')):
621f8833479SBarry Smith      try:
622f8833479SBarry Smith        flag = getattr(self.setCompilers, self.language[-1]+'SharedLinkerFlag')
623f8833479SBarry Smith      except AttributeError:
624f8833479SBarry Smith        flag = None
625f8833479SBarry Smith      if flag is None:
626f8833479SBarry Smith        self.compilers.LIBS += ' -L/usr/ucblib'
627f8833479SBarry Smith      else:
628f8833479SBarry Smith        self.compilers.LIBS += ' '+flag+'/usr/ucblib'
629f8833479SBarry Smith    return
630f8833479SBarry Smith
631f8833479SBarry Smith  def configureLinux(self):
632f8833479SBarry Smith    '''Linux specific stuff'''
6339f15855cSMatthew G Knepley    # TODO: Test for this by mallocing an odd number of floats and checking the address
634f8833479SBarry Smith    self.addDefine('HAVE_DOUBLE_ALIGN_MALLOC', 1)
635f8833479SBarry Smith    return
636f8833479SBarry Smith
637f8833479SBarry Smith  def configureWin32(self):
638f8833479SBarry Smith    '''Win32 non-cygwin specific stuff'''
639f8833479SBarry Smith    kernel32=0
640f8833479SBarry Smith    if self.libraries.add('Kernel32.lib','GetComputerName',prototype='#include <Windows.h>', call='GetComputerName(NULL,NULL);'):
641f8833479SBarry Smith      self.addDefine('HAVE_WINDOWS_H',1)
642f8833479SBarry Smith      self.addDefine('HAVE_GETCOMPUTERNAME',1)
643f8833479SBarry Smith      kernel32=1
644f8833479SBarry Smith    elif self.libraries.add('kernel32','GetComputerName',prototype='#include <Windows.h>', call='GetComputerName(NULL,NULL);'):
645f8833479SBarry Smith      self.addDefine('HAVE_WINDOWS_H',1)
646f8833479SBarry Smith      self.addDefine('HAVE_GETCOMPUTERNAME',1)
647f8833479SBarry Smith      kernel32=1
648f8833479SBarry Smith    if kernel32:
649eed94e11SSatish Balay      if self.framework.argDB['with-windows-graphics']:
650eed94e11SSatish Balay        self.addDefine('USE_WINDOWS_GRAPHICS',1)
651f8833479SBarry Smith      if self.checkLink('#include <Windows.h>','LoadLibrary(0)'):
652f8833479SBarry Smith        self.addDefine('HAVE_LOADLIBRARY',1)
653b50f6d9eSLisandro Dalcin      if self.checkLink('#include <Windows.h>','GetProcAddress(0,0)'):
654b50f6d9eSLisandro Dalcin        self.addDefine('HAVE_GETPROCADDRESS',1)
655b50f6d9eSLisandro Dalcin      if self.checkLink('#include <Windows.h>','FreeLibrary(0)'):
656b50f6d9eSLisandro Dalcin        self.addDefine('HAVE_FREELIBRARY',1)
657a21658a3SLisandro Dalcin      if self.checkLink('#include <Windows.h>','GetLastError()'):
658a21658a3SLisandro Dalcin        self.addDefine('HAVE_GETLASTERROR',1)
659a21658a3SLisandro Dalcin      if self.checkLink('#include <Windows.h>','SetLastError(0)'):
660a21658a3SLisandro Dalcin        self.addDefine('HAVE_SETLASTERROR',1)
661f8833479SBarry Smith      if self.checkLink('#include <Windows.h>\n','QueryPerformanceCounter(0);\n'):
662f8833479SBarry Smith        self.addDefine('USE_NT_TIME',1)
663f8833479SBarry Smith    if self.libraries.add('Advapi32.lib','GetUserName',prototype='#include <Windows.h>', call='GetUserName(NULL,NULL);'):
664f8833479SBarry Smith      self.addDefine('HAVE_GET_USER_NAME',1)
665f8833479SBarry Smith    elif self.libraries.add('advapi32','GetUserName',prototype='#include <Windows.h>', call='GetUserName(NULL,NULL);'):
666f8833479SBarry Smith      self.addDefine('HAVE_GET_USER_NAME',1)
667f8833479SBarry Smith
668f8833479SBarry Smith    if not self.libraries.add('User32.lib','GetDC',prototype='#include <Windows.h>',call='GetDC(0);'):
669f8833479SBarry Smith      self.libraries.add('user32','GetDC',prototype='#include <Windows.h>',call='GetDC(0);')
670f8833479SBarry Smith    if not self.libraries.add('Gdi32.lib','CreateCompatibleDC',prototype='#include <Windows.h>',call='CreateCompatibleDC(0);'):
671f8833479SBarry Smith      self.libraries.add('gdi32','CreateCompatibleDC',prototype='#include <Windows.h>',call='CreateCompatibleDC(0);')
672f8833479SBarry Smith
673f8833479SBarry Smith    self.types.check('int32_t', 'int')
674f8833479SBarry Smith    if not self.checkCompile('#include <sys/types.h>\n','uid_t u;\n'):
675f8833479SBarry Smith      self.addTypedef('int', 'uid_t')
676f8833479SBarry Smith      self.addTypedef('int', 'gid_t')
677f8833479SBarry Smith    if not self.checkLink('#if defined(PETSC_HAVE_UNISTD_H)\n#include <unistd.h>\n#endif\n','int a=R_OK;\n'):
678f8833479SBarry Smith      self.framework.addDefine('R_OK', '04')
679f8833479SBarry Smith      self.framework.addDefine('W_OK', '02')
680f8833479SBarry Smith      self.framework.addDefine('X_OK', '01')
681f8833479SBarry Smith    if not self.checkLink('#include <sys/stat.h>\n','int a=0;\nif (S_ISDIR(a)){}\n'):
682f8833479SBarry Smith      self.framework.addDefine('S_ISREG(a)', '(((a)&_S_IFMT) == _S_IFREG)')
683f8833479SBarry Smith      self.framework.addDefine('S_ISDIR(a)', '(((a)&_S_IFMT) == _S_IFDIR)')
684f8833479SBarry Smith    if self.checkCompile('#include <Windows.h>\n','LARGE_INTEGER a;\nDWORD b=a.u.HighPart;\n'):
685f8833479SBarry Smith      self.addDefine('HAVE_LARGE_INTEGER_U',1)
686f8833479SBarry Smith
687f8833479SBarry Smith    # Windows requires a Binary file creation flag when creating/opening binary files.  Is a better test in order?
688ef2cfba3SSatish Balay    if self.checkCompile('#include <Windows.h>\n#include <fcntl.h>\n', 'int flags = O_BINARY;'):
689f8833479SBarry Smith      self.addDefine('HAVE_O_BINARY',1)
690f8833479SBarry Smith
691f8833479SBarry Smith    if self.compilers.CC.find('win32fe') >= 0:
692f8833479SBarry Smith      self.addDefine('PATH_SEPARATOR','\';\'')
693f8833479SBarry Smith      self.addDefine('DIR_SEPARATOR','\'\\\\\'')
694f8833479SBarry Smith      self.addDefine('REPLACE_DIR_SEPARATOR','\'/\'')
695f8833479SBarry Smith      self.addDefine('CANNOT_START_DEBUGGER',1)
696f8833479SBarry Smith    else:
697f8833479SBarry Smith      self.addDefine('PATH_SEPARATOR','\':\'')
698f8833479SBarry Smith      self.addDefine('REPLACE_DIR_SEPARATOR','\'\\\\\'')
699f8833479SBarry Smith      self.addDefine('DIR_SEPARATOR','\'/\'')
700bfef2c86SBarry Smith
701f8833479SBarry Smith    return
702f8833479SBarry Smith
703f8833479SBarry Smith#-----------------------------------------------------------------------------------------------------
704569865ddSSatish Balay  def configureDefaultArch(self):
705569865ddSSatish Balay    conffile = os.path.join('conf', 'petscvariables')
706569865ddSSatish Balay    if self.framework.argDB['with-default-arch']:
707569865ddSSatish Balay      fd = file(conffile, 'w')
708569865ddSSatish Balay      fd.write('PETSC_ARCH='+self.arch.arch+'\n')
709da93591fSBarry Smith      fd.write('PETSC_DIR='+self.petscdir.dir+'\n')
710569865ddSSatish Balay      fd.write('include ${PETSC_DIR}/${PETSC_ARCH}/conf/petscvariables\n')
711569865ddSSatish Balay      fd.close()
712569865ddSSatish Balay      self.framework.actions.addArgument('PETSc', 'Build', 'Set default architecture to '+self.arch.arch+' in '+conffile)
713569865ddSSatish Balay    elif os.path.isfile(conffile):
714569865ddSSatish Balay      try:
715569865ddSSatish Balay        os.unlink(conffile)
716569865ddSSatish Balay      except:
717569865ddSSatish Balay        raise RuntimeError('Unable to remove file '+conffile+'. Did a different user create it?')
718569865ddSSatish Balay    return
719569865ddSSatish Balay
720569865ddSSatish Balay#-----------------------------------------------------------------------------------------------------
721f8833479SBarry Smith  def configureScript(self):
722f8833479SBarry Smith    '''Output a script in the conf directory which will reproduce the configuration'''
723f8833479SBarry Smith    import nargs
724495bf1a9SSatish Balay    import sys
725495bf1a9SSatish Balay    scriptName = os.path.join(self.arch.arch,'conf', 'reconfigure-'+self.arch.arch+'.py')
726f8833479SBarry Smith    args = dict([(nargs.Arg.parseArgument(arg)[0], arg) for arg in self.framework.clArgs])
727f8833479SBarry Smith    if 'configModules' in args:
7281063a081SSatish Balay      if nargs.Arg.parseArgument(args['configModules'])[1] == 'PETSc.Configure':
729f8833479SBarry Smith        del args['configModules']
730f8833479SBarry Smith    if 'optionsModule' in args:
731c1486898SMatthew Knepley      if nargs.Arg.parseArgument(args['optionsModule'])[1] == 'PETSc.compilerOptions':
732f8833479SBarry Smith        del args['optionsModule']
733f8833479SBarry Smith    if not 'PETSC_ARCH' in args:
7341063a081SSatish Balay      args['PETSC_ARCH'] = 'PETSC_ARCH='+str(self.arch.arch)
735f8833479SBarry Smith    f = file(scriptName, 'w')
736495bf1a9SSatish Balay    f.write('#!'+sys.executable+'\n')
737f8833479SBarry Smith    f.write('if __name__ == \'__main__\':\n')
738f8833479SBarry Smith    f.write('  import sys\n')
7397561c02cSSatish Balay    f.write('  import os\n')
7407561c02cSSatish Balay    f.write('  sys.path.insert(0, os.path.abspath(\'config\'))\n')
741f8833479SBarry Smith    f.write('  import configure\n')
7421063a081SSatish Balay    # pretty print repr(args.values())
7431063a081SSatish Balay    f.write('  configure_options = [\n')
7448bec23c5SJed Brown    for itm in sorted(args.values()):
7451063a081SSatish Balay      f.write('    \''+str(itm)+'\',\n')
7461063a081SSatish Balay    f.write('  ]\n')
747f8833479SBarry Smith    f.write('  configure.petsc_configure(configure_options)\n')
748f8833479SBarry Smith    f.close()
749f8833479SBarry Smith    try:
750f8833479SBarry Smith      os.chmod(scriptName, 0775)
751f8833479SBarry Smith    except OSError, e:
752f8833479SBarry Smith      self.framework.logPrint('Unable to make reconfigure script executable:\n'+str(e))
753f8833479SBarry Smith    self.framework.actions.addArgument('PETSc', 'File creation', 'Created '+scriptName+' for automatic reconfiguration')
754f8833479SBarry Smith    return
755f8833479SBarry Smith
756f8833479SBarry Smith  def configureInstall(self):
757f8833479SBarry Smith    '''Setup the directories for installation'''
758f8833479SBarry Smith    if self.framework.argDB['prefix']:
759f8833479SBarry Smith      self.installdir = self.framework.argDB['prefix']
760824e893fSSatish Balay      self.addMakeRule('shared_install','',['-@echo "Now to install the libraries do:"',\
761824e893fSSatish Balay                                              '-@echo "make PETSC_DIR=${PETSC_DIR} PETSC_ARCH=${PETSC_ARCH} install"',\
762315b77e6SSatish Balay                                              '-@echo "========================================="'])
763f8833479SBarry Smith    else:
764b559b537SSatish Balay      self.installdir = os.path.join(self.petscdir.dir,self.arch.arch)
765824e893fSSatish Balay      self.addMakeRule('shared_install','',['-@echo "Now to check if the libraries are working do:"',\
766824e893fSSatish Balay                                              '-@echo "make PETSC_DIR=${PETSC_DIR} PETSC_ARCH=${PETSC_ARCH} test"',\
767315b77e6SSatish Balay                                              '-@echo "========================================="'])
768f8833479SBarry Smith      return
769f8833479SBarry Smith
770f8833479SBarry Smith  def configureGCOV(self):
771f8833479SBarry Smith    if self.framework.argDB['with-gcov']:
772f8833479SBarry Smith      self.addDefine('USE_GCOV','1')
773f8833479SBarry Smith    return
774f8833479SBarry Smith
775f8833479SBarry Smith  def configureFortranFlush(self):
776f8833479SBarry Smith    if hasattr(self.compilers, 'FC'):
777f8833479SBarry Smith      for baseName in ['flush','flush_']:
778f8833479SBarry Smith        if self.libraries.check('', baseName, otherLibs = self.compilers.flibs, fortranMangle = 1):
779f8833479SBarry Smith          self.addDefine('HAVE_'+baseName.upper(), 1)
780f8833479SBarry Smith          return
781f8833479SBarry Smith
782f8833479SBarry Smith
783f8833479SBarry Smith  def configure(self):
784f8833479SBarry Smith    if not os.path.samefile(self.petscdir.dir, os.getcwd()):
785f8833479SBarry Smith      raise RuntimeError('Wrong PETSC_DIR option specified: '+str(self.petscdir.dir) + '\n  Configure invoked in: '+os.path.realpath(os.getcwd()))
786550489e3SMatthew 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):
7873552d8fbSSatish Balay      raise RuntimeError('Incorrect option --prefix='+self.framework.argDB['prefix']+' specified. It cannot be same as PETSC_DIR!')
788c16c35a9SSatish 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)):
789c16c35a9SSatish Balay      raise RuntimeError('Incorrect option --prefix='+self.framework.argDB['prefix']+' specified. It cannot be same as PETSC_DIR/PETSC_ARCH!')
790f16c1317SJed Brown    self.framework.header          = os.path.join(self.arch.arch,'include','petscconf.h')
791f16c1317SJed Brown    self.framework.cHeader         = os.path.join(self.arch.arch,'include','petscfix.h')
792f16c1317SJed Brown    self.framework.makeMacroHeader = os.path.join(self.arch.arch,'conf','petscvariables')
793f16c1317SJed Brown    self.framework.makeRuleHeader  = os.path.join(self.arch.arch,'conf','petscrules')
794f8833479SBarry Smith    if self.libraries.math is None:
795f8833479SBarry Smith      raise RuntimeError('PETSc requires a functional math library. Please send configure.log to petsc-maint@mcs.anl.gov.')
796f8833479SBarry Smith    if self.languages.clanguage == 'Cxx' and not hasattr(self.compilers, 'CXX'):
797f8833479SBarry Smith      raise RuntimeError('Cannot set C language to C++ without a functional C++ compiler.')
798f8833479SBarry Smith    self.executeTest(self.configureInline)
799b2843cf1SBarry Smith    self.executeTest(self.configurePrefetch)
8002400fdedSBarry Smith    self.executeTest(self.configureUnused)
8019800092aSJed Brown    self.executeTest(self.configureExpect);
80253c77d0aSJed Brown    self.executeTest(self.configureFunctionName);
803753ebd1dSJed Brown    self.executeTest(self.configureIntptrt);
804f8833479SBarry Smith    self.executeTest(self.configureSolaris)
805f8833479SBarry Smith    self.executeTest(self.configureLinux)
806f8833479SBarry Smith    self.executeTest(self.configureWin32)
807569865ddSSatish Balay    self.executeTest(self.configureDefaultArch)
808f8833479SBarry Smith    self.executeTest(self.configureScript)
809f8833479SBarry Smith    self.executeTest(self.configureInstall)
810f8833479SBarry Smith    self.executeTest(self.configureGCOV)
811f8833479SBarry Smith    self.executeTest(self.configureFortranFlush)
81273fca5a0SBarry Smith    self.executeTest(self.configureFeatureTestMacros)
81309bc878fSSatish Balay    self.executeTest(self.configureAtoll)
814f8833479SBarry Smith    # dummy rules, always needed except for remote builds
815f8833479SBarry Smith    self.addMakeRule('remote','')
816f8833479SBarry Smith    self.addMakeRule('remoteclean','')
817f8833479SBarry Smith
818f8833479SBarry Smith    self.Dump()
819f8833479SBarry Smith    self.dumpConfigInfo()
8202a4161d9SMatthew G Knepley    self.dumpMachineInfo()
821511a6afcSJed Brown    self.dumpCMakeConfig()
8228b0282a9SJed Brown    self.dumpCMakeLists()
8238b0282a9SJed Brown    self.cmakeBoot()
824f8833479SBarry Smith    self.framework.log.write('================================================================================\n')
825f8833479SBarry Smith    self.logClear()
826f8833479SBarry Smith    return
827