mirror of
https://github.com/samba-team/samba.git
synced 2024-12-27 03:21:53 +03:00
306 lines
8.8 KiB
Python
306 lines
8.8 KiB
Python
# a set of config tests that use the samba_autoconf functions
|
|
# to test for commonly needed configuration options
|
|
|
|
import os, Build, shutil, Utils, re
|
|
from Configure import conf
|
|
from samba_utils import *
|
|
|
|
@conf
|
|
def CHECK_ICONV(conf, define='HAVE_NATIVE_ICONV'):
|
|
'''check if the iconv library is installed
|
|
optionally pass a define'''
|
|
if conf.CHECK_FUNCS_IN('iconv_open', 'iconv', checklibc=True, headers='iconv.h'):
|
|
conf.DEFINE(define, 1)
|
|
return True
|
|
return False
|
|
|
|
|
|
@conf
|
|
def CHECK_LARGEFILE(conf, define='HAVE_LARGEFILE'):
|
|
'''see what we need for largefile support'''
|
|
if conf.CHECK_CODE('return !(sizeof(off_t) >= 8)',
|
|
define,
|
|
execute=True,
|
|
msg='Checking for large file support'):
|
|
return True
|
|
if conf.CHECK_CODE('return !(sizeof(off_t) >= 8)',
|
|
define,
|
|
execute=True,
|
|
cflags='-D_FILE_OFFSET_BITS=64',
|
|
msg='Checking for -D_FILE_OFFSET_BITS=64'):
|
|
conf.DEFINE('_FILE_OFFSET_BITS', 64)
|
|
return True
|
|
return False
|
|
|
|
|
|
@conf
|
|
def CHECK_C_PROTOTYPE(conf, function, prototype, define, headers=None):
|
|
'''verify that a C prototype matches the one on the current system'''
|
|
if not conf.CHECK_DECLS(function, headers=headers):
|
|
return False
|
|
return conf.CHECK_CODE('%s; void *_x = (void *)%s' % (prototype, function),
|
|
define=define,
|
|
local_include=False,
|
|
headers=headers,
|
|
link=False,
|
|
execute=False,
|
|
msg='Checking C prototype for %s' % function)
|
|
|
|
|
|
@conf
|
|
def CHECK_CHARSET_EXISTS(conf, charset, outcharset='UCS-2LE', headers=None, define=None):
|
|
'''check that a named charset is able to be used with iconv_open() for conversion
|
|
to a target charset
|
|
'''
|
|
msg = 'Checking if can we convert from %s to %s' % (charset, outcharset)
|
|
if define is None:
|
|
define = 'HAVE_CHARSET_%s' % charset.upper().replace('-','_')
|
|
return conf.CHECK_CODE('''
|
|
iconv_t cd = iconv_open("%s", "%s");
|
|
if (cd == 0 || cd == (iconv_t)-1) return -1;
|
|
''' % (charset, outcharset),
|
|
define=define,
|
|
execute=True,
|
|
msg=msg,
|
|
lib='iconv',
|
|
headers=headers)
|
|
|
|
def find_config_dir(conf):
|
|
'''find a directory to run tests in'''
|
|
k = 0
|
|
while k < 10000:
|
|
dir = os.path.join(conf.blddir, '.conf_check_%d' % k)
|
|
try:
|
|
shutil.rmtree(dir)
|
|
except OSError:
|
|
pass
|
|
try:
|
|
os.stat(dir)
|
|
except:
|
|
break
|
|
k += 1
|
|
|
|
try:
|
|
os.makedirs(dir)
|
|
except:
|
|
conf.fatal('cannot create a configuration test folder %r' % dir)
|
|
|
|
try:
|
|
os.stat(dir)
|
|
except:
|
|
conf.fatal('cannot use the configuration test folder %r' % dir)
|
|
return dir
|
|
|
|
|
|
# this one is quite complex, and should probably be broken up
|
|
# into several parts. I'd quite like to create a set of CHECK_COMPOUND()
|
|
# functions that make writing complex compound tests like this much easier
|
|
@conf
|
|
def CHECK_LIBRARY_SUPPORT(conf, rpath=False, msg=None):
|
|
'''see if the platform supports building libraries'''
|
|
|
|
if msg is None:
|
|
if rpath:
|
|
msg = "rpath library support"
|
|
else:
|
|
msg = "building library support"
|
|
|
|
dir = find_config_dir(conf)
|
|
|
|
bdir = os.path.join(dir, 'testbuild')
|
|
if not os.path.exists(bdir):
|
|
os.makedirs(bdir)
|
|
|
|
env = conf.env
|
|
|
|
subdir = os.path.join(dir, "libdir")
|
|
|
|
os.makedirs(subdir)
|
|
|
|
dest = open(os.path.join(subdir, 'lib1.c'), 'w')
|
|
dest.write('int lib_func(void) { return 42; }\n')
|
|
dest.close()
|
|
|
|
dest = open(os.path.join(dir, 'main.c'), 'w')
|
|
dest.write('int main(void) {return !(lib_func() == 42);}\n')
|
|
dest.close()
|
|
|
|
bld = Build.BuildContext()
|
|
bld.log = conf.log
|
|
bld.all_envs.update(conf.all_envs)
|
|
bld.all_envs['default'] = env
|
|
bld.lst_variants = bld.all_envs.keys()
|
|
bld.load_dirs(dir, bdir)
|
|
|
|
bld.rescan(bld.srcnode)
|
|
|
|
bld(features='cc cshlib',
|
|
source='libdir/lib1.c',
|
|
target='libdir/lib1',
|
|
name='lib1')
|
|
|
|
o = bld(features='cc cprogram',
|
|
source='main.c',
|
|
target='prog1',
|
|
uselib_local='lib1')
|
|
|
|
if rpath:
|
|
o.rpath=os.path.join(bdir, 'default/libdir')
|
|
|
|
# compile the program
|
|
try:
|
|
bld.compile()
|
|
except:
|
|
conf.check_message(msg, '', False)
|
|
return False
|
|
|
|
# path for execution
|
|
lastprog = o.link_task.outputs[0].abspath(env)
|
|
|
|
if not rpath:
|
|
if 'LD_LIBRARY_PATH' in os.environ:
|
|
old_ld_library_path = os.environ['LD_LIBRARY_PATH']
|
|
else:
|
|
old_ld_library_path = None
|
|
ADD_LD_LIBRARY_PATH(os.path.join(bdir, 'default/libdir'))
|
|
|
|
# we need to run the program, try to get its result
|
|
args = conf.SAMBA_CROSS_ARGS(msg=msg)
|
|
proc = Utils.pproc.Popen([lastprog] + args, stdout=Utils.pproc.PIPE, stderr=Utils.pproc.PIPE)
|
|
(out, err) = proc.communicate()
|
|
w = conf.log.write
|
|
w(str(out))
|
|
w('\n')
|
|
w(str(err))
|
|
w('\nreturncode %r\n' % proc.returncode)
|
|
ret = (proc.returncode == 0)
|
|
|
|
if not rpath:
|
|
os.environ['LD_LIBRARY_PATH'] = old_ld_library_path or ''
|
|
|
|
conf.check_message(msg, '', ret)
|
|
return ret
|
|
|
|
|
|
|
|
@conf
|
|
def CHECK_PERL_MANPAGE(conf, msg=None, section=None):
|
|
'''work out what extension perl uses for manpages'''
|
|
|
|
if msg is None:
|
|
if section:
|
|
msg = "perl man%s extension" % section
|
|
else:
|
|
msg = "perl manpage generation"
|
|
|
|
conf.check_message_1(msg)
|
|
|
|
dir = find_config_dir(conf)
|
|
|
|
bdir = os.path.join(dir, 'testbuild')
|
|
if not os.path.exists(bdir):
|
|
os.makedirs(bdir)
|
|
|
|
dest = open(os.path.join(bdir, 'Makefile.PL'), 'w')
|
|
dest.write("""
|
|
use ExtUtils::MakeMaker;
|
|
WriteMakefile(
|
|
'NAME' => 'WafTest',
|
|
'EXE_FILES' => [ 'WafTest' ]
|
|
);
|
|
""")
|
|
dest.close()
|
|
back = os.path.abspath('.')
|
|
os.chdir(bdir)
|
|
proc = Utils.pproc.Popen(['perl', 'Makefile.PL'],
|
|
stdout=Utils.pproc.PIPE,
|
|
stderr=Utils.pproc.PIPE)
|
|
(out, err) = proc.communicate()
|
|
os.chdir(back)
|
|
|
|
ret = (proc.returncode == 0)
|
|
if not ret:
|
|
conf.check_message_2('not found', color='YELLOW')
|
|
return
|
|
|
|
if section:
|
|
f = open(os.path.join(bdir,'Makefile'), 'r')
|
|
man = f.read()
|
|
f.close()
|
|
m = re.search('MAN%sEXT\s+=\s+(\w+)' % section, man)
|
|
if not m:
|
|
conf.check_message_2('not found', color='YELLOW')
|
|
return
|
|
ext = m.group(1)
|
|
conf.check_message_2(ext)
|
|
return ext
|
|
|
|
conf.check_message_2('ok')
|
|
return True
|
|
|
|
|
|
@conf
|
|
def CHECK_COMMAND(conf, cmd, msg=None, define=None, on_target=True):
|
|
'''run a command and return result'''
|
|
if msg is None:
|
|
msg = 'Checking %s' % ' '.join(cmd)
|
|
conf.COMPOUND_START(msg)
|
|
cmd = cmd[:]
|
|
if on_target:
|
|
cmd.extend(conf.SAMBA_CROSS_ARGS(msg=msg))
|
|
try:
|
|
ret = Utils.cmd_output(cmd)
|
|
except:
|
|
conf.COMPOUND_END(False)
|
|
return False
|
|
ret = ret.strip()
|
|
conf.COMPOUND_END(ret)
|
|
if define:
|
|
conf.DEFINE(define, ret, quote=True)
|
|
return ret
|
|
|
|
|
|
@conf
|
|
def CHECK_UNAME(conf):
|
|
'''setup SYSTEM_UNAME_* defines'''
|
|
ret = True
|
|
for v in "sysname machine release version".split():
|
|
if not conf.CHECK_CODE('''
|
|
struct utsname n;
|
|
if (uname(&n) == -1) return -1;
|
|
printf("%%s", n.%s);
|
|
''' % v,
|
|
define='SYSTEM_UNAME_%s' % v.upper(),
|
|
execute=True,
|
|
define_ret=True,
|
|
quote=True,
|
|
headers='sys/utsname.h',
|
|
local_include=False,
|
|
msg="Checking uname %s type" % v):
|
|
ret = False
|
|
return ret
|
|
|
|
@conf
|
|
def CHECK_INLINE(conf):
|
|
'''check for the right value for inline'''
|
|
conf.COMPOUND_START('Checking for inline')
|
|
for i in ['inline', '__inline__', '__inline']:
|
|
ret = conf.CHECK_CODE('''
|
|
typedef int foo_t;
|
|
static %s foo_t static_foo () {return 0; }
|
|
%s foo_t foo () {return 0; }''' % (i, i),
|
|
define='INLINE_MACRO',
|
|
addmain=False,
|
|
link=False)
|
|
if ret:
|
|
if i != 'inline':
|
|
conf.DEFINE('inline', i, quote=False)
|
|
break
|
|
if not ret:
|
|
conf.COMPOUND_END(ret)
|
|
else:
|
|
conf.COMPOUND_END(i)
|
|
return ret
|
|
|
|
|