1 # a set of config tests that use the samba_autoconf functions
2 # to test for commonly needed configuration options
5 from waflib
import Build
, Configure
, Utils
, Options
, Logs
, Errors
6 from waflib
.Configure
import conf
7 from samba_utils
import TO_LIST
, ADD_LD_LIBRARY_PATH
, get_string
10 def add_option(self
, *k
, **kw
):
11 '''syntax help: provide the "match" attribute to opt.add_option() so that folders can be added to specific config tests'''
12 Options
.OptionsContext
.parser
= self
13 match
= kw
.get('match', [])
16 opt
= self
.parser
.add_option(*k
, **kw
)
19 Options
.OptionsContext
.add_option
= add_option
22 def check(self
, *k
, **kw
):
23 '''Override the waf defaults to inject --with-directory options'''
26 kw
['env'] = self
.env
.derive()
28 # match the configuration test with specific options, for example:
29 # --with-libiconv -> Options.options.iconv_open -> "Checking for library iconv"
33 for x
in Options
.OptionsContext
.parser
.parser
.option_list
:
34 if getattr(x
, 'match', None) and msg
in x
.match
:
35 d
= getattr(Options
.options
, x
.dest
, '')
37 additional_dirs
.append(d
)
39 # we add the additional dirs twice: once for the test data, and again if the compilation test succeeds below
40 def add_options_dir(dirs
, env
):
42 if not x
in env
.CPPPATH
:
43 env
.CPPPATH
= [os
.path
.join(x
, 'include')] + env
.CPPPATH
44 if not x
in env
.LIBPATH
:
45 env
.LIBPATH
= [os
.path
.join(x
, 'lib')] + env
.LIBPATH
47 add_options_dir(additional_dirs
, kw
['env'])
50 self
.start_msg(kw
['msg'])
53 ret
= self
.run_c_code(*k
, **kw
)
54 except Configure
.ConfigurationError
as e
:
55 self
.end_msg(kw
['errmsg'], 'YELLOW')
56 if 'mandatory' in kw
and kw
['mandatory']:
60 self
.fatal('the configuration failed (see %r)' % self
.log
.name
)
63 self
.end_msg(self
.ret_msg(kw
['okmsg'], kw
))
65 # success! keep the CPPPATH/LIBPATH
66 add_options_dir(additional_dirs
, self
.env
)
68 self
.post_check(*k
, **kw
)
69 if not kw
.get('execute', False):
75 def CHECK_ICONV(conf
, define
='HAVE_NATIVE_ICONV'):
76 '''check if the iconv library is installed
77 optionally pass a define'''
78 if conf
.CHECK_FUNCS_IN('iconv_open', 'iconv', checklibc
=True, headers
='iconv.h'):
79 conf
.DEFINE(define
, 1)
85 def CHECK_LARGEFILE(conf
, define
='HAVE_LARGEFILE'):
86 '''see what we need for largefile support'''
87 getconf_cflags
= conf
.CHECK_COMMAND(['getconf', 'LFS_CFLAGS'])
88 if getconf_cflags
is not False:
89 if (conf
.CHECK_CODE('if (sizeof(off_t) < 8) return 1',
90 define
='WORKING_GETCONF_LFS_CFLAGS',
92 cflags
=getconf_cflags
,
93 msg
='Checking getconf large file support flags work')):
94 conf
.ADD_CFLAGS(getconf_cflags
)
95 getconf_cflags_list
=TO_LIST(getconf_cflags
)
96 for flag
in getconf_cflags_list
:
98 flag_split
= flag
[2:].split('=')
99 if len(flag_split
) == 1:
100 conf
.DEFINE(flag_split
[0], '1')
102 conf
.DEFINE(flag_split
[0], flag_split
[1])
104 if conf
.CHECK_CODE('if (sizeof(off_t) < 8) return 1',
107 msg
='Checking for large file support without additional flags'):
110 if conf
.CHECK_CODE('if (sizeof(off_t) < 8) return 1',
113 cflags
='-D_FILE_OFFSET_BITS=64',
114 msg
='Checking for -D_FILE_OFFSET_BITS=64'):
115 conf
.DEFINE('_FILE_OFFSET_BITS', 64)
118 if conf
.CHECK_CODE('if (sizeof(off_t) < 8) return 1',
121 cflags
='-D_LARGE_FILES',
122 msg
='Checking for -D_LARGE_FILES'):
123 conf
.DEFINE('_LARGE_FILES', 1)
129 def CHECK_C_PROTOTYPE(conf
, function
, prototype
, define
, headers
=None, msg
=None, lib
=None):
130 '''verify that a C prototype matches the one on the current system'''
131 if not conf
.CHECK_DECLS(function
, headers
=headers
):
134 msg
= 'Checking C prototype for %s' % function
135 return conf
.CHECK_CODE('%s; void *_x = (void *)%s' % (prototype
, function
),
146 def CHECK_CHARSET_EXISTS(conf
, charset
, outcharset
='UCS-2LE', headers
=None, define
=None):
147 '''check that a named charset is able to be used with iconv_open() for conversion
150 msg
= 'Checking if can we convert from %s to %s' % (charset
, outcharset
)
152 define
= 'HAVE_CHARSET_%s' % charset
.upper().replace('-','_')
153 return conf
.CHECK_CODE('''
154 iconv_t cd = iconv_open("%s", "%s");
155 if (cd == 0 || cd == (iconv_t)-1) return -1;
156 ''' % (charset
, outcharset
),
163 def find_config_dir(conf
):
164 '''find a directory to run tests in'''
167 dir = os
.path
.join(conf
.bldnode
.abspath(), '.conf_check_%d' % k
)
181 conf
.fatal('cannot create a configuration test folder %r' % dir)
186 conf
.fatal('cannot use the configuration test folder %r' % dir)
190 def CHECK_SHLIB_INTRASINC_NAME_FLAGS(conf
, msg
):
192 check if the waf default flags for setting the name of lib
201 return conf
.check(features
='c cshlib',vnum
="1",fragment
=snip
,msg
=msg
, mandatory
=False)
204 def CHECK_NEED_LC(conf
, msg
):
205 '''check if we need -lc'''
207 dir = find_config_dir(conf
)
211 bdir
= os
.path
.join(dir, 'testbuild2')
212 if not os
.path
.exists(bdir
):
216 subdir
= os
.path
.join(dir, "liblctest")
220 Utils
.writef(os
.path
.join(subdir
, 'liblc1.c'), '#include <stdio.h>\nint lib_func(void) { FILE *f = fopen("foo", "r");}\n')
222 bld
= Build
.BuildContext()
224 bld
.all_envs
.update(conf
.all_envs
)
225 bld
.all_envs
['default'] = env
226 bld
.lst_variants
= bld
.all_envs
.keys()
227 bld
.load_dirs(dir, bdir
)
229 bld
.rescan(bld
.srcnode
)
231 bld(features
='c cshlib',
232 source
='liblctest/liblc1.c',
233 ldflags
=conf
.env
['EXTRA_LDFLAGS'],
239 conf
.check_message(msg
, '', True)
242 conf
.check_message(msg
, '', False)
247 def CHECK_SHLIB_W_PYTHON(conf
, msg
):
248 '''check if we need -undefined dynamic_lookup'''
250 dir = find_config_dir(conf
)
253 #include <crt_externs.h>
254 #define environ (*_NSGetEnviron())
256 static PyObject *ldb_module = NULL;
258 extern char **environ;
260 ldb_module = PyImport_ImportModule("ldb");
264 return conf
.check(features
='c cshlib',uselib
='PYEMBED',fragment
=snip
,msg
=msg
, mandatory
=False)
266 # this one is quite complex, and should probably be broken up
267 # into several parts. I'd quite like to create a set of CHECK_COMPOUND()
268 # functions that make writing complex compound tests like this much easier
270 def CHECK_LIBRARY_SUPPORT(conf
, rpath
=False, version_script
=False, msg
=None):
271 '''see if the platform supports building libraries'''
275 msg
= "rpath library support"
277 msg
= "building library support"
279 dir = find_config_dir(conf
)
281 bdir
= os
.path
.join(dir, 'testbuild')
282 if not os
.path
.exists(bdir
):
287 subdir
= os
.path
.join(dir, "libdir")
291 Utils
.writef(os
.path
.join(subdir
, 'lib1.c'), 'int lib_func(void) { return 42; }\n')
292 Utils
.writef(os
.path
.join(dir, 'main.c'),
293 'int lib_func(void);\n'
294 'int main(void) {return !(lib_func() == 42);}\n')
296 bld
= Build
.BuildContext()
298 bld
.all_envs
.update(conf
.all_envs
)
299 bld
.all_envs
['default'] = env
300 bld
.lst_variants
= bld
.all_envs
.keys()
301 bld
.load_dirs(dir, bdir
)
303 bld
.rescan(bld
.srcnode
)
307 ldflags
.append("-Wl,--version-script=%s/vscript" % bld
.path
.abspath())
308 Utils
.writef(os
.path
.join(dir,'vscript'), 'TEST_1.0A2 { global: *; };\n')
310 bld(features
='c cshlib',
311 source
='libdir/lib1.c',
312 target
='libdir/lib1',
316 o
= bld(features
='c cprogram',
322 o
.rpath
=os
.path
.join(bdir
, 'default/libdir')
324 # compile the program
328 conf
.check_message(msg
, '', False)
332 lastprog
= o
.link_task
.outputs
[0].abspath(env
)
335 if 'LD_LIBRARY_PATH' in os
.environ
:
336 old_ld_library_path
= os
.environ
['LD_LIBRARY_PATH']
338 old_ld_library_path
= None
339 ADD_LD_LIBRARY_PATH(os
.path
.join(bdir
, 'default/libdir'))
341 # we need to run the program, try to get its result
342 args
= conf
.SAMBA_CROSS_ARGS(msg
=msg
)
343 proc
= Utils
.subprocess
.Popen([lastprog
] + args
,
344 stdout
=Utils
.subprocess
.PIPE
, stderr
=Utils
.subprocess
.PIPE
)
345 (out
, err
) = proc
.communicate()
350 w('\nreturncode %r\n' % proc
.returncode
)
351 ret
= (proc
.returncode
== 0)
354 os
.environ
['LD_LIBRARY_PATH'] = old_ld_library_path
or ''
356 conf
.check_message(msg
, '', ret
)
362 def CHECK_PERL_MANPAGE(conf
, msg
=None, section
=None):
363 '''work out what extension perl uses for manpages'''
367 msg
= "perl man%s extension" % section
369 msg
= "perl manpage generation"
373 dir = find_config_dir(conf
)
375 bdir
= os
.path
.join(dir, 'testbuild')
376 if not os
.path
.exists(bdir
):
379 Utils
.writef(os
.path
.join(bdir
, 'Makefile.PL'), """
380 use ExtUtils::MakeMaker;
383 'EXE_FILES' => [ 'WafTest' ]
386 back
= os
.path
.abspath('.')
388 proc
= Utils
.subprocess
.Popen(['perl', 'Makefile.PL'],
389 stdout
=Utils
.subprocess
.PIPE
,
390 stderr
=Utils
.subprocess
.PIPE
)
391 (out
, err
) = proc
.communicate()
394 ret
= (proc
.returncode
== 0)
396 conf
.end_msg('not found', color
='YELLOW')
400 man
= Utils
.readf(os
.path
.join(bdir
,'Makefile'))
401 m
= re
.search(r
'MAN%sEXT\s+=\s+(\w+)' % section
, man
)
403 conf
.end_msg('not found', color
='YELLOW')
414 def CHECK_COMMAND(conf
, cmd
, msg
=None, define
=None, on_target
=True, boolean
=False):
415 '''run a command and return result'''
417 msg
= 'Checking %s' % ' '.join(cmd
)
418 conf
.COMPOUND_START(msg
)
421 cmd
.extend(conf
.SAMBA_CROSS_ARGS(msg
=msg
))
423 ret
= get_string(Utils
.cmd_output(cmd
))
425 conf
.COMPOUND_END(False)
428 conf
.COMPOUND_END('ok')
430 conf
.DEFINE(define
, '1')
433 conf
.COMPOUND_END(ret
)
435 conf
.DEFINE(define
, ret
, quote
=True)
440 def CHECK_UNAME(conf
):
441 '''setup SYSTEM_UNAME_* defines'''
443 for v
in "sysname machine release version".split():
444 if not conf
.CHECK_CODE('''
445 int printf(const char *format, ...);
447 if (uname(&n) == -1) return -1;
450 define
='SYSTEM_UNAME_%s' % v
.upper(),
454 headers
='sys/utsname.h',
456 msg
="Checking uname %s type" % v
):
461 def CHECK_INLINE(conf
):
462 '''check for the right value for inline'''
463 conf
.COMPOUND_START('Checking for inline')
464 for i
in ['inline', '__inline__', '__inline']:
465 ret
= conf
.CHECK_CODE('''
467 static %s foo_t static_foo () {return 0; }
468 %s foo_t foo () {return 0; }\n''' % (i
, i
),
469 define
='INLINE_MACRO',
474 conf
.DEFINE('inline', i
, quote
=False)
477 conf
.COMPOUND_END(ret
)
483 def CHECK_XSLTPROC_MANPAGES(conf
):
484 '''check if xsltproc can run with the given stylesheets'''
487 if not conf
.CONFIG_SET('XSLTPROC'):
488 conf
.find_program('xsltproc', var
='XSLTPROC')
489 if not conf
.CONFIG_SET('XSLTPROC'):
492 s
='http://docbook.sourceforge.net/release/xsl/current/manpages/docbook.xsl'
493 conf
.CHECK_COMMAND('%s --nonet %s 2> /dev/null' % (conf
.env
.get_flat('XSLTPROC'), s
),
494 msg
='Checking for stylesheet %s' % s
,
495 define
='XSLTPROC_MANPAGES', on_target
=False,
497 if not conf
.CONFIG_SET('XSLTPROC_MANPAGES'):
498 print("A local copy of the docbook.xsl wasn't found on your system" \
499 " consider installing package like docbook-xsl")
502 # Determine the standard libpath for the used compiler,
503 # so we can later use that to filter out these standard
504 # library paths when some tools like cups-config or
505 # python-config report standard lib paths with their
509 def CHECK_STANDARD_LIBPATH(conf
):
510 # at least gcc and clang support this:
512 cmd
= conf
.env
.CC
+ ['-print-search-dirs']
513 out
= get_string(Utils
.cmd_output(cmd
)).split('\n')
515 # option not supported by compiler - use a standard list of directories
516 dirlist
= [ '/usr/lib', '/usr/lib64' ]
518 raise Errors
.WafError('Unexpected error running "%s"' % (cmd
))
523 if line
.startswith("libraries: ="):
524 dirliststr
= line
[len("libraries: ="):]
525 dirlist
= [ os
.path
.normpath(x
) for x
in dirliststr
.split(':') ]
528 conf
.env
.STANDARD_LIBPATH
= dirlist