xref: /petsc/config/PETSc/Configure.py (revision ce63c4c16a990ef3f7b411630a7c76f93edb2e1f)
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')
237c939e48SSatish Balay    desc.append('   Configure stage complete. Now build PETSc libraries with:')
247c939e48SSatish Balay    desc.append('   make PETSC_DIR='+self.petscdir.dir+' PETSC_ARCH='+self.arch.arch+' all')
25a0022257SSatish Balay    desc.append('xxx=========================================================================xxx')
267c939e48SSatish Balay    return '\n'.join(desc)+'\n'
27f8833479SBarry Smith
28f8833479SBarry Smith  def setupHelp(self, help):
29f8833479SBarry Smith    import nargs
30ce0b2093SBarry Smith    help.addArgument('PETSc',  '-prefix=<dir>',                  nargs.Arg(None, '', 'Specifiy location to install PETSc (eg. /usr/local)'))
31eed94e11SSatish Balay    help.addArgument('Windows','-with-windows-graphics=<bool>',   nargs.ArgBool(None, 1,'Enable check for Windows Graphics'))
32569865ddSSatish Balay    help.addArgument('PETSc', '-with-default-arch=<bool>',        nargs.ArgBool(None, 1, 'Allow using the last configured arch without setting PETSC_ARCH'))
3357cb31baSSatish Balay    help.addArgument('PETSc','-with-single-library=<bool>',       nargs.ArgBool(None, 1,'Put all PETSc code into the single -lpetsc library'))
34ce0b2093SBarry Smith    help.addArgument('PETSc', '-with-iphone=<bool>',              nargs.ArgBool(None, 0, 'Build an iPhone version of PETSc'))
35f8833479SBarry Smith    return
36f8833479SBarry Smith
37f8833479SBarry Smith  def setupDependencies(self, framework):
38f8833479SBarry Smith    config.base.Configure.setupDependencies(self, framework)
39f8833479SBarry Smith    self.setCompilers  = framework.require('config.setCompilers',      self)
40f8833479SBarry Smith    self.arch          = framework.require('PETSc.utilities.arch',     self.setCompilers)
41f8833479SBarry Smith    self.petscdir      = framework.require('PETSc.utilities.petscdir', self.setCompilers)
42f8833479SBarry Smith    self.languages     = framework.require('PETSc.utilities.languages',self.setCompilers)
43f8833479SBarry Smith    self.debugging     = framework.require('PETSc.utilities.debugging',self.setCompilers)
44a2b1cf82SBarry Smith    self.CHUD          = framework.require('PETSc.utilities.CHUD',     self)
45f8833479SBarry Smith    self.compilers     = framework.require('config.compilers',         self)
46f8833479SBarry Smith    self.types         = framework.require('config.types',             self)
47f8833479SBarry Smith    self.headers       = framework.require('config.headers',           self)
48f8833479SBarry Smith    self.functions     = framework.require('config.functions',         self)
49f8833479SBarry Smith    self.libraries     = framework.require('config.libraries',         self)
50f8833479SBarry Smith    if os.path.isdir(os.path.join('config', 'PETSc')):
51f8833479SBarry Smith      for d in ['utilities', 'packages']:
52f8833479SBarry Smith        for utility in os.listdir(os.path.join('config', 'PETSc', d)):
53f8833479SBarry Smith          (utilityName, ext) = os.path.splitext(utility)
54f8833479SBarry Smith          if not utilityName.startswith('.') and not utilityName.startswith('#') and ext == '.py' and not utilityName == '__init__':
55f8833479SBarry Smith            utilityObj                    = self.framework.require('PETSc.'+d+'.'+utilityName, self)
56f8833479SBarry Smith            utilityObj.headerPrefix       = self.headerPrefix
57f1ecedd2SMatthew Knepley            utilityObj.archProvider       = self.arch
58fdfda77fSMatthew Knepley            utilityObj.languageProvider   = self.languages
59fdfda77fSMatthew Knepley            utilityObj.installDirProvider = self.petscdir
60f8833479SBarry Smith            setattr(self, utilityName.lower(), utilityObj)
618cf378d1SBarry Smith
62d37554e4SMatthew G Knepley    for package in config.packages.all:
63d37554e4SMatthew G Knepley      if not package == 'PETSc':
64d37554e4SMatthew G Knepley        packageObj                    = framework.require('config.packages.'+package, self)
65d37554e4SMatthew G Knepley        packageObj.archProvider       = self.arch
66d37554e4SMatthew G Knepley        packageObj.languageProvider   = self.languages
67d37554e4SMatthew G Knepley        packageObj.installDirProvider = self.petscdir
68d37554e4SMatthew G Knepley        setattr(self, package.lower(), packageObj)
69d37554e4SMatthew G Knepley    # Force blaslapack to depend on scalarType so precision is set before BlasLapack is built
708cf378d1SBarry Smith    framework.require('PETSc.utilities.scalarTypes', self.blaslapack)
71f8833479SBarry Smith    self.blaslapack.precisionProvider = self.scalartypes
72f8833479SBarry Smith
73f8833479SBarry Smith    self.compilers.headerPrefix  = self.headerPrefix
74f8833479SBarry Smith    self.types.headerPrefix      = self.headerPrefix
75f8833479SBarry Smith    self.headers.headerPrefix    = self.headerPrefix
76f8833479SBarry Smith    self.functions.headerPrefix  = self.headerPrefix
77f8833479SBarry Smith    self.libraries.headerPrefix  = self.headerPrefix
78f8833479SBarry Smith    self.blaslapack.headerPrefix = self.headerPrefix
79f8833479SBarry Smith    self.mpi.headerPrefix        = self.headerPrefix
80f8833479SBarry Smith    headersC = map(lambda name: name+'.h', ['dos', 'endian', 'fcntl', 'float', 'io', 'limits', 'malloc', 'pwd', 'search', 'strings',
81f8833479SBarry Smith                                            'unistd', 'machine/endian', 'sys/param', 'sys/procfs', 'sys/resource',
82b4bb646cSSatish Balay                                            'sys/systeminfo', 'sys/times', 'sys/utsname','string', 'stdlib','memory',
83f8833479SBarry Smith                                            'sys/socket','sys/wait','netinet/in','netdb','Direct','time','Ws2tcpip','sys/types',
84b014e56cSJed Brown                                            'WindowsX', 'cxxabi','float','ieeefp','stdint','fenv'])
85f8833479SBarry Smith    functions = ['access', '_access', 'clock', 'drand48', 'getcwd', '_getcwd', 'getdomainname', 'gethostname', 'getpwuid',
86f8833479SBarry Smith                 'gettimeofday', 'getwd', 'memalign', 'memmove', 'mkstemp', 'popen', 'PXFGETARG', 'rand', 'getpagesize',
87a6d0e24fSJed Brown                 'readlink', 'realpath',  'sigaction', 'signal', 'sigset', 'nanosleep', 'usleep', 'sleep', '_sleep', 'socket',
88a6d0e24fSJed Brown                 'times', 'gethostbyname', 'uname','snprintf','_snprintf','_fullpath','lseek','_lseek','time','fork','stricmp',
89a6d0e24fSJed Brown                 'strcasecmp', 'bzero', 'dlopen', 'dlsym', 'dlclose', 'dlerror',
9064ffe704SBarry Smith                 '_intel_fast_memcpy','_intel_fast_memset']
91f8833479SBarry Smith    libraries1 = [(['socket', 'nsl'], 'socket'), (['fpe'], 'handle_sigfpes')]
92f8833479SBarry Smith    self.headers.headers.extend(headersC)
93f8833479SBarry Smith    self.functions.functions.extend(functions)
94f8833479SBarry Smith    self.libraries.libraries.extend(libraries1)
95f8833479SBarry Smith    return
96f8833479SBarry Smith
97f8833479SBarry Smith  def Dump(self):
98f8833479SBarry Smith    ''' Actually put the values into the configuration files '''
99f8833479SBarry Smith    # eventually everything between -- should be gone
100f8833479SBarry Smith#-----------------------------------------------------------------------------------------------------
101f8833479SBarry Smith
102f8833479SBarry Smith    # Sometimes we need C compiler, even if built with C++
103f8833479SBarry Smith    self.setCompilers.pushLanguage('C')
104f8833479SBarry Smith    self.addMakeMacro('CC_FLAGS',self.setCompilers.getCompilerFlags())
105f8833479SBarry Smith    self.setCompilers.popLanguage()
106f8833479SBarry Smith
107f8833479SBarry Smith    # C preprocessor values
108a2b1cf82SBarry Smith    self.addMakeMacro('CPP_FLAGS',self.setCompilers.CPPFLAGS+self.CHUD.CPPFLAGS)
109f8833479SBarry Smith
110f8833479SBarry Smith    # compiler values
111f8833479SBarry Smith    self.setCompilers.pushLanguage(self.languages.clanguage)
112f8833479SBarry Smith    self.addMakeMacro('PCC',self.setCompilers.getCompiler())
113f8833479SBarry Smith    self.addMakeMacro('PCC_FLAGS',self.setCompilers.getCompilerFlags())
114f8833479SBarry Smith    self.setCompilers.popLanguage()
115f8833479SBarry Smith    # .o or .obj
116f8833479SBarry Smith    self.addMakeMacro('CC_SUFFIX','o')
117f8833479SBarry Smith
118f8833479SBarry Smith    # executable linker values
119f8833479SBarry Smith    self.setCompilers.pushLanguage(self.languages.clanguage)
120f8833479SBarry Smith    pcc_linker = self.setCompilers.getLinker()
121f8833479SBarry Smith    self.addMakeMacro('PCC_LINKER',pcc_linker)
122c84a332bSSatish Balay    self.addMakeMacro('PCC_LINKER_FLAGS',self.setCompilers.getLinkerFlags())
123f8833479SBarry Smith    self.setCompilers.popLanguage()
124f8833479SBarry Smith    # '' for Unix, .exe for Windows
125f8833479SBarry Smith    self.addMakeMacro('CC_LINKER_SUFFIX','')
126f8833479SBarry Smith
127f8833479SBarry Smith    if hasattr(self.compilers, 'FC'):
128f8833479SBarry Smith      self.setCompilers.pushLanguage('FC')
129f8833479SBarry Smith      # need FPPFLAGS in config/setCompilers
130f8833479SBarry Smith      self.addDefine('HAVE_FORTRAN','1')
131f8833479SBarry Smith      self.addMakeMacro('FPP_FLAGS',self.setCompilers.CPPFLAGS)
132f8833479SBarry Smith
133f8833479SBarry Smith      # compiler values
134f8833479SBarry Smith      self.addMakeMacro('FC_FLAGS',self.setCompilers.getCompilerFlags())
135f8833479SBarry Smith      self.setCompilers.popLanguage()
136f8833479SBarry Smith      # .o or .obj
137f8833479SBarry Smith      self.addMakeMacro('FC_SUFFIX','o')
138f8833479SBarry Smith
139f8833479SBarry Smith      # executable linker values
140f8833479SBarry Smith      self.setCompilers.pushLanguage('FC')
141f8833479SBarry Smith      # Cannot have NAG f90 as the linker - so use pcc_linker as fc_linker
142f8833479SBarry Smith      fc_linker = self.setCompilers.getLinker()
143f8833479SBarry Smith      if config.setCompilers.Configure.isNAG(fc_linker):
144f8833479SBarry Smith        self.addMakeMacro('FC_LINKER',pcc_linker)
145f8833479SBarry Smith      else:
146f8833479SBarry Smith        self.addMakeMacro('FC_LINKER',fc_linker)
1476d53d35eSSatish Balay      self.addMakeMacro('FC_LINKER_FLAGS',self.setCompilers.getLinkerFlags())
1483feacd00SBarry Smith      # apple requires this shared library linker flag on SOME versions of the os
1493feacd00SBarry Smith      if self.setCompilers.getLinkerFlags().find('-Wl,-commons,use_dylibs') > -1:
1503feacd00SBarry Smith        self.addMakeMacro('DARWIN_COMMONS_USE_DYLIBS',' -Wl,-commons,use_dylibs ')
151bb82cf9cSSatish Balay      self.setCompilers.popLanguage()
1525d631499SMatthew Knepley
1535d631499SMatthew Knepley      # F90 Modules
1545d631499SMatthew Knepley      if self.setCompilers.fortranModuleIncludeFlag:
1555d631499SMatthew Knepley        self.addMakeMacro('FC_MODULE_FLAG', self.setCompilers.fortranModuleIncludeFlag)
1566ddd6694SSatish Balay      else: # for non-f90 compilers like g77
1576ddd6694SSatish Balay        self.addMakeMacro('FC_MODULE_FLAG', '-I')
158a324c51cSMatthew G Knepley      if self.setCompilers.fortranModuleIncludeFlag:
159a324c51cSMatthew G Knepley        self.addMakeMacro('FC_MODULE_OUTPUT_FLAG', self.setCompilers.fortranModuleOutputFlag)
160f8833479SBarry Smith    else:
161f8833479SBarry Smith      self.addMakeMacro('FC','')
162f8833479SBarry Smith
16346a3958fSBarry Smith    if hasattr(self.compilers, 'CUDAC'):
1647ff2890cSSatish Balay      self.setCompilers.pushLanguage('CUDA')
165d93a25ecSSatish Balay      self.addMakeMacro('CUDAC_FLAGS',self.setCompilers.getCompilerFlags())
1667ff2890cSSatish Balay      self.setCompilers.popLanguage()
1677ff2890cSSatish Balay
168f8833479SBarry Smith    # shared library linker values
169f8833479SBarry Smith    self.setCompilers.pushLanguage(self.languages.clanguage)
170f8833479SBarry Smith    # need to fix BuildSystem to collect these separately
171f8833479SBarry Smith    self.addMakeMacro('SL_LINKER',self.setCompilers.getLinker())
17270db8aa6SSatish Balay    self.addMakeMacro('SL_LINKER_FLAGS','${PCC_LINKER_FLAGS}')
173f8833479SBarry Smith    self.setCompilers.popLanguage()
174f8833479SBarry Smith    # One of 'a', 'so', 'lib', 'dll', 'dylib' (perhaps others also?) depending on the library generator and architecture
175f8833479SBarry Smith    # Note: . is not included in this macro, consistent with AR_LIB_SUFFIX
176f8833479SBarry Smith    if self.setCompilers.sharedLibraryExt == self.setCompilers.AR_LIB_SUFFIX:
177f8833479SBarry Smith      self.addMakeMacro('SL_LINKER_SUFFIX', '')
17846bc77b6SBarry Smith      self.addDefine('SLSUFFIX','""')
179f8833479SBarry Smith    else:
180f8833479SBarry Smith      self.addMakeMacro('SL_LINKER_SUFFIX', self.setCompilers.sharedLibraryExt)
18146bc77b6SBarry Smith      self.addDefine('SLSUFFIX','"'+self.setCompilers.sharedLibraryExt+'"')
182bb82cf9cSSatish Balay
18323e93537SBarry Smith    self.addMakeMacro('SL_LINKER_LIBS','${PETSC_EXTERNAL_LIB_BASIC}')
184bb82cf9cSSatish Balay
185f8833479SBarry Smith#-----------------------------------------------------------------------------------------------------
186f8833479SBarry Smith
187f8833479SBarry Smith    # CONLY or CPP. We should change the PETSc makefiles to do this better
188f8833479SBarry Smith    if self.languages.clanguage == 'C': lang = 'CONLY'
189f8833479SBarry Smith    else: lang = 'CXXONLY'
190f8833479SBarry Smith    self.addMakeMacro('PETSC_LANGUAGE',lang)
191f8833479SBarry Smith
192f8833479SBarry Smith    # real or complex
193f8833479SBarry Smith    self.addMakeMacro('PETSC_SCALAR',self.scalartypes.scalartype)
194f8833479SBarry Smith    # double or float
195f8833479SBarry Smith    self.addMakeMacro('PETSC_PRECISION',self.scalartypes.precision)
196f8833479SBarry Smith
197f8833479SBarry Smith    if self.framework.argDB['with-batch']:
198f8833479SBarry Smith      self.addMakeMacro('PETSC_WITH_BATCH','1')
199f8833479SBarry Smith
200f8833479SBarry Smith    # Test for compiler-specific macros that need to be defined.
201b409243cSBarry Smith    if self.setCompilers.isCrayVector('CC'):
202b409243cSBarry Smith      self.addDefine('HAVE_CRAY_VECTOR','1')
203f8833479SBarry Smith
204f8833479SBarry Smith#-----------------------------------------------------------------------------------------------------
205f8833479SBarry Smith    if self.functions.haveFunction('gethostbyname') and self.functions.haveFunction('socket'):
206f8833479SBarry Smith      self.addDefine('USE_SOCKET_VIEWER','1')
207f8833479SBarry Smith
208f8833479SBarry Smith#-----------------------------------------------------------------------------------------------------
209a6cc6bb1SBarry Smith    # print include and lib for makefiles
210f8833479SBarry Smith    self.framework.packages.reverse()
211a6cc6bb1SBarry Smith    includes = [os.path.join(self.petscdir.dir,'include'),os.path.join(self.petscdir.dir,self.arch.arch,'include')]
212996b3231SBarry Smith    libs = []
213f8833479SBarry Smith    for i in self.framework.packages:
214898a086dSBarry Smith      if i.useddirectly:
215898a086dSBarry Smith        self.addDefine('HAVE_'+i.PACKAGE, 1)  # ONLY list package if it is used directly by PETSc (and not only by another package)
216f8833479SBarry Smith      if not isinstance(i.lib, list):
217f8833479SBarry Smith        i.lib = [i.lib]
218ac9e4c42SSatish Balay      libs.extend(i.lib)
21934cdeb2aSSatish Balay      self.addMakeMacro(i.PACKAGE+'_LIB', self.libraries.toStringNoDupes(i.lib))
220f8833479SBarry Smith      if hasattr(i,'include'):
221f8833479SBarry Smith        if not isinstance(i.include,list):
222f8833479SBarry Smith          i.include = [i.include]
223ac9e4c42SSatish Balay        includes.extend(i.include)
22434cdeb2aSSatish Balay        self.addMakeMacro(i.PACKAGE+'_INCLUDE',self.headers.toStringNoDupes(i.include))
2252df986feSBarry Smith    if self.framework.argDB['with-single-library']:
226ea820d49SSatish 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)
22723e93537SBarry 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)
22823e93537SBarry 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
229a6cc6bb1SBarry Smith    self.addMakeMacro('PETSC_CC_INCLUDES',self.headers.toStringNoDupes(includes))
230a6cc6bb1SBarry Smith    self.PETSC_CC_INCLUDES = self.headers.toStringNoDupes(includes)
231cbd5cc15SBarry Smith    if hasattr(self.compilers, 'FC'):
232208c3fd5SBarry Smith      if self.compilers.fortranIsF90:
23343a63bfbSSatish Balay        self.addMakeMacro('PETSC_FC_INCLUDES',self.headers.toStringNoDupes(includes,includes))
23430d43657SSatish Balay      else:
23530d43657SSatish Balay        self.addMakeMacro('PETSC_FC_INCLUDES',self.headers.toStringNoDupes(includes))
236f8833479SBarry Smith
2371b1e03dfSSatish Balay    self.addMakeMacro('DESTDIR',self.installdir)
238c6f99f23SBarry Smith    self.addDefine('LIB_DIR','"'+os.path.join(self.installdir,'lib')+'"')
239f8833479SBarry Smith
2400f3b21c2SBarry Smith    if self.framework.argDB['with-single-library']:
2410f3b21c2SBarry Smith      # overrides the values set in conf/variables
2420f3b21c2SBarry Smith      self.addMakeMacro('LIBNAME','${INSTALL_LIB_DIR}/libpetsc.${AR_LIB_SUFFIX}')
24357cb31baSSatish Balay      self.addMakeMacro('SHLIBS','libpetsc')
244bccf1c12SBarry Smith      self.addMakeMacro('PETSC_LIB_BASIC','-lpetsc')
245797063a9SSatish Balay      self.addMakeMacro('PETSC_KSP_LIB_BASIC','-lpetsc')
246797063a9SSatish Balay      self.addMakeMacro('PETSC_TS_LIB_BASIC','-lpetsc')
247bb84e0fdSBarry Smith      self.addDefine('USE_SINGLE_LIBRARY', '1')
2482df986feSBarry Smith      if self.sharedlibraries.useShared:
249ea820d49SSatish Balay        self.addMakeMacro('PETSC_SYS_LIB','${C_SH_LIB_PATH} ${PETSC_WITH_EXTERNAL_LIB}')
250ea820d49SSatish Balay        self.addMakeMacro('PETSC_VEC_LIB','${C_SH_LIB_PATH} ${PETSC_WITH_EXTERNAL_LIB}')
251ea820d49SSatish Balay        self.addMakeMacro('PETSC_MAT_LIB','${C_SH_LIB_PATH} ${PETSC_WITH_EXTERNAL_LIB}')
252ea820d49SSatish Balay        self.addMakeMacro('PETSC_DM_LIB','${C_SH_LIB_PATH} ${PETSC_WITH_EXTERNAL_LIB}')
253ea820d49SSatish Balay        self.addMakeMacro('PETSC_KSP_LIB','${C_SH_LIB_PATH} ${PETSC_WITH_EXTERNAL_LIB}')
254ea820d49SSatish Balay        self.addMakeMacro('PETSC_SNES_LIB','${C_SH_LIB_PATH} ${PETSC_WITH_EXTERNAL_LIB}')
255ea820d49SSatish Balay        self.addMakeMacro('PETSC_TS_LIB','${C_SH_LIB_PATH} ${PETSC_WITH_EXTERNAL_LIB}')
256fdb87e33SJed Brown        self.addMakeMacro('PETSC_CHARACTERISTIC_LIB','${C_SH_LIB_PATH} ${PETSC_WITH_EXTERNAL_LIB}')
257ea820d49SSatish Balay        self.addMakeMacro('PETSC_LIB','${C_SH_LIB_PATH} ${PETSC_WITH_EXTERNAL_LIB}')
258ea820d49SSatish Balay        self.addMakeMacro('PETSC_CONTRIB','${C_SH_LIB_PATH} ${PETSC_WITH_EXTERNAL_LIB}')
2592df986feSBarry Smith      else:
260ea820d49SSatish Balay        self.addMakeMacro('PETSC_SYS_LIB','${PETSC_WITH_EXTERNAL_LIB}')
261ea820d49SSatish Balay        self.addMakeMacro('PETSC_VEC_LIB','${PETSC_WITH_EXTERNAL_LIB}')
262ea820d49SSatish Balay        self.addMakeMacro('PETSC_MAT_LIB','${PETSC_WITH_EXTERNAL_LIB}')
263ea820d49SSatish Balay        self.addMakeMacro('PETSC_DM_LIB','${PETSC_WITH_EXTERNAL_LIB}')
264ea820d49SSatish Balay        self.addMakeMacro('PETSC_KSP_LIB','${PETSC_WITH_EXTERNAL_LIB}')
265ea820d49SSatish Balay        self.addMakeMacro('PETSC_SNES_LIB','${PETSC_WITH_EXTERNAL_LIB}')
266ea820d49SSatish Balay        self.addMakeMacro('PETSC_TS_LIB','${PETSC_WITH_EXTERNAL_LIB}')
267fdb87e33SJed Brown        self.addMakeMacro('PETSC_CHARACTERISTIC_LIB','${PETSC_WITH_EXTERNAL_LIB}')
268ea820d49SSatish Balay        self.addMakeMacro('PETSC_LIB','${PETSC_WITH_EXTERNAL_LIB}')
269ea820d49SSatish Balay        self.addMakeMacro('PETSC_CONTRIB','${PETSC_WITH_EXTERNAL_LIB}')
2700f3b21c2SBarry Smith
271f8833479SBarry Smith    if not os.path.exists(os.path.join(self.petscdir.dir,self.arch.arch,'lib')):
272f8833479SBarry Smith      os.makedirs(os.path.join(self.petscdir.dir,self.arch.arch,'lib'))
273f8833479SBarry Smith
274f8833479SBarry Smith    # add a makefile entry for configure options
275f8833479SBarry Smith    self.addMakeMacro('CONFIGURE_OPTIONS', self.framework.getOptionsString(['configModules', 'optionsModule']).replace('\"','\\"'))
276f8833479SBarry Smith    return
277f8833479SBarry Smith
278f8833479SBarry Smith  def dumpConfigInfo(self):
279f8833479SBarry Smith    import time
280f8833479SBarry Smith    fd = file(os.path.join(self.arch.arch,'include','petscconfiginfo.h'),'w')
281f8833479SBarry Smith    fd.write('static const char *petscconfigureruntime = "'+time.ctime(time.time())+'";\n')
282f8833479SBarry Smith    fd.write('static const char *petscconfigureoptions = "'+self.framework.getOptionsString(['configModules', 'optionsModule']).replace('\"','\\"')+'";\n')
283f8833479SBarry Smith    fd.close()
284f8833479SBarry Smith    return
285f8833479SBarry Smith
2862a4161d9SMatthew G Knepley  def dumpMachineInfo(self):
2872a4161d9SMatthew G Knepley    import platform
2882a4161d9SMatthew G Knepley    import time
28940373944SSatish Balay    import script
2902a4161d9SMatthew G Knepley    fd = file(os.path.join(self.arch.arch,'include','petscmachineinfo.h'),'w')
2912a4161d9SMatthew G Knepley    fd.write('static const char *petscmachineinfo = \"\\n\"\n')
2922a4161d9SMatthew G Knepley    fd.write('\"-----------------------------------------\\n\"\n')
2932a4161d9SMatthew G Knepley    fd.write('\"Libraries compiled on %s on %s \\n\"\n' % (time.ctime(time.time()), platform.node()))
29460acdfe7SSatish Balay    fd.write('\"Machine characteristics: %s\\n\"\n' % (platform.platform()))
29560acdfe7SSatish Balay    fd.write('\"Using PETSc directory: %s\\n\"\n' % (self.petscdir.dir))
29660acdfe7SSatish Balay    fd.write('\"Using PETSc arch: %s\\n\"\n' % (self.arch.arch))
297cdec380aSBarry Smith    fd.write('\"-----------------------------------------\\n\";\n')
2982a4161d9SMatthew G Knepley    fd.write('static const char *petsccompilerinfo = \"\\n\"\n')
2992a4161d9SMatthew G Knepley    self.setCompilers.pushLanguage(self.languages.clanguage)
30060acdfe7SSatish Balay    fd.write('\"Using C compiler: %s %s ${COPTFLAGS} ${CFLAGS}\\n\"\n' % (self.setCompilers.getCompiler(), self.setCompilers.getCompilerFlags()))
3012a4161d9SMatthew G Knepley    self.setCompilers.popLanguage()
3028782282cSMatthew G Knepley    if hasattr(self.compilers, 'FC'):
3032a4161d9SMatthew G Knepley      self.setCompilers.pushLanguage('FC')
30460acdfe7SSatish Balay      fd.write('\"Using Fortran compiler: %s %s ${FOPTFLAGS} ${FFLAGS} %s\\n\"\n' % (self.setCompilers.getCompiler(), self.setCompilers.getCompilerFlags(), self.setCompilers.CPPFLAGS))
3052a4161d9SMatthew G Knepley      self.setCompilers.popLanguage()
306cdec380aSBarry Smith    fd.write('\"-----------------------------------------\\n\";\n')
3072a4161d9SMatthew G Knepley    fd.write('static const char *petsccompilerflagsinfo = \"\\n\"\n')
308a6cc6bb1SBarry Smith    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))
309cdec380aSBarry Smith    fd.write('\"-----------------------------------------\\n\";\n')
3102a4161d9SMatthew G Knepley    fd.write('static const char *petsclinkerinfo = \"\\n\"\n')
3112a4161d9SMatthew G Knepley    self.setCompilers.pushLanguage(self.languages.clanguage)
31260acdfe7SSatish Balay    fd.write('\"Using C linker: %s\\n\"\n' % (self.setCompilers.getLinker()))
3132a4161d9SMatthew G Knepley    self.setCompilers.popLanguage()
3148782282cSMatthew G Knepley    if hasattr(self.compilers, 'FC'):
3152a4161d9SMatthew G Knepley      self.setCompilers.pushLanguage('FC')
31660acdfe7SSatish Balay      fd.write('\"Using Fortran linker: %s\\n\"\n' % (self.setCompilers.getLinker()))
3172a4161d9SMatthew G Knepley      self.setCompilers.popLanguage()
31823e93537SBarry Smith    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'), '-lpetscts -lpetscsnes -lpetscksp -lpetscdm -lpetscmat -lpetscvec -lpetscsys', self.PETSC_EXTERNAL_LIB_BASIC))
319cdec380aSBarry Smith    fd.write('\"-----------------------------------------\\n\";\n')
3202a4161d9SMatthew G Knepley    fd.close()
3212a4161d9SMatthew G Knepley    return
322b2843cf1SBarry Smith
323511a6afcSJed Brown  def dumpCMakeConfig(self):
324511a6afcSJed Brown    '''
32534ed7027SBarry Smith    Writes configuration-specific values to ${PETSC_ARCH}/conf/PETScConfig.cmake.
326511a6afcSJed Brown    This file is private to PETSc and should not be included by third parties
327511a6afcSJed Brown    (a suitable file can be produced later by CMake, but this is not it).
328511a6afcSJed Brown    '''
329511a6afcSJed Brown    def cmakeset(fd,key,val=True):
330511a6afcSJed Brown      if val == True: val = 'YES'
331511a6afcSJed Brown      if val == False: val = 'NO'
332511a6afcSJed Brown      fd.write('set (' + key + ' ' + val + ')\n')
333511a6afcSJed Brown    def ensurelist(a):
334826d9344SJed Brown      if isinstance(a,list):
335826d9344SJed Brown        return a
336826d9344SJed Brown      else:
337826d9344SJed Brown        return [a]
338511a6afcSJed Brown    def libpath(lib):
339511a6afcSJed Brown      'Returns a search path if that is what this item provides, else "" which will be cleaned out later'
340511a6afcSJed Brown      if lib.startswith('-L'): return lib[2:]
341511a6afcSJed Brown      if lib.startswith('-R'): return lib[2:]
342511a6afcSJed Brown      if lib.startswith('-Wl,-rpath,'):
343511a6afcSJed Brown        # This case occurs when an external package needs a specific system library that is normally provided by the compiler.
344511a6afcSJed Brown        # In other words, the -L path is builtin to the wrapper or compiler, here we provide it so that CMake can locate the
345511a6afcSJed Brown        # corresponding library.
346511a6afcSJed Brown        return lib[len('-Wl,-rpath,'):]
347511a6afcSJed Brown      if lib.startswith('-'): return ''
348511a6afcSJed Brown      return os.path.dirname(lib)
349511a6afcSJed Brown    def cleanlib(lib):
350511a6afcSJed Brown      'Returns a library name if that is what this item provides, else "" which will be cleaned out later'
351511a6afcSJed Brown      if lib.startswith('-l'):  return lib[2:]
352511a6afcSJed Brown      if lib.startswith('-Wl') or lib.startswith('-L'): return ''
353511a6afcSJed Brown      lib = os.path.splitext(os.path.basename(lib))[0]
354511a6afcSJed Brown      if lib.startswith('lib'): return lib[3:]
355511a6afcSJed Brown      return lib
356511a6afcSJed Brown    def nub(lst):
357511a6afcSJed Brown      unique = []
358511a6afcSJed Brown      for elem in lst:
359511a6afcSJed Brown        if elem not in unique and elem != '':
360511a6afcSJed Brown          unique.append(elem)
361511a6afcSJed Brown      return unique
362511a6afcSJed Brown    def cmakeexpand(varname):
363511a6afcSJed Brown      return r'"${' + varname + r'}"'
364582751aaSJed Brown    def uniqextend(lst,new):
365511a6afcSJed Brown      for x in ensurelist(new):
366582751aaSJed Brown        if x not in lst:
367582751aaSJed Brown          lst.append(x)
368511a6afcSJed Brown    def notstandardinclude(path):
369511a6afcSJed Brown      return path not in '/usr/include /usr/local/include'.split()
370511a6afcSJed Brown    def writeMacroDefinitions(fd):
371511a6afcSJed Brown      if self.mpi.usingMPIUni:
372511a6afcSJed Brown        cmakeset(fd,'PETSC_HAVE_MPIUNI')
373511a6afcSJed Brown      for pkg in self.framework.packages:
374511a6afcSJed Brown        if pkg.useddirectly:
375511a6afcSJed Brown          cmakeset(fd,'PETSC_HAVE_' + pkg.PACKAGE)
376511a6afcSJed Brown      for name,val in self.functions.defines.items():
377511a6afcSJed Brown        cmakeset(fd,'PETSC_'+name,val)
378511a6afcSJed Brown      for dct in [self.defines, self.libraryoptions.defines]:
379511a6afcSJed Brown        for k,v in dct.items():
380511a6afcSJed Brown          if k.startswith('USE_'):
381511a6afcSJed Brown            cmakeset(fd,'PETSC_' + k, v)
382511a6afcSJed Brown      cmakeset(fd,'PETSC_USE_COMPLEX', self.scalartypes.scalartype == 'complex')
383*ce63c4c1SBarry Smith      cmakeset(fd,'PETSC_USE_REAL_' + self.scalartypes.precision.upper())
384511a6afcSJed Brown      cmakeset(fd,'PETSC_CLANGUAGE_'+self.languages.clanguage)
385511a6afcSJed Brown      if hasattr(self.compilers, 'FC'):
386511a6afcSJed Brown        cmakeset(fd,'PETSC_HAVE_FORTRAN')
387511a6afcSJed Brown        if self.compilers.fortranIsF90:
388511a6afcSJed Brown          cmakeset(fd,'PETSC_USING_F90')
389511a6afcSJed Brown      if self.sharedlibraries.useShared:
390511a6afcSJed Brown        cmakeset(fd,'BUILD_SHARED_LIBS')
391511a6afcSJed Brown    def writeBuildFlags(fd):
392511a6afcSJed Brown      lib_paths = []
393511a6afcSJed Brown      lib_libs  = []
394511a6afcSJed Brown      includes  = []
395511a6afcSJed Brown      libvars   = []
396511a6afcSJed Brown      for pkg in self.framework.packages:
397511a6afcSJed Brown        libs = ensurelist(pkg.lib)
398511a6afcSJed Brown        lib_paths.extend(map(libpath,libs))
399511a6afcSJed Brown        lib_libs.extend(map(cleanlib,libs))
400511a6afcSJed Brown        uniqextend(includes,pkg.include)
40190dd2285SSatish Balay      if self.libraries.math: lib_libs.extend(map(cleanlib,self.libraries.math))
40290dd2285SSatish Balay      if self.libraries.rt: lib_libs.extend(map(cleanlib,self.libraries.rt))
403511a6afcSJed Brown      for libname in nub(lib_libs):
404511a6afcSJed Brown        libvar = 'PETSC_' + libname.upper() + '_LIB'
4054c0032a9SSatish Balay        addpath = ''
4064c0032a9SSatish Balay        for lpath in nub(lib_paths):
4074c0032a9SSatish Balay          addpath += '"' + str(lpath) + '" '
4084c0032a9SSatish Balay        fd.write('find_library (' + libvar + ' ' + libname + ' HINTS ' + addpath + ')\n')
409511a6afcSJed Brown        libvars.append(libvar)
410511a6afcSJed Brown      fd.write('mark_as_advanced (' + ' '.join(libvars) + ')\n')
411511a6afcSJed Brown      fd.write('set (PETSC_PACKAGE_LIBS ' + ' '.join(map(cmakeexpand,libvars)) + ')\n')
412511a6afcSJed Brown      fd.write('set (PETSC_PACKAGE_INCLUDES ' + ' '.join(map(lambda i: '"'+i+'"',filter(notstandardinclude,includes))) + ')\n')
413511a6afcSJed Brown    fd = open(os.path.join(self.arch.arch,'conf','PETScConfig.cmake'), 'w')
414511a6afcSJed Brown    writeMacroDefinitions(fd)
415511a6afcSJed Brown    writeBuildFlags(fd)
416511a6afcSJed Brown    fd.close()
417511a6afcSJed Brown    return
418511a6afcSJed Brown
4198b0282a9SJed Brown  def dumpCMakeLists(self):
4208b0282a9SJed Brown    import sys
4218b0282a9SJed Brown    if sys.version_info >= (2,5):
4228b0282a9SJed Brown      import cmakegen
4238b0282a9SJed Brown      try:
4248b0282a9SJed Brown        cmakegen.main(self.petscdir.dir)
4258b0282a9SJed Brown      except (OSError), e:
4268b0282a9SJed Brown        self.framework.logPrint('Generating CMakeLists.txt failed:\n' + str(e))
4278b0282a9SJed Brown
4288b0282a9SJed Brown  def cmakeBoot(self):
4298b0282a9SJed Brown    import sys
4300b7111d2SJed Brown    if sys.version_info >= (2,5) and hasattr(self.cmake,'cmake'):
431356464bcSMatthew G Knepley      try:
4328b0282a9SJed Brown        import cmakeboot
4337c9cf364SJed Brown        cmakeboot.main(petscdir=self.petscdir.dir,petscarch=self.arch.arch,argDB=self.argDB,framework=self.framework,log=self.framework.log)
4348b0282a9SJed Brown      except (OSError), e:
4358b0282a9SJed Brown        self.framework.logPrint('Booting CMake in PETSC_ARCH failed:\n' + str(e))
436356464bcSMatthew G Knepley      except (ImportError, KeyError), e:
437356464bcSMatthew G Knepley        self.framework.logPrint('Importing cmakeboot failed:\n' + str(e))
438356464bcSMatthew G Knepley    return
4398b0282a9SJed Brown
440b2843cf1SBarry Smith  def configurePrefetch(self):
441b2843cf1SBarry Smith    '''Sees if there are any prefetch functions supported'''
4423649974fSBarry Smith    if config.setCompilers.Configure.isSolaris() or self.framework.argDB['with-iphone'] or self.framework.argDB['with-cuda']:
44393f78423SSatish Balay      self.addDefine('Prefetch(a,b,c)', ' ')
44493f78423SSatish Balay      return
445ec284106SBarry Smith    self.pushLanguage(self.languages.clanguage)
44610699583SJed Brown    if self.checkLink('#include <xmmintrin.h>', 'void *v = 0;_mm_prefetch((const char*)v,_MM_HINT_NTA);\n'):
44750d8bf02SJed Brown      # The Intel Intrinsics manual [1] specifies the prototype
44850d8bf02SJed Brown      #
44950d8bf02SJed Brown      #   void _mm_prefetch(char const *a, int sel);
45050d8bf02SJed Brown      #
45150d8bf02SJed Brown      # but other vendors seem to insist on using subtly different
45250d8bf02SJed Brown      # prototypes, including void* for the pointer, and an enum for
45350d8bf02SJed Brown      # sel.  These are both reasonable changes, but negatively impact
45450d8bf02SJed Brown      # portability.
45550d8bf02SJed Brown      #
45650d8bf02SJed Brown      # [1] http://software.intel.com/file/6373
45750d8bf02SJed Brown      self.addDefine('HAVE_XMMINTRIN_H', 1)
45850d8bf02SJed Brown      self.addDefine('Prefetch(a,b,c)', '_mm_prefetch((const char*)(a),(c))')
45950d8bf02SJed Brown      self.addDefine('PREFETCH_HINT_NTA', '_MM_HINT_NTA')
46050d8bf02SJed Brown      self.addDefine('PREFETCH_HINT_T0',  '_MM_HINT_T0')
46150d8bf02SJed Brown      self.addDefine('PREFETCH_HINT_T1',  '_MM_HINT_T1')
46250d8bf02SJed Brown      self.addDefine('PREFETCH_HINT_T2',  '_MM_HINT_T2')
46350d8bf02SJed Brown    elif self.checkLink('#include <xmmintrin.h>', 'void *v = 0;_mm_prefetch(v,_MM_HINT_NTA);\n'):
46450d8bf02SJed Brown      self.addDefine('HAVE_XMMINTRIN_H', 1)
46550d8bf02SJed Brown      self.addDefine('Prefetch(a,b,c)', '_mm_prefetch((const void*)(a),(c))')
46650d8bf02SJed Brown      self.addDefine('PREFETCH_HINT_NTA', '_MM_HINT_NTA')
46750d8bf02SJed Brown      self.addDefine('PREFETCH_HINT_T0',  '_MM_HINT_T0')
46850d8bf02SJed Brown      self.addDefine('PREFETCH_HINT_T1',  '_MM_HINT_T1')
46950d8bf02SJed Brown      self.addDefine('PREFETCH_HINT_T2',  '_MM_HINT_T2')
47010699583SJed Brown    elif self.checkLink('', 'void *v = 0;__builtin_prefetch(v,0,0);\n'):
47110699583SJed Brown      # From GCC docs: void __builtin_prefetch(const void *addr,int rw,int locality)
47210699583SJed Brown      #
47310699583SJed Brown      #   The value of rw is a compile-time constant one or zero; one
47410699583SJed Brown      #   means that the prefetch is preparing for a write to the memory
47510699583SJed Brown      #   address and zero, the default, means that the prefetch is
47610699583SJed Brown      #   preparing for a read. The value locality must be a compile-time
47710699583SJed Brown      #   constant integer between zero and three. A value of zero means
47810699583SJed Brown      #   that the data has no temporal locality, so it need not be left
47910699583SJed Brown      #   in the cache after the access. A value of three means that the
48010699583SJed Brown      #   data has a high degree of temporal locality and should be left
48110699583SJed Brown      #   in all levels of cache possible. Values of one and two mean,
48210699583SJed Brown      #   respectively, a low or moderate degree of temporal locality.
48310699583SJed Brown      #
48410699583SJed Brown      # Here we adopt Intel's x86/x86-64 naming scheme for the locality
48510699583SJed Brown      # hints.  Using macros for these values in necessary since some
48610699583SJed Brown      # compilers require an enum.
48710699583SJed Brown      self.addDefine('Prefetch(a,b,c)', '__builtin_prefetch((a),(b),(c))')
48810699583SJed Brown      self.addDefine('PREFETCH_HINT_NTA', '0')
48910699583SJed Brown      self.addDefine('PREFETCH_HINT_T0',  '3')
49010699583SJed Brown      self.addDefine('PREFETCH_HINT_T1',  '2')
49110699583SJed Brown      self.addDefine('PREFETCH_HINT_T2',  '1')
492b2843cf1SBarry Smith    else:
493b2843cf1SBarry Smith      self.addDefine('Prefetch(a,b,c)', ' ')
4947d490b44SBarry Smith    self.popLanguage()
495b2843cf1SBarry Smith
4962400fdedSBarry Smith  def configureUnused(self):
4972400fdedSBarry Smith    '''Sees if __attribute((unused)) is supported'''
4982400fdedSBarry Smith    if self.framework.argDB['with-iphone'] or self.framework.argDB['with-cuda']:
4992400fdedSBarry Smith      self.addDefine('UNUSED', ' ')
5002400fdedSBarry Smith      return
5012400fdedSBarry Smith    self.pushLanguage(self.languages.clanguage)
5022400fdedSBarry Smith    if self.checkLink('__attribute((unused)) static int myfunc(void){ return 1;}', 'int i = myfunc();\n'):
5032400fdedSBarry Smith      self.addDefine('UNUSED', '__attribute((unused))')
5042400fdedSBarry Smith    else:
5052400fdedSBarry Smith      self.addDefine('UNUSED', ' ')
5062400fdedSBarry Smith    self.popLanguage()
5072400fdedSBarry Smith
5089800092aSJed Brown  def configureExpect(self):
5099800092aSJed Brown    '''Sees if the __builtin_expect directive is supported'''
5109800092aSJed Brown    self.pushLanguage(self.languages.clanguage)
5119800092aSJed Brown    if self.checkLink('', 'if (__builtin_expect(0,1)) return 1;'):
5129800092aSJed Brown      self.addDefine('HAVE_BUILTIN_EXPECT', 1)
5139800092aSJed Brown    self.popLanguage()
5149800092aSJed Brown
51553c77d0aSJed Brown  def configureFunctionName(self):
5161ec50b02SJed Brown    '''Sees if the compiler supports __func__ or a variant.  Falls back
5171ec50b02SJed Brown    on __FUNCT__ which PETSc source defines, but most users do not, thus
5181ec50b02SJed Brown    stack traces through user code are better when the compiler's
5191ec50b02SJed Brown    variant is used.'''
5201ec50b02SJed Brown    def getFunctionName(lang):
5211ec50b02SJed Brown      name = '__FUNCT__'
5221ec50b02SJed Brown      self.pushLanguage(lang)
52353c77d0aSJed Brown      if self.checkLink('', "if (__func__[0] != 'm') return 1;"):
5241ec50b02SJed Brown        name = '__func__'
52553c77d0aSJed Brown      elif self.checkLink('', "if (__FUNCTION__[0] != 'm') return 1;"):
5261ec50b02SJed Brown        name = '__FUNCTION__'
5271ec50b02SJed Brown      self.popLanguage()
5281ec50b02SJed Brown      return name
5291ec50b02SJed Brown    langs = []
530628773c9SSatish Balay
531628773c9SSatish Balay    self.addDefine('FUNCTION_NAME_C', getFunctionName('C'))
532628773c9SSatish Balay    if hasattr(self.compilers, 'CXX'):
533628773c9SSatish Balay      self.addDefine('FUNCTION_NAME_CXX', getFunctionName('Cxx'))
53412607bf0SSatish Balay    else:
53512607bf0SSatish Balay      self.addDefine('FUNCTION_NAME_CXX', '__FUNCT__')
53653c77d0aSJed Brown
537753ebd1dSJed Brown  def configureIntptrt(self):
538753ebd1dSJed Brown    '''Determine what to use for uintptr_t'''
539753ebd1dSJed Brown    def staticAssertSizeMatchesVoidStar(inc,typename):
540753ebd1dSJed Brown      # The declaration is an error if either array size is negative.
541753ebd1dSJed 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
542d26187a0SJed Brown      return self.checkCompile(inc, ('#define STATIC_ASSERT(cond) char negative_length_if_false[2*(!!(cond))-1]\n'
543979939cdSSatish Balay                                     + 'STATIC_ASSERT(sizeof(void*) == sizeof(%s));'%typename))
544753ebd1dSJed Brown    self.pushLanguage(self.languages.clanguage)
545753ebd1dSJed Brown    if self.checkCompile('#include <stdint.h>', 'int x; uintptr_t i = (uintptr_t)&x;'):
546753ebd1dSJed Brown      self.addDefine('UINTPTR_T', 'uintptr_t')
547753ebd1dSJed Brown    elif staticAssertSizeMatchesVoidStar('','unsigned long long'):
548753ebd1dSJed Brown      self.addDefine('UINTPTR_T', 'unsigned long long')
549753ebd1dSJed Brown    elif staticAssertSizeMatchesVoidStar('#include <stdlib.h>','size_t') or staticAssertSizeMatchesVoidStar('#include <string.h>', 'size_t'):
550753ebd1dSJed Brown      self.addDefine('UINTPTR_T', 'size_t')
551c82284b1SJed Brown    elif staticAssertSizeMatchesVoidStar('','unsigned long'):
552c82284b1SJed Brown      self.addDefine('UINTPTR_T', 'unsigned long')
5532d1b7972SSatish Balay    elif staticAssertSizeMatchesVoidStar('','unsigned'):
554753ebd1dSJed Brown      self.addDefine('UINTPTR_T', 'unsigned')
555d26187a0SJed Brown    else:
556d26187a0SJed Brown      raise RuntimeError('Could not find any unsigned integer type matching void*')
557753ebd1dSJed Brown    self.popLanguage()
558753ebd1dSJed Brown
559f8833479SBarry Smith  def configureInline(self):
560f8833479SBarry Smith    '''Get a generic inline keyword, depending on the language'''
561f8833479SBarry Smith    if self.languages.clanguage == 'C':
562f8833479SBarry Smith      self.addDefine('STATIC_INLINE', self.compilers.cStaticInlineKeyword)
563f8833479SBarry Smith      self.addDefine('RESTRICT', self.compilers.cRestrict)
564f8833479SBarry Smith    elif self.languages.clanguage == 'Cxx':
565f8833479SBarry Smith      self.addDefine('STATIC_INLINE', self.compilers.cxxStaticInlineKeyword)
566f8833479SBarry Smith      self.addDefine('RESTRICT', self.compilers.cxxRestrict)
567f8833479SBarry Smith    return
568f8833479SBarry Smith
569f8833479SBarry Smith  def configureSolaris(self):
570f8833479SBarry Smith    '''Solaris specific stuff'''
571f8833479SBarry Smith    if os.path.isdir(os.path.join('/usr','ucblib')):
572f8833479SBarry Smith      try:
573f8833479SBarry Smith        flag = getattr(self.setCompilers, self.language[-1]+'SharedLinkerFlag')
574f8833479SBarry Smith      except AttributeError:
575f8833479SBarry Smith        flag = None
576f8833479SBarry Smith      if flag is None:
577f8833479SBarry Smith        self.compilers.LIBS += ' -L/usr/ucblib'
578f8833479SBarry Smith      else:
579f8833479SBarry Smith        self.compilers.LIBS += ' '+flag+'/usr/ucblib'
580f8833479SBarry Smith    return
581f8833479SBarry Smith
582f8833479SBarry Smith  def configureLinux(self):
583f8833479SBarry Smith    '''Linux specific stuff'''
5849f15855cSMatthew G Knepley    # TODO: Test for this by mallocing an odd number of floats and checking the address
585f8833479SBarry Smith    self.addDefine('HAVE_DOUBLE_ALIGN_MALLOC', 1)
586f8833479SBarry Smith    return
587f8833479SBarry Smith
588f8833479SBarry Smith  def configureWin32(self):
589f8833479SBarry Smith    '''Win32 non-cygwin specific stuff'''
590f8833479SBarry Smith    kernel32=0
591f8833479SBarry Smith    if self.libraries.add('Kernel32.lib','GetComputerName',prototype='#include <Windows.h>', call='GetComputerName(NULL,NULL);'):
592f8833479SBarry Smith      self.addDefine('HAVE_WINDOWS_H',1)
593f8833479SBarry Smith      self.addDefine('HAVE_GETCOMPUTERNAME',1)
594f8833479SBarry Smith      kernel32=1
595f8833479SBarry Smith    elif self.libraries.add('kernel32','GetComputerName',prototype='#include <Windows.h>', call='GetComputerName(NULL,NULL);'):
596f8833479SBarry Smith      self.addDefine('HAVE_WINDOWS_H',1)
597f8833479SBarry Smith      self.addDefine('HAVE_GETCOMPUTERNAME',1)
598f8833479SBarry Smith      kernel32=1
599f8833479SBarry Smith    if kernel32:
600eed94e11SSatish Balay      if self.framework.argDB['with-windows-graphics']:
601eed94e11SSatish Balay        self.addDefine('USE_WINDOWS_GRAPHICS',1)
602f8833479SBarry Smith      if self.checkLink('#include <Windows.h>','LoadLibrary(0)'):
603f8833479SBarry Smith        self.addDefine('HAVE_LOADLIBRARY',1)
604b50f6d9eSLisandro Dalcin      if self.checkLink('#include <Windows.h>','GetProcAddress(0,0)'):
605b50f6d9eSLisandro Dalcin        self.addDefine('HAVE_GETPROCADDRESS',1)
606b50f6d9eSLisandro Dalcin      if self.checkLink('#include <Windows.h>','FreeLibrary(0)'):
607b50f6d9eSLisandro Dalcin        self.addDefine('HAVE_FREELIBRARY',1)
608a21658a3SLisandro Dalcin      if self.checkLink('#include <Windows.h>','GetLastError()'):
609a21658a3SLisandro Dalcin        self.addDefine('HAVE_GETLASTERROR',1)
610a21658a3SLisandro Dalcin      if self.checkLink('#include <Windows.h>','SetLastError(0)'):
611a21658a3SLisandro Dalcin        self.addDefine('HAVE_SETLASTERROR',1)
612f8833479SBarry Smith      if self.checkLink('#include <Windows.h>\n','QueryPerformanceCounter(0);\n'):
613f8833479SBarry Smith        self.addDefine('USE_NT_TIME',1)
614f8833479SBarry Smith    if self.libraries.add('Advapi32.lib','GetUserName',prototype='#include <Windows.h>', call='GetUserName(NULL,NULL);'):
615f8833479SBarry Smith      self.addDefine('HAVE_GET_USER_NAME',1)
616f8833479SBarry Smith    elif self.libraries.add('advapi32','GetUserName',prototype='#include <Windows.h>', call='GetUserName(NULL,NULL);'):
617f8833479SBarry Smith      self.addDefine('HAVE_GET_USER_NAME',1)
618f8833479SBarry Smith
619f8833479SBarry Smith    if not self.libraries.add('User32.lib','GetDC',prototype='#include <Windows.h>',call='GetDC(0);'):
620f8833479SBarry Smith      self.libraries.add('user32','GetDC',prototype='#include <Windows.h>',call='GetDC(0);')
621f8833479SBarry Smith    if not self.libraries.add('Gdi32.lib','CreateCompatibleDC',prototype='#include <Windows.h>',call='CreateCompatibleDC(0);'):
622f8833479SBarry Smith      self.libraries.add('gdi32','CreateCompatibleDC',prototype='#include <Windows.h>',call='CreateCompatibleDC(0);')
623f8833479SBarry Smith
624f8833479SBarry Smith    self.types.check('int32_t', 'int')
625f8833479SBarry Smith    if not self.checkCompile('#include <sys/types.h>\n','uid_t u;\n'):
626f8833479SBarry Smith      self.addTypedef('int', 'uid_t')
627f8833479SBarry Smith      self.addTypedef('int', 'gid_t')
628f8833479SBarry Smith    if not self.checkLink('#if defined(PETSC_HAVE_UNISTD_H)\n#include <unistd.h>\n#endif\n','int a=R_OK;\n'):
629f8833479SBarry Smith      self.framework.addDefine('R_OK', '04')
630f8833479SBarry Smith      self.framework.addDefine('W_OK', '02')
631f8833479SBarry Smith      self.framework.addDefine('X_OK', '01')
632f8833479SBarry Smith    if not self.checkLink('#include <sys/stat.h>\n','int a=0;\nif (S_ISDIR(a)){}\n'):
633f8833479SBarry Smith      self.framework.addDefine('S_ISREG(a)', '(((a)&_S_IFMT) == _S_IFREG)')
634f8833479SBarry Smith      self.framework.addDefine('S_ISDIR(a)', '(((a)&_S_IFMT) == _S_IFDIR)')
635f8833479SBarry Smith    if self.checkCompile('#include <Windows.h>\n','LARGE_INTEGER a;\nDWORD b=a.u.HighPart;\n'):
636f8833479SBarry Smith      self.addDefine('HAVE_LARGE_INTEGER_U',1)
637f8833479SBarry Smith
638f8833479SBarry Smith    # Windows requires a Binary file creation flag when creating/opening binary files.  Is a better test in order?
63949cb358eSMatthew G Knepley    if self.checkCompile('#include <Windows.h>\n', 'int flags = O_BINARY;'):
640f8833479SBarry Smith      self.addDefine('HAVE_O_BINARY',1)
641f8833479SBarry Smith
642f8833479SBarry Smith    if self.compilers.CC.find('win32fe') >= 0:
643f8833479SBarry Smith      self.addDefine('PATH_SEPARATOR','\';\'')
644f8833479SBarry Smith      self.addDefine('DIR_SEPARATOR','\'\\\\\'')
645f8833479SBarry Smith      self.addDefine('REPLACE_DIR_SEPARATOR','\'/\'')
646f8833479SBarry Smith      self.addDefine('CANNOT_START_DEBUGGER',1)
647f8833479SBarry Smith    else:
648f8833479SBarry Smith      self.addDefine('PATH_SEPARATOR','\':\'')
649f8833479SBarry Smith      self.addDefine('REPLACE_DIR_SEPARATOR','\'\\\\\'')
650f8833479SBarry Smith      self.addDefine('DIR_SEPARATOR','\'/\'')
651f8833479SBarry Smith    return
652f8833479SBarry Smith
653f8833479SBarry Smith#-----------------------------------------------------------------------------------------------------
654569865ddSSatish Balay  def configureDefaultArch(self):
655569865ddSSatish Balay    conffile = os.path.join('conf', 'petscvariables')
656569865ddSSatish Balay    if self.framework.argDB['with-default-arch']:
657569865ddSSatish Balay      fd = file(conffile, 'w')
658569865ddSSatish Balay      fd.write('PETSC_ARCH='+self.arch.arch+'\n')
659da93591fSBarry Smith      fd.write('PETSC_DIR='+self.petscdir.dir+'\n')
660569865ddSSatish Balay      fd.write('include ${PETSC_DIR}/${PETSC_ARCH}/conf/petscvariables\n')
661569865ddSSatish Balay      fd.close()
662569865ddSSatish Balay      self.framework.actions.addArgument('PETSc', 'Build', 'Set default architecture to '+self.arch.arch+' in '+conffile)
663569865ddSSatish Balay    elif os.path.isfile(conffile):
664569865ddSSatish Balay      try:
665569865ddSSatish Balay        os.unlink(conffile)
666569865ddSSatish Balay      except:
667569865ddSSatish Balay        raise RuntimeError('Unable to remove file '+conffile+'. Did a different user create it?')
668569865ddSSatish Balay    return
669569865ddSSatish Balay
670569865ddSSatish Balay#-----------------------------------------------------------------------------------------------------
671f8833479SBarry Smith  def configureScript(self):
672f8833479SBarry Smith    '''Output a script in the conf directory which will reproduce the configuration'''
673f8833479SBarry Smith    import nargs
674495bf1a9SSatish Balay    import sys
675495bf1a9SSatish Balay    scriptName = os.path.join(self.arch.arch,'conf', 'reconfigure-'+self.arch.arch+'.py')
676f8833479SBarry Smith    args = dict([(nargs.Arg.parseArgument(arg)[0], arg) for arg in self.framework.clArgs])
677f8833479SBarry Smith    if 'configModules' in args:
6781063a081SSatish Balay      if nargs.Arg.parseArgument(args['configModules'])[1] == 'PETSc.Configure':
679f8833479SBarry Smith        del args['configModules']
680f8833479SBarry Smith    if 'optionsModule' in args:
681c1486898SMatthew Knepley      if nargs.Arg.parseArgument(args['optionsModule'])[1] == 'PETSc.compilerOptions':
682f8833479SBarry Smith        del args['optionsModule']
683f8833479SBarry Smith    if not 'PETSC_ARCH' in args:
6841063a081SSatish Balay      args['PETSC_ARCH'] = 'PETSC_ARCH='+str(self.arch.arch)
685f8833479SBarry Smith    f = file(scriptName, 'w')
686495bf1a9SSatish Balay    f.write('#!'+sys.executable+'\n')
687f8833479SBarry Smith    f.write('if __name__ == \'__main__\':\n')
688f8833479SBarry Smith    f.write('  import sys\n')
6897561c02cSSatish Balay    f.write('  import os\n')
6907561c02cSSatish Balay    f.write('  sys.path.insert(0, os.path.abspath(\'config\'))\n')
691f8833479SBarry Smith    f.write('  import configure\n')
6921063a081SSatish Balay    # pretty print repr(args.values())
6931063a081SSatish Balay    f.write('  configure_options = [\n')
6948bec23c5SJed Brown    for itm in sorted(args.values()):
6951063a081SSatish Balay      f.write('    \''+str(itm)+'\',\n')
6961063a081SSatish Balay    f.write('  ]\n')
697f8833479SBarry Smith    f.write('  configure.petsc_configure(configure_options)\n')
698f8833479SBarry Smith    f.close()
699f8833479SBarry Smith    try:
700f8833479SBarry Smith      os.chmod(scriptName, 0775)
701f8833479SBarry Smith    except OSError, e:
702f8833479SBarry Smith      self.framework.logPrint('Unable to make reconfigure script executable:\n'+str(e))
703f8833479SBarry Smith    self.framework.actions.addArgument('PETSc', 'File creation', 'Created '+scriptName+' for automatic reconfiguration')
704f8833479SBarry Smith    return
705f8833479SBarry Smith
706f8833479SBarry Smith  def configureInstall(self):
707f8833479SBarry Smith    '''Setup the directories for installation'''
708f8833479SBarry Smith    if self.framework.argDB['prefix']:
709f8833479SBarry Smith      self.installdir = self.framework.argDB['prefix']
710824e893fSSatish Balay      self.addMakeRule('shared_install','',['-@echo "Now to install the libraries do:"',\
711824e893fSSatish Balay                                              '-@echo "make PETSC_DIR=${PETSC_DIR} PETSC_ARCH=${PETSC_ARCH} install"',\
712315b77e6SSatish Balay                                              '-@echo "========================================="'])
713f8833479SBarry Smith    else:
714b559b537SSatish Balay      self.installdir = os.path.join(self.petscdir.dir,self.arch.arch)
715824e893fSSatish Balay      self.addMakeRule('shared_install','',['-@echo "Now to check if the libraries are working do:"',\
716824e893fSSatish Balay                                              '-@echo "make PETSC_DIR=${PETSC_DIR} PETSC_ARCH=${PETSC_ARCH} test"',\
717315b77e6SSatish Balay                                              '-@echo "========================================="'])
718f8833479SBarry Smith      return
719f8833479SBarry Smith
720f8833479SBarry Smith  def configureGCOV(self):
721f8833479SBarry Smith    if self.framework.argDB['with-gcov']:
722f8833479SBarry Smith      self.addDefine('USE_GCOV','1')
723f8833479SBarry Smith    return
724f8833479SBarry Smith
725f8833479SBarry Smith  def configureFortranFlush(self):
726f8833479SBarry Smith    if hasattr(self.compilers, 'FC'):
727f8833479SBarry Smith      for baseName in ['flush','flush_']:
728f8833479SBarry Smith        if self.libraries.check('', baseName, otherLibs = self.compilers.flibs, fortranMangle = 1):
729f8833479SBarry Smith          self.addDefine('HAVE_'+baseName.upper(), 1)
730f8833479SBarry Smith          return
731f8833479SBarry Smith
732f8833479SBarry Smith
733f8833479SBarry Smith  def configure(self):
734f8833479SBarry Smith    if not os.path.samefile(self.petscdir.dir, os.getcwd()):
735f8833479SBarry Smith      raise RuntimeError('Wrong PETSC_DIR option specified: '+str(self.petscdir.dir) + '\n  Configure invoked in: '+os.path.realpath(os.getcwd()))
736550489e3SMatthew 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):
7373552d8fbSSatish Balay      raise RuntimeError('Incorrect option --prefix='+self.framework.argDB['prefix']+' specified. It cannot be same as PETSC_DIR!')
738c16c35a9SSatish 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)):
739c16c35a9SSatish Balay      raise RuntimeError('Incorrect option --prefix='+self.framework.argDB['prefix']+' specified. It cannot be same as PETSC_DIR/PETSC_ARCH!')
740f16c1317SJed Brown    self.framework.header          = os.path.join(self.arch.arch,'include','petscconf.h')
741f16c1317SJed Brown    self.framework.cHeader         = os.path.join(self.arch.arch,'include','petscfix.h')
742f16c1317SJed Brown    self.framework.makeMacroHeader = os.path.join(self.arch.arch,'conf','petscvariables')
743f16c1317SJed Brown    self.framework.makeRuleHeader  = os.path.join(self.arch.arch,'conf','petscrules')
744f8833479SBarry Smith    if self.libraries.math is None:
745f8833479SBarry Smith      raise RuntimeError('PETSc requires a functional math library. Please send configure.log to petsc-maint@mcs.anl.gov.')
746f8833479SBarry Smith    if self.languages.clanguage == 'Cxx' and not hasattr(self.compilers, 'CXX'):
747f8833479SBarry Smith      raise RuntimeError('Cannot set C language to C++ without a functional C++ compiler.')
748f8833479SBarry Smith    self.executeTest(self.configureInline)
749b2843cf1SBarry Smith    self.executeTest(self.configurePrefetch)
7502400fdedSBarry Smith    self.executeTest(self.configureUnused)
7519800092aSJed Brown    self.executeTest(self.configureExpect);
75253c77d0aSJed Brown    self.executeTest(self.configureFunctionName);
753753ebd1dSJed Brown    self.executeTest(self.configureIntptrt);
754f8833479SBarry Smith    self.executeTest(self.configureSolaris)
755f8833479SBarry Smith    self.executeTest(self.configureLinux)
756f8833479SBarry Smith    self.executeTest(self.configureWin32)
757569865ddSSatish Balay    self.executeTest(self.configureDefaultArch)
758f8833479SBarry Smith    self.executeTest(self.configureScript)
759f8833479SBarry Smith    self.executeTest(self.configureInstall)
760f8833479SBarry Smith    self.executeTest(self.configureGCOV)
761f8833479SBarry Smith    self.executeTest(self.configureFortranFlush)
762f8833479SBarry Smith    # dummy rules, always needed except for remote builds
763f8833479SBarry Smith    self.addMakeRule('remote','')
764f8833479SBarry Smith    self.addMakeRule('remoteclean','')
765f8833479SBarry Smith
766f8833479SBarry Smith    self.Dump()
767f8833479SBarry Smith    self.dumpConfigInfo()
7682a4161d9SMatthew G Knepley    self.dumpMachineInfo()
769511a6afcSJed Brown    self.dumpCMakeConfig()
7708b0282a9SJed Brown    self.dumpCMakeLists()
7718b0282a9SJed Brown    self.cmakeBoot()
772f8833479SBarry Smith    self.framework.log.write('================================================================================\n')
773f8833479SBarry Smith    self.logClear()
774f8833479SBarry Smith    return
775