xref: /petsc/config/PETSc/Configure.py (revision 8c1ceebf810b8cb94f76e7ffaecfbe6acf0809ef)
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')
23*8c1ceebfSJed Brown    if self.getMakeMacro('PETSC_BUILD_USING_CMAKE'):
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',
9224edbb7eSSatish Balay                                            'WindowsX', 'cxxabi','float','ieeefp','stdint','fenv','sched','pthread'])
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))
4689b12c9c7SJed Brown      if self.cmakeboot_success:
4699b12c9c7SJed Brown        if self.framework.argDB['with-cuda']: # Our CMake build does not support CUDA at this time
4709b12c9c7SJed Brown          self.framework.logPrint('CMake configured successfully, but could not be used by default because --with-cuda was used\n')
4719b12c9c7SJed Brown        else:
4729b12c9c7SJed Brown          self.framework.logPrint('CMake configured successfully, using as default build\n')
473f7b66a64SJed Brown          self.addMakeMacro('PETSC_BUILD_USING_CMAKE',1)
474aac20692SSatish Balay      else:
4759b12c9c7SJed Brown        self.framework.logPrint('CMake configuration was unsuccessful\n')
4769b12c9c7SJed Brown    else:
477aac20692SSatish Balay      self.framework.logPrint('Skipping cmakeboot due to old python version: ' +str(sys.version_info) )
478356464bcSMatthew G Knepley    return
4798b0282a9SJed Brown
480b2843cf1SBarry Smith  def configurePrefetch(self):
481b2843cf1SBarry Smith    '''Sees if there are any prefetch functions supported'''
4823649974fSBarry Smith    if config.setCompilers.Configure.isSolaris() or self.framework.argDB['with-iphone'] or self.framework.argDB['with-cuda']:
48393f78423SSatish Balay      self.addDefine('Prefetch(a,b,c)', ' ')
48493f78423SSatish Balay      return
485ec284106SBarry Smith    self.pushLanguage(self.languages.clanguage)
48610699583SJed Brown    if self.checkLink('#include <xmmintrin.h>', 'void *v = 0;_mm_prefetch((const char*)v,_MM_HINT_NTA);\n'):
48750d8bf02SJed Brown      # The Intel Intrinsics manual [1] specifies the prototype
48850d8bf02SJed Brown      #
48950d8bf02SJed Brown      #   void _mm_prefetch(char const *a, int sel);
49050d8bf02SJed Brown      #
49150d8bf02SJed Brown      # but other vendors seem to insist on using subtly different
49250d8bf02SJed Brown      # prototypes, including void* for the pointer, and an enum for
49350d8bf02SJed Brown      # sel.  These are both reasonable changes, but negatively impact
49450d8bf02SJed Brown      # portability.
49550d8bf02SJed Brown      #
49650d8bf02SJed Brown      # [1] http://software.intel.com/file/6373
49750d8bf02SJed Brown      self.addDefine('HAVE_XMMINTRIN_H', 1)
49850d8bf02SJed Brown      self.addDefine('Prefetch(a,b,c)', '_mm_prefetch((const char*)(a),(c))')
49950d8bf02SJed Brown      self.addDefine('PREFETCH_HINT_NTA', '_MM_HINT_NTA')
50050d8bf02SJed Brown      self.addDefine('PREFETCH_HINT_T0',  '_MM_HINT_T0')
50150d8bf02SJed Brown      self.addDefine('PREFETCH_HINT_T1',  '_MM_HINT_T1')
50250d8bf02SJed Brown      self.addDefine('PREFETCH_HINT_T2',  '_MM_HINT_T2')
50350d8bf02SJed Brown    elif self.checkLink('#include <xmmintrin.h>', 'void *v = 0;_mm_prefetch(v,_MM_HINT_NTA);\n'):
50450d8bf02SJed Brown      self.addDefine('HAVE_XMMINTRIN_H', 1)
50550d8bf02SJed Brown      self.addDefine('Prefetch(a,b,c)', '_mm_prefetch((const void*)(a),(c))')
50650d8bf02SJed Brown      self.addDefine('PREFETCH_HINT_NTA', '_MM_HINT_NTA')
50750d8bf02SJed Brown      self.addDefine('PREFETCH_HINT_T0',  '_MM_HINT_T0')
50850d8bf02SJed Brown      self.addDefine('PREFETCH_HINT_T1',  '_MM_HINT_T1')
50950d8bf02SJed Brown      self.addDefine('PREFETCH_HINT_T2',  '_MM_HINT_T2')
51010699583SJed Brown    elif self.checkLink('', 'void *v = 0;__builtin_prefetch(v,0,0);\n'):
51110699583SJed Brown      # From GCC docs: void __builtin_prefetch(const void *addr,int rw,int locality)
51210699583SJed Brown      #
51310699583SJed Brown      #   The value of rw is a compile-time constant one or zero; one
51410699583SJed Brown      #   means that the prefetch is preparing for a write to the memory
51510699583SJed Brown      #   address and zero, the default, means that the prefetch is
51610699583SJed Brown      #   preparing for a read. The value locality must be a compile-time
51710699583SJed Brown      #   constant integer between zero and three. A value of zero means
51810699583SJed Brown      #   that the data has no temporal locality, so it need not be left
51910699583SJed Brown      #   in the cache after the access. A value of three means that the
52010699583SJed Brown      #   data has a high degree of temporal locality and should be left
52110699583SJed Brown      #   in all levels of cache possible. Values of one and two mean,
52210699583SJed Brown      #   respectively, a low or moderate degree of temporal locality.
52310699583SJed Brown      #
52410699583SJed Brown      # Here we adopt Intel's x86/x86-64 naming scheme for the locality
52510699583SJed Brown      # hints.  Using macros for these values in necessary since some
52610699583SJed Brown      # compilers require an enum.
52710699583SJed Brown      self.addDefine('Prefetch(a,b,c)', '__builtin_prefetch((a),(b),(c))')
52810699583SJed Brown      self.addDefine('PREFETCH_HINT_NTA', '0')
52910699583SJed Brown      self.addDefine('PREFETCH_HINT_T0',  '3')
53010699583SJed Brown      self.addDefine('PREFETCH_HINT_T1',  '2')
53110699583SJed Brown      self.addDefine('PREFETCH_HINT_T2',  '1')
532b2843cf1SBarry Smith    else:
533b2843cf1SBarry Smith      self.addDefine('Prefetch(a,b,c)', ' ')
5347d490b44SBarry Smith    self.popLanguage()
535b2843cf1SBarry Smith
53673fca5a0SBarry Smith  def configureFeatureTestMacros(self):
53773fca5a0SBarry Smith    '''Checks if certain feature test macros are support'''
53809bc878fSSatish Balay    if self.checkCompile('#define _POSIX_C_SOURCE 200112L\n#include <stdlib.h>',''):
53973fca5a0SBarry Smith       self.addDefine('_POSIX_C_SOURCE_200112L', '1')
54009bc878fSSatish Balay    if self.checkCompile('#define _BSD_SOURCE\n#include<stdlib.h>',''):
54173fca5a0SBarry Smith       self.addDefine('_BSD_SOURCE', '1')
54273fca5a0SBarry Smith
54309bc878fSSatish Balay  def configureAtoll(self):
54409bc878fSSatish Balay    '''Checks if atoll exists'''
54509bc878fSSatish 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")'):
54609bc878fSSatish Balay       self.addDefine('HAVE_ATOLL', '1')
54709bc878fSSatish Balay
5482400fdedSBarry Smith  def configureUnused(self):
5492400fdedSBarry Smith    '''Sees if __attribute((unused)) is supported'''
5502400fdedSBarry Smith    if self.framework.argDB['with-iphone'] or self.framework.argDB['with-cuda']:
5512400fdedSBarry Smith      self.addDefine('UNUSED', ' ')
5522400fdedSBarry Smith      return
5532400fdedSBarry Smith    self.pushLanguage(self.languages.clanguage)
5547bf3d34bSSatish Balay    if self.checkLink('__attribute((unused)) static int myfunc(void){ return 1;}', 'int i = myfunc();\ntypedef void* atype;\n__attribute((unused))  atype a;\n'):
5552400fdedSBarry Smith      self.addDefine('UNUSED', '__attribute((unused))')
5562400fdedSBarry Smith    else:
5572400fdedSBarry Smith      self.addDefine('UNUSED', ' ')
5582400fdedSBarry Smith    self.popLanguage()
5592400fdedSBarry Smith
5609800092aSJed Brown  def configureExpect(self):
5619800092aSJed Brown    '''Sees if the __builtin_expect directive is supported'''
5629800092aSJed Brown    self.pushLanguage(self.languages.clanguage)
5639800092aSJed Brown    if self.checkLink('', 'if (__builtin_expect(0,1)) return 1;'):
5649800092aSJed Brown      self.addDefine('HAVE_BUILTIN_EXPECT', 1)
5659800092aSJed Brown    self.popLanguage()
5669800092aSJed Brown
56753c77d0aSJed Brown  def configureFunctionName(self):
5681ec50b02SJed Brown    '''Sees if the compiler supports __func__ or a variant.  Falls back
5691ec50b02SJed Brown    on __FUNCT__ which PETSc source defines, but most users do not, thus
5701ec50b02SJed Brown    stack traces through user code are better when the compiler's
5711ec50b02SJed Brown    variant is used.'''
5721ec50b02SJed Brown    def getFunctionName(lang):
5731ec50b02SJed Brown      name = '__FUNCT__'
5741ec50b02SJed Brown      self.pushLanguage(lang)
57553c77d0aSJed Brown      if self.checkLink('', "if (__func__[0] != 'm') return 1;"):
5761ec50b02SJed Brown        name = '__func__'
57753c77d0aSJed Brown      elif self.checkLink('', "if (__FUNCTION__[0] != 'm') return 1;"):
5781ec50b02SJed Brown        name = '__FUNCTION__'
5791ec50b02SJed Brown      self.popLanguage()
5801ec50b02SJed Brown      return name
5811ec50b02SJed Brown    langs = []
582628773c9SSatish Balay
583628773c9SSatish Balay    self.addDefine('FUNCTION_NAME_C', getFunctionName('C'))
584628773c9SSatish Balay    if hasattr(self.compilers, 'CXX'):
585628773c9SSatish Balay      self.addDefine('FUNCTION_NAME_CXX', getFunctionName('Cxx'))
58612607bf0SSatish Balay    else:
58712607bf0SSatish Balay      self.addDefine('FUNCTION_NAME_CXX', '__FUNCT__')
58853c77d0aSJed Brown
589753ebd1dSJed Brown  def configureIntptrt(self):
590753ebd1dSJed Brown    '''Determine what to use for uintptr_t'''
591753ebd1dSJed Brown    def staticAssertSizeMatchesVoidStar(inc,typename):
592753ebd1dSJed Brown      # The declaration is an error if either array size is negative.
593753ebd1dSJed 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
594d26187a0SJed Brown      return self.checkCompile(inc, ('#define STATIC_ASSERT(cond) char negative_length_if_false[2*(!!(cond))-1]\n'
595979939cdSSatish Balay                                     + 'STATIC_ASSERT(sizeof(void*) == sizeof(%s));'%typename))
596753ebd1dSJed Brown    self.pushLanguage(self.languages.clanguage)
597753ebd1dSJed Brown    if self.checkCompile('#include <stdint.h>', 'int x; uintptr_t i = (uintptr_t)&x;'):
598753ebd1dSJed Brown      self.addDefine('UINTPTR_T', 'uintptr_t')
599753ebd1dSJed Brown    elif staticAssertSizeMatchesVoidStar('','unsigned long long'):
600753ebd1dSJed Brown      self.addDefine('UINTPTR_T', 'unsigned long long')
601753ebd1dSJed Brown    elif staticAssertSizeMatchesVoidStar('#include <stdlib.h>','size_t') or staticAssertSizeMatchesVoidStar('#include <string.h>', 'size_t'):
602753ebd1dSJed Brown      self.addDefine('UINTPTR_T', 'size_t')
603c82284b1SJed Brown    elif staticAssertSizeMatchesVoidStar('','unsigned long'):
604c82284b1SJed Brown      self.addDefine('UINTPTR_T', 'unsigned long')
6052d1b7972SSatish Balay    elif staticAssertSizeMatchesVoidStar('','unsigned'):
606753ebd1dSJed Brown      self.addDefine('UINTPTR_T', 'unsigned')
607d26187a0SJed Brown    else:
608d26187a0SJed Brown      raise RuntimeError('Could not find any unsigned integer type matching void*')
609753ebd1dSJed Brown    self.popLanguage()
610753ebd1dSJed Brown
611f8833479SBarry Smith  def configureInline(self):
612f8833479SBarry Smith    '''Get a generic inline keyword, depending on the language'''
613f8833479SBarry Smith    if self.languages.clanguage == 'C':
614f8833479SBarry Smith      self.addDefine('STATIC_INLINE', self.compilers.cStaticInlineKeyword)
615f8833479SBarry Smith      self.addDefine('RESTRICT', self.compilers.cRestrict)
616f8833479SBarry Smith    elif self.languages.clanguage == 'Cxx':
617f8833479SBarry Smith      self.addDefine('STATIC_INLINE', self.compilers.cxxStaticInlineKeyword)
618f8833479SBarry Smith      self.addDefine('RESTRICT', self.compilers.cxxRestrict)
619bfef2c86SBarry Smith
620bfef2c86SBarry Smith    if self.checkCompile('#include <dlfcn.h>\n void *ptr =  RTLD_DEFAULT;'):
621bfef2c86SBarry Smith      self.addDefine('RTLD_DEFAULT','1')
622f8833479SBarry Smith    return
623f8833479SBarry Smith
624f8833479SBarry Smith  def configureSolaris(self):
625f8833479SBarry Smith    '''Solaris specific stuff'''
626f8833479SBarry Smith    if os.path.isdir(os.path.join('/usr','ucblib')):
627f8833479SBarry Smith      try:
628f8833479SBarry Smith        flag = getattr(self.setCompilers, self.language[-1]+'SharedLinkerFlag')
629f8833479SBarry Smith      except AttributeError:
630f8833479SBarry Smith        flag = None
631f8833479SBarry Smith      if flag is None:
632f8833479SBarry Smith        self.compilers.LIBS += ' -L/usr/ucblib'
633f8833479SBarry Smith      else:
634f8833479SBarry Smith        self.compilers.LIBS += ' '+flag+'/usr/ucblib'
635f8833479SBarry Smith    return
636f8833479SBarry Smith
637f8833479SBarry Smith  def configureLinux(self):
638f8833479SBarry Smith    '''Linux specific stuff'''
6399f15855cSMatthew G Knepley    # TODO: Test for this by mallocing an odd number of floats and checking the address
640f8833479SBarry Smith    self.addDefine('HAVE_DOUBLE_ALIGN_MALLOC', 1)
641f8833479SBarry Smith    return
642f8833479SBarry Smith
643f8833479SBarry Smith  def configureWin32(self):
644f8833479SBarry Smith    '''Win32 non-cygwin specific stuff'''
645f8833479SBarry Smith    kernel32=0
646f8833479SBarry Smith    if self.libraries.add('Kernel32.lib','GetComputerName',prototype='#include <Windows.h>', call='GetComputerName(NULL,NULL);'):
647f8833479SBarry Smith      self.addDefine('HAVE_WINDOWS_H',1)
648f8833479SBarry Smith      self.addDefine('HAVE_GETCOMPUTERNAME',1)
649f8833479SBarry Smith      kernel32=1
650f8833479SBarry Smith    elif self.libraries.add('kernel32','GetComputerName',prototype='#include <Windows.h>', call='GetComputerName(NULL,NULL);'):
651f8833479SBarry Smith      self.addDefine('HAVE_WINDOWS_H',1)
652f8833479SBarry Smith      self.addDefine('HAVE_GETCOMPUTERNAME',1)
653f8833479SBarry Smith      kernel32=1
654f8833479SBarry Smith    if kernel32:
655eed94e11SSatish Balay      if self.framework.argDB['with-windows-graphics']:
656eed94e11SSatish Balay        self.addDefine('USE_WINDOWS_GRAPHICS',1)
657f8833479SBarry Smith      if self.checkLink('#include <Windows.h>','LoadLibrary(0)'):
658f8833479SBarry Smith        self.addDefine('HAVE_LOADLIBRARY',1)
659b50f6d9eSLisandro Dalcin      if self.checkLink('#include <Windows.h>','GetProcAddress(0,0)'):
660b50f6d9eSLisandro Dalcin        self.addDefine('HAVE_GETPROCADDRESS',1)
661b50f6d9eSLisandro Dalcin      if self.checkLink('#include <Windows.h>','FreeLibrary(0)'):
662b50f6d9eSLisandro Dalcin        self.addDefine('HAVE_FREELIBRARY',1)
663a21658a3SLisandro Dalcin      if self.checkLink('#include <Windows.h>','GetLastError()'):
664a21658a3SLisandro Dalcin        self.addDefine('HAVE_GETLASTERROR',1)
665a21658a3SLisandro Dalcin      if self.checkLink('#include <Windows.h>','SetLastError(0)'):
666a21658a3SLisandro Dalcin        self.addDefine('HAVE_SETLASTERROR',1)
667f8833479SBarry Smith      if self.checkLink('#include <Windows.h>\n','QueryPerformanceCounter(0);\n'):
668f8833479SBarry Smith        self.addDefine('USE_NT_TIME',1)
669f8833479SBarry Smith    if self.libraries.add('Advapi32.lib','GetUserName',prototype='#include <Windows.h>', call='GetUserName(NULL,NULL);'):
670f8833479SBarry Smith      self.addDefine('HAVE_GET_USER_NAME',1)
671f8833479SBarry Smith    elif self.libraries.add('advapi32','GetUserName',prototype='#include <Windows.h>', call='GetUserName(NULL,NULL);'):
672f8833479SBarry Smith      self.addDefine('HAVE_GET_USER_NAME',1)
673f8833479SBarry Smith
674f8833479SBarry Smith    if not self.libraries.add('User32.lib','GetDC',prototype='#include <Windows.h>',call='GetDC(0);'):
675f8833479SBarry Smith      self.libraries.add('user32','GetDC',prototype='#include <Windows.h>',call='GetDC(0);')
676f8833479SBarry Smith    if not self.libraries.add('Gdi32.lib','CreateCompatibleDC',prototype='#include <Windows.h>',call='CreateCompatibleDC(0);'):
677f8833479SBarry Smith      self.libraries.add('gdi32','CreateCompatibleDC',prototype='#include <Windows.h>',call='CreateCompatibleDC(0);')
678f8833479SBarry Smith
679f8833479SBarry Smith    self.types.check('int32_t', 'int')
680f8833479SBarry Smith    if not self.checkCompile('#include <sys/types.h>\n','uid_t u;\n'):
681f8833479SBarry Smith      self.addTypedef('int', 'uid_t')
682f8833479SBarry Smith      self.addTypedef('int', 'gid_t')
683f8833479SBarry Smith    if not self.checkLink('#if defined(PETSC_HAVE_UNISTD_H)\n#include <unistd.h>\n#endif\n','int a=R_OK;\n'):
684f8833479SBarry Smith      self.framework.addDefine('R_OK', '04')
685f8833479SBarry Smith      self.framework.addDefine('W_OK', '02')
686f8833479SBarry Smith      self.framework.addDefine('X_OK', '01')
687f8833479SBarry Smith    if not self.checkLink('#include <sys/stat.h>\n','int a=0;\nif (S_ISDIR(a)){}\n'):
688f8833479SBarry Smith      self.framework.addDefine('S_ISREG(a)', '(((a)&_S_IFMT) == _S_IFREG)')
689f8833479SBarry Smith      self.framework.addDefine('S_ISDIR(a)', '(((a)&_S_IFMT) == _S_IFDIR)')
690f8833479SBarry Smith    if self.checkCompile('#include <Windows.h>\n','LARGE_INTEGER a;\nDWORD b=a.u.HighPart;\n'):
691f8833479SBarry Smith      self.addDefine('HAVE_LARGE_INTEGER_U',1)
692f8833479SBarry Smith
693f8833479SBarry Smith    # Windows requires a Binary file creation flag when creating/opening binary files.  Is a better test in order?
694ef2cfba3SSatish Balay    if self.checkCompile('#include <Windows.h>\n#include <fcntl.h>\n', 'int flags = O_BINARY;'):
695f8833479SBarry Smith      self.addDefine('HAVE_O_BINARY',1)
696f8833479SBarry Smith
697f8833479SBarry Smith    if self.compilers.CC.find('win32fe') >= 0:
698f8833479SBarry Smith      self.addDefine('PATH_SEPARATOR','\';\'')
699f8833479SBarry Smith      self.addDefine('DIR_SEPARATOR','\'\\\\\'')
700f8833479SBarry Smith      self.addDefine('REPLACE_DIR_SEPARATOR','\'/\'')
701f8833479SBarry Smith      self.addDefine('CANNOT_START_DEBUGGER',1)
702f8833479SBarry Smith    else:
703f8833479SBarry Smith      self.addDefine('PATH_SEPARATOR','\':\'')
704f8833479SBarry Smith      self.addDefine('REPLACE_DIR_SEPARATOR','\'\\\\\'')
705f8833479SBarry Smith      self.addDefine('DIR_SEPARATOR','\'/\'')
706bfef2c86SBarry Smith
707f8833479SBarry Smith    return
708f8833479SBarry Smith
709f8833479SBarry Smith#-----------------------------------------------------------------------------------------------------
710569865ddSSatish Balay  def configureDefaultArch(self):
711569865ddSSatish Balay    conffile = os.path.join('conf', 'petscvariables')
712569865ddSSatish Balay    if self.framework.argDB['with-default-arch']:
713569865ddSSatish Balay      fd = file(conffile, 'w')
714569865ddSSatish Balay      fd.write('PETSC_ARCH='+self.arch.arch+'\n')
715da93591fSBarry Smith      fd.write('PETSC_DIR='+self.petscdir.dir+'\n')
7163435dde8SSatish Balay      fd.write('include '+os.path.join(self.petscdir.dir,self.arch.arch,'conf','petscvariables')+'\n')
717569865ddSSatish Balay      fd.close()
718569865ddSSatish Balay      self.framework.actions.addArgument('PETSc', 'Build', 'Set default architecture to '+self.arch.arch+' in '+conffile)
719569865ddSSatish Balay    elif os.path.isfile(conffile):
720569865ddSSatish Balay      try:
721569865ddSSatish Balay        os.unlink(conffile)
722569865ddSSatish Balay      except:
723569865ddSSatish Balay        raise RuntimeError('Unable to remove file '+conffile+'. Did a different user create it?')
724569865ddSSatish Balay    return
725569865ddSSatish Balay
726569865ddSSatish Balay#-----------------------------------------------------------------------------------------------------
727f8833479SBarry Smith  def configureScript(self):
728f8833479SBarry Smith    '''Output a script in the conf directory which will reproduce the configuration'''
729f8833479SBarry Smith    import nargs
730495bf1a9SSatish Balay    import sys
731495bf1a9SSatish Balay    scriptName = os.path.join(self.arch.arch,'conf', 'reconfigure-'+self.arch.arch+'.py')
732f8833479SBarry Smith    args = dict([(nargs.Arg.parseArgument(arg)[0], arg) for arg in self.framework.clArgs])
733f8833479SBarry Smith    if 'configModules' in args:
7341063a081SSatish Balay      if nargs.Arg.parseArgument(args['configModules'])[1] == 'PETSc.Configure':
735f8833479SBarry Smith        del args['configModules']
736f8833479SBarry Smith    if 'optionsModule' in args:
737c1486898SMatthew Knepley      if nargs.Arg.parseArgument(args['optionsModule'])[1] == 'PETSc.compilerOptions':
738f8833479SBarry Smith        del args['optionsModule']
739f8833479SBarry Smith    if not 'PETSC_ARCH' in args:
7401063a081SSatish Balay      args['PETSC_ARCH'] = 'PETSC_ARCH='+str(self.arch.arch)
741f8833479SBarry Smith    f = file(scriptName, 'w')
742495bf1a9SSatish Balay    f.write('#!'+sys.executable+'\n')
743f8833479SBarry Smith    f.write('if __name__ == \'__main__\':\n')
744f8833479SBarry Smith    f.write('  import sys\n')
7457561c02cSSatish Balay    f.write('  import os\n')
7467561c02cSSatish Balay    f.write('  sys.path.insert(0, os.path.abspath(\'config\'))\n')
747f8833479SBarry Smith    f.write('  import configure\n')
7481063a081SSatish Balay    # pretty print repr(args.values())
7491063a081SSatish Balay    f.write('  configure_options = [\n')
7508bec23c5SJed Brown    for itm in sorted(args.values()):
7511063a081SSatish Balay      f.write('    \''+str(itm)+'\',\n')
7521063a081SSatish Balay    f.write('  ]\n')
753f8833479SBarry Smith    f.write('  configure.petsc_configure(configure_options)\n')
754f8833479SBarry Smith    f.close()
755f8833479SBarry Smith    try:
756f8833479SBarry Smith      os.chmod(scriptName, 0775)
757f8833479SBarry Smith    except OSError, e:
758f8833479SBarry Smith      self.framework.logPrint('Unable to make reconfigure script executable:\n'+str(e))
759f8833479SBarry Smith    self.framework.actions.addArgument('PETSc', 'File creation', 'Created '+scriptName+' for automatic reconfiguration')
760f8833479SBarry Smith    return
761f8833479SBarry Smith
762f8833479SBarry Smith  def configureInstall(self):
763f8833479SBarry Smith    '''Setup the directories for installation'''
764f8833479SBarry Smith    if self.framework.argDB['prefix']:
765f8833479SBarry Smith      self.installdir = self.framework.argDB['prefix']
766824e893fSSatish Balay      self.addMakeRule('shared_install','',['-@echo "Now to install the libraries do:"',\
767824e893fSSatish Balay                                              '-@echo "make PETSC_DIR=${PETSC_DIR} PETSC_ARCH=${PETSC_ARCH} install"',\
768315b77e6SSatish Balay                                              '-@echo "========================================="'])
769f8833479SBarry Smith    else:
770b559b537SSatish Balay      self.installdir = os.path.join(self.petscdir.dir,self.arch.arch)
771824e893fSSatish Balay      self.addMakeRule('shared_install','',['-@echo "Now to check if the libraries are working do:"',\
772824e893fSSatish Balay                                              '-@echo "make PETSC_DIR=${PETSC_DIR} PETSC_ARCH=${PETSC_ARCH} test"',\
773315b77e6SSatish Balay                                              '-@echo "========================================="'])
774f8833479SBarry Smith      return
775f8833479SBarry Smith
776f8833479SBarry Smith  def configureGCOV(self):
777f8833479SBarry Smith    if self.framework.argDB['with-gcov']:
778f8833479SBarry Smith      self.addDefine('USE_GCOV','1')
779f8833479SBarry Smith    return
780f8833479SBarry Smith
781f8833479SBarry Smith  def configureFortranFlush(self):
782f8833479SBarry Smith    if hasattr(self.compilers, 'FC'):
783f8833479SBarry Smith      for baseName in ['flush','flush_']:
784f8833479SBarry Smith        if self.libraries.check('', baseName, otherLibs = self.compilers.flibs, fortranMangle = 1):
785f8833479SBarry Smith          self.addDefine('HAVE_'+baseName.upper(), 1)
786f8833479SBarry Smith          return
787f8833479SBarry Smith
788f8833479SBarry Smith
789f8833479SBarry Smith  def configure(self):
790f8833479SBarry Smith    if not os.path.samefile(self.petscdir.dir, os.getcwd()):
791f8833479SBarry Smith      raise RuntimeError('Wrong PETSC_DIR option specified: '+str(self.petscdir.dir) + '\n  Configure invoked in: '+os.path.realpath(os.getcwd()))
792550489e3SMatthew 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):
7933552d8fbSSatish Balay      raise RuntimeError('Incorrect option --prefix='+self.framework.argDB['prefix']+' specified. It cannot be same as PETSC_DIR!')
794c16c35a9SSatish 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)):
795c16c35a9SSatish Balay      raise RuntimeError('Incorrect option --prefix='+self.framework.argDB['prefix']+' specified. It cannot be same as PETSC_DIR/PETSC_ARCH!')
796f16c1317SJed Brown    self.framework.header          = os.path.join(self.arch.arch,'include','petscconf.h')
797f16c1317SJed Brown    self.framework.cHeader         = os.path.join(self.arch.arch,'include','petscfix.h')
798f16c1317SJed Brown    self.framework.makeMacroHeader = os.path.join(self.arch.arch,'conf','petscvariables')
799f16c1317SJed Brown    self.framework.makeRuleHeader  = os.path.join(self.arch.arch,'conf','petscrules')
800f8833479SBarry Smith    if self.libraries.math is None:
801f8833479SBarry Smith      raise RuntimeError('PETSc requires a functional math library. Please send configure.log to petsc-maint@mcs.anl.gov.')
802f8833479SBarry Smith    if self.languages.clanguage == 'Cxx' and not hasattr(self.compilers, 'CXX'):
803f8833479SBarry Smith      raise RuntimeError('Cannot set C language to C++ without a functional C++ compiler.')
804f8833479SBarry Smith    self.executeTest(self.configureInline)
805b2843cf1SBarry Smith    self.executeTest(self.configurePrefetch)
8062400fdedSBarry Smith    self.executeTest(self.configureUnused)
8079800092aSJed Brown    self.executeTest(self.configureExpect);
80853c77d0aSJed Brown    self.executeTest(self.configureFunctionName);
809753ebd1dSJed Brown    self.executeTest(self.configureIntptrt);
810f8833479SBarry Smith    self.executeTest(self.configureSolaris)
811f8833479SBarry Smith    self.executeTest(self.configureLinux)
812f8833479SBarry Smith    self.executeTest(self.configureWin32)
813569865ddSSatish Balay    self.executeTest(self.configureDefaultArch)
814f8833479SBarry Smith    self.executeTest(self.configureScript)
815f8833479SBarry Smith    self.executeTest(self.configureInstall)
816f8833479SBarry Smith    self.executeTest(self.configureGCOV)
817f8833479SBarry Smith    self.executeTest(self.configureFortranFlush)
81873fca5a0SBarry Smith    self.executeTest(self.configureFeatureTestMacros)
81909bc878fSSatish Balay    self.executeTest(self.configureAtoll)
820f8833479SBarry Smith    # dummy rules, always needed except for remote builds
821f8833479SBarry Smith    self.addMakeRule('remote','')
822f8833479SBarry Smith    self.addMakeRule('remoteclean','')
823f8833479SBarry Smith
824f8833479SBarry Smith    self.Dump()
825f8833479SBarry Smith    self.dumpConfigInfo()
8262a4161d9SMatthew G Knepley    self.dumpMachineInfo()
827511a6afcSJed Brown    self.dumpCMakeConfig()
8288b0282a9SJed Brown    self.dumpCMakeLists()
8298b0282a9SJed Brown    self.cmakeBoot()
830f8833479SBarry Smith    self.framework.log.write('================================================================================\n')
831f8833479SBarry Smith    self.logClear()
832f8833479SBarry Smith    return
833