blob: 580eac734e02ee327dd8090b9c5f90086f173382 [file] [log] [blame]
akmhoquefa8ee9b2014-03-14 09:06:24 -05001#! /usr/bin/env python
2# encoding: utf-8
3# WARNING! Do not edit! http://waf.googlecode.com/git/docs/wafbook/single.html#_obtaining_the_waf_file
4
5import re,shutil,os,sys,string,shlex
6from waflib.Configure import conf
7from waflib.TaskGen import feature,after_method,before_method
8from waflib import Build,Utils
9FC_FRAGMENT=' program main\n end program main\n'
10FC_FRAGMENT2=' PROGRAM MAIN\n END\n'
11@conf
12def fc_flags(conf):
13 v=conf.env
14 v['FC_SRC_F']=[]
15 v['FC_TGT_F']=['-c','-o']
16 v['FCINCPATH_ST']='-I%s'
17 v['FCDEFINES_ST']='-D%s'
18 if not v['LINK_FC']:v['LINK_FC']=v['FC']
19 v['FCLNK_SRC_F']=[]
20 v['FCLNK_TGT_F']=['-o']
21 v['FCFLAGS_fcshlib']=['-fpic']
22 v['LINKFLAGS_fcshlib']=['-shared']
23 v['fcshlib_PATTERN']='lib%s.so'
24 v['fcstlib_PATTERN']='lib%s.a'
25 v['FCLIB_ST']='-l%s'
26 v['FCLIBPATH_ST']='-L%s'
27 v['FCSTLIB_ST']='-l%s'
28 v['FCSTLIBPATH_ST']='-L%s'
29 v['FCSTLIB_MARKER']='-Wl,-Bstatic'
30 v['FCSHLIB_MARKER']='-Wl,-Bdynamic'
31 v['SONAME_ST']='-Wl,-h,%s'
32@conf
33def fc_add_flags(conf):
34 conf.add_os_flags('FCFLAGS')
35 conf.add_os_flags('LDFLAGS','LINKFLAGS')
36@conf
37def check_fortran(self,*k,**kw):
38 self.check_cc(fragment=FC_FRAGMENT,compile_filename='test.f',features='fc fcprogram',msg='Compiling a simple fortran app')
39@conf
40def check_fc(self,*k,**kw):
41 kw['compiler']='fc'
42 if not'compile_mode'in kw:
43 kw['compile_mode']='fc'
44 if not'type'in kw:
45 kw['type']='fcprogram'
46 if not'compile_filename'in kw:
47 kw['compile_filename']='test.f90'
48 if not'code'in kw:
49 kw['code']=FC_FRAGMENT
50 return self.check(*k,**kw)
51@conf
52def fortran_modifier_darwin(conf):
53 v=conf.env
54 v['FCFLAGS_fcshlib']=['-fPIC']
55 v['LINKFLAGS_fcshlib']=['-dynamiclib','-Wl,-compatibility_version,1','-Wl,-current_version,1']
56 v['fcshlib_PATTERN']='lib%s.dylib'
57 v['FRAMEWORKPATH_ST']='-F%s'
58 v['FRAMEWORK_ST']='-framework %s'
59 v['LINKFLAGS_fcstlib']=[]
60 v['FCSHLIB_MARKER']=''
61 v['FCSTLIB_MARKER']=''
62 v['SONAME_ST']=''
63@conf
64def fortran_modifier_win32(conf):
65 v=conf.env
66 v['fcprogram_PATTERN']=v['fcprogram_test_PATTERN']='%s.exe'
67 v['fcshlib_PATTERN']='%s.dll'
68 v['implib_PATTERN']='lib%s.dll.a'
69 v['IMPLIB_ST']='-Wl,--out-implib,%s'
70 v['FCFLAGS_fcshlib']=[]
71 v.append_value('FCFLAGS_fcshlib',['-DDLL_EXPORT'])
72 v.append_value('LINKFLAGS',['-Wl,--enable-auto-import'])
73@conf
74def fortran_modifier_cygwin(conf):
75 fortran_modifier_win32(conf)
76 v=conf.env
77 v['fcshlib_PATTERN']='cyg%s.dll'
78 v.append_value('LINKFLAGS_fcshlib',['-Wl,--enable-auto-image-base'])
79 v['FCFLAGS_fcshlib']=[]
80@conf
81def check_fortran_dummy_main(self,*k,**kw):
82 if not self.env.CC:
83 self.fatal('A c compiler is required for check_fortran_dummy_main')
84 lst=['MAIN__','__MAIN','_MAIN','MAIN_','MAIN']
85 lst.extend([m.lower()for m in lst])
86 lst.append('')
87 self.start_msg('Detecting whether we need a dummy main')
88 for main in lst:
89 kw['fortran_main']=main
90 try:
91 self.check_cc(fragment='int %s() { return 0; }\n'%(main or'test'),features='c fcprogram',mandatory=True)
92 if not main:
93 self.env.FC_MAIN=-1
94 self.end_msg('no')
95 else:
96 self.env.FC_MAIN=main
97 self.end_msg('yes %s'%main)
98 break
99 except self.errors.ConfigurationError:
100 pass
101 else:
102 self.end_msg('not found')
103 self.fatal('could not detect whether fortran requires a dummy main, see the config.log')
104GCC_DRIVER_LINE=re.compile('^Driving:')
105POSIX_STATIC_EXT=re.compile('\S+\.a')
106POSIX_LIB_FLAGS=re.compile('-l\S+')
107@conf
108def is_link_verbose(self,txt):
109 assert isinstance(txt,str)
110 for line in txt.splitlines():
111 if not GCC_DRIVER_LINE.search(line):
112 if POSIX_STATIC_EXT.search(line)or POSIX_LIB_FLAGS.search(line):
113 return True
114 return False
115@conf
116def check_fortran_verbose_flag(self,*k,**kw):
117 self.start_msg('fortran link verbose flag')
118 for x in['-v','--verbose','-verbose','-V']:
119 try:
120 self.check_cc(features='fc fcprogram_test',fragment=FC_FRAGMENT2,compile_filename='test.f',linkflags=[x],mandatory=True)
121 except self.errors.ConfigurationError:
122 pass
123 else:
124 if self.is_link_verbose(self.test_bld.err)or self.is_link_verbose(self.test_bld.out):
125 self.end_msg(x)
126 break
127 else:
128 self.end_msg('failure')
129 self.fatal('Could not obtain the fortran link verbose flag (see config.log)')
130 self.env.FC_VERBOSE_FLAG=x
131 return x
132LINKFLAGS_IGNORED=[r'-lang*',r'-lcrt[a-zA-Z0-9\.]*\.o',r'-lc$',r'-lSystem',r'-libmil',r'-LIST:*',r'-LNO:*']
133if os.name=='nt':
134 LINKFLAGS_IGNORED.extend([r'-lfrt*',r'-luser32',r'-lkernel32',r'-ladvapi32',r'-lmsvcrt',r'-lshell32',r'-lmingw',r'-lmoldname'])
135else:
136 LINKFLAGS_IGNORED.append(r'-lgcc*')
137RLINKFLAGS_IGNORED=[re.compile(f)for f in LINKFLAGS_IGNORED]
138def _match_ignore(line):
139 for i in RLINKFLAGS_IGNORED:
140 if i.match(line):
141 return True
142 return False
143def parse_fortran_link(lines):
144 final_flags=[]
145 for line in lines:
146 if not GCC_DRIVER_LINE.match(line):
147 _parse_flink_line(line,final_flags)
148 return final_flags
149SPACE_OPTS=re.compile('^-[LRuYz]$')
150NOSPACE_OPTS=re.compile('^-[RL]')
151def _parse_flink_line(line,final_flags):
152 lexer=shlex.shlex(line,posix=True)
153 lexer.whitespace_split=True
154 t=lexer.get_token()
155 tmp_flags=[]
156 while t:
157 def parse(token):
158 if _match_ignore(token):
159 pass
160 elif token.startswith('-lkernel32')and sys.platform=='cygwin':
161 tmp_flags.append(token)
162 elif SPACE_OPTS.match(token):
163 t=lexer.get_token()
164 if t.startswith('P,'):
165 t=t[2:]
166 for opt in t.split(os.pathsep):
167 tmp_flags.append('-L%s'%opt)
168 elif NOSPACE_OPTS.match(token):
169 tmp_flags.append(token)
170 elif POSIX_LIB_FLAGS.match(token):
171 tmp_flags.append(token)
172 else:
173 pass
174 t=lexer.get_token()
175 return t
176 t=parse(t)
177 final_flags.extend(tmp_flags)
178 return final_flags
179@conf
180def check_fortran_clib(self,autoadd=True,*k,**kw):
181 if not self.env.FC_VERBOSE_FLAG:
182 self.fatal('env.FC_VERBOSE_FLAG is not set: execute check_fortran_verbose_flag?')
183 self.start_msg('Getting fortran runtime link flags')
184 try:
185 self.check_cc(fragment=FC_FRAGMENT2,compile_filename='test.f',features='fc fcprogram_test',linkflags=[self.env.FC_VERBOSE_FLAG])
186 except Exception:
187 self.end_msg(False)
188 if kw.get('mandatory',True):
189 conf.fatal('Could not find the c library flags')
190 else:
191 out=self.test_bld.err
192 flags=parse_fortran_link(out.splitlines())
193 self.end_msg('ok (%s)'%' '.join(flags))
194 self.env.LINKFLAGS_CLIB=flags
195 return flags
196 return[]
197def getoutput(conf,cmd,stdin=False):
198 if stdin:
199 stdin=Utils.subprocess.PIPE
200 else:
201 stdin=None
202 env=conf.env.env or None
203 try:
204 p=Utils.subprocess.Popen(cmd,stdin=stdin,stdout=Utils.subprocess.PIPE,stderr=Utils.subprocess.PIPE,env=env)
205 if stdin:
206 p.stdin.write('\n')
207 out,err=p.communicate()
208 except Exception:
209 conf.fatal('could not determine the compiler version %r'%cmd)
210 if not isinstance(out,str):
211 out=out.decode(sys.stdout.encoding or'iso8859-1')
212 if not isinstance(err,str):
213 err=err.decode(sys.stdout.encoding or'iso8859-1')
214 return(out,err)
215ROUTINES_CODE="""\
216 subroutine foobar()
217 return
218 end
219 subroutine foo_bar()
220 return
221 end
222"""
223MAIN_CODE="""
224void %(dummy_func_nounder)s(void);
225void %(dummy_func_under)s(void);
226int %(main_func_name)s() {
227 %(dummy_func_nounder)s();
228 %(dummy_func_under)s();
229 return 0;
230}
231"""
232@feature('link_main_routines_func')
233@before_method('process_source')
234def link_main_routines_tg_method(self):
235 def write_test_file(task):
236 task.outputs[0].write(task.generator.code)
237 bld=self.bld
238 bld(rule=write_test_file,target='main.c',code=MAIN_CODE%self.__dict__)
239 bld(rule=write_test_file,target='test.f',code=ROUTINES_CODE)
240 bld(features='fc fcstlib',source='test.f',target='test')
241 bld(features='c fcprogram',source='main.c',target='app',use='test')
242def mangling_schemes():
243 for u in['_','']:
244 for du in['','_']:
245 for c in["lower","upper"]:
246 yield(u,du,c)
247def mangle_name(u,du,c,name):
248 return getattr(name,c)()+u+(name.find('_')!=-1 and du or'')
249@conf
250def check_fortran_mangling(self,*k,**kw):
251 if not self.env.CC:
252 self.fatal('A c compiler is required for link_main_routines')
253 if not self.env.FC:
254 self.fatal('A fortran compiler is required for link_main_routines')
255 if not self.env.FC_MAIN:
256 self.fatal('Checking for mangling requires self.env.FC_MAIN (execute "check_fortran_dummy_main" first?)')
257 self.start_msg('Getting fortran mangling scheme')
258 for(u,du,c)in mangling_schemes():
259 try:
260 self.check_cc(compile_filename=[],features='link_main_routines_func',msg='nomsg',errmsg='nomsg',mandatory=True,dummy_func_nounder=mangle_name(u,du,c,"foobar"),dummy_func_under=mangle_name(u,du,c,"foo_bar"),main_func_name=self.env.FC_MAIN)
261 except self.errors.ConfigurationError:
262 pass
263 else:
264 self.end_msg("ok ('%s', '%s', '%s-case')"%(u,du,c))
265 self.env.FORTRAN_MANGLING=(u,du,c)
266 break
267 else:
268 self.end_msg(False)
269 self.fatal('mangler not found')
270 return(u,du,c)
271@feature('pyext')
272@before_method('propagate_uselib_vars','apply_link')
273def set_lib_pat(self):
274 self.env['fcshlib_PATTERN']=self.env['pyext_PATTERN']
275@conf
276def detect_openmp(self):
277 for x in['-fopenmp','-openmp','-mp','-xopenmp','-omp','-qsmp=omp']:
278 try:
279 self.check_fc(msg='Checking for OpenMP flag %s'%x,fragment='program main\n call omp_get_num_threads()\nend program main',fcflags=x,linkflags=x,uselib_store='OPENMP')
280 except self.errors.ConfigurationError:
281 pass
282 else:
283 break
284 else:
285 self.fatal('Could not find OpenMP')