1 | #! /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 | |
---|
5 | import re,os,sys,shlex |
---|
6 | from waflib.Configure import conf |
---|
7 | from waflib.TaskGen import feature,before_method |
---|
8 | from waflib import Utils |
---|
9 | FC_FRAGMENT=' program main\n end program main\n' |
---|
10 | FC_FRAGMENT2=' PROGRAM MAIN\n END\n' |
---|
11 | @conf |
---|
12 | def 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 |
---|
33 | def fc_add_flags(conf): |
---|
34 | conf.add_os_flags('FCFLAGS') |
---|
35 | conf.add_os_flags('LDFLAGS','LINKFLAGS') |
---|
36 | @conf |
---|
37 | def 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 |
---|
40 | def 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 |
---|
52 | def 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 |
---|
64 | def 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 |
---|
74 | def 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 |
---|
81 | def 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') |
---|
104 | GCC_DRIVER_LINE=re.compile('^Driving:') |
---|
105 | POSIX_STATIC_EXT=re.compile('\S+\.a') |
---|
106 | POSIX_LIB_FLAGS=re.compile('-l\S+') |
---|
107 | @conf |
---|
108 | def 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 |
---|
116 | def 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 |
---|
132 | LINKFLAGS_IGNORED=[r'-lang*',r'-lcrt[a-zA-Z0-9\.]*\.o',r'-lc$',r'-lSystem',r'-libmil',r'-LIST:*',r'-LNO:*'] |
---|
133 | if os.name=='nt': |
---|
134 | LINKFLAGS_IGNORED.extend([r'-lfrt*',r'-luser32',r'-lkernel32',r'-ladvapi32',r'-lmsvcrt',r'-lshell32',r'-lmingw',r'-lmoldname']) |
---|
135 | else: |
---|
136 | LINKFLAGS_IGNORED.append(r'-lgcc*') |
---|
137 | RLINKFLAGS_IGNORED=[re.compile(f)for f in LINKFLAGS_IGNORED] |
---|
138 | def _match_ignore(line): |
---|
139 | for i in RLINKFLAGS_IGNORED: |
---|
140 | if i.match(line): |
---|
141 | return True |
---|
142 | return False |
---|
143 | def 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 |
---|
149 | SPACE_OPTS=re.compile('^-[LRuYz]$') |
---|
150 | NOSPACE_OPTS=re.compile('^-[RL]') |
---|
151 | def _parse_flink_token(lexer,token,tmp_flags): |
---|
152 | if _match_ignore(token): |
---|
153 | pass |
---|
154 | elif token.startswith('-lkernel32')and sys.platform=='cygwin': |
---|
155 | tmp_flags.append(token) |
---|
156 | elif SPACE_OPTS.match(token): |
---|
157 | t=lexer.get_token() |
---|
158 | if t.startswith('P,'): |
---|
159 | t=t[2:] |
---|
160 | for opt in t.split(os.pathsep): |
---|
161 | tmp_flags.append('-L%s'%opt) |
---|
162 | elif NOSPACE_OPTS.match(token): |
---|
163 | tmp_flags.append(token) |
---|
164 | elif POSIX_LIB_FLAGS.match(token): |
---|
165 | tmp_flags.append(token) |
---|
166 | else: |
---|
167 | pass |
---|
168 | t=lexer.get_token() |
---|
169 | return t |
---|
170 | def _parse_flink_line(line,final_flags): |
---|
171 | lexer=shlex.shlex(line,posix=True) |
---|
172 | lexer.whitespace_split=True |
---|
173 | t=lexer.get_token() |
---|
174 | tmp_flags=[] |
---|
175 | while t: |
---|
176 | t=_parse_flink_token(lexer,t,tmp_flags) |
---|
177 | final_flags.extend(tmp_flags) |
---|
178 | return final_flags |
---|
179 | @conf |
---|
180 | def 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[] |
---|
197 | def 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) |
---|
215 | ROUTINES_CODE="""\ |
---|
216 | subroutine foobar() |
---|
217 | return |
---|
218 | end |
---|
219 | subroutine foo_bar() |
---|
220 | return |
---|
221 | end |
---|
222 | """ |
---|
223 | MAIN_CODE=""" |
---|
224 | void %(dummy_func_nounder)s(void); |
---|
225 | void %(dummy_func_under)s(void); |
---|
226 | int %(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') |
---|
234 | def 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') |
---|
242 | def mangling_schemes(): |
---|
243 | for u in('_',''): |
---|
244 | for du in('','_'): |
---|
245 | for c in("lower","upper"): |
---|
246 | yield(u,du,c) |
---|
247 | def mangle_name(u,du,c,name): |
---|
248 | return getattr(name,c)()+u+(name.find('_')!=-1 and du or'') |
---|
249 | @conf |
---|
250 | def 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') |
---|
273 | def set_lib_pat(self): |
---|
274 | self.env['fcshlib_PATTERN']=self.env['pyext_PATTERN'] |
---|
275 | @conf |
---|
276 | def 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') |
---|