2007-05-07 12:01:51 +01:00
|
|
|
## -*- Mode: python; py-indent-offset: 4; indent-tabs-mode: nil; coding: utf-8; -*-
|
|
|
|
|
import os
|
2007-05-17 17:02:03 +01:00
|
|
|
import sys
|
2007-05-23 19:20:54 +01:00
|
|
|
import shlex
|
2007-05-24 19:21:50 +01:00
|
|
|
import shutil
|
2007-05-07 12:01:51 +01:00
|
|
|
|
|
|
|
|
import Params
|
|
|
|
|
import Object
|
2007-05-17 18:22:10 +01:00
|
|
|
import pproc as subprocess
|
2007-06-21 12:26:46 +01:00
|
|
|
import optparse
|
2007-07-07 18:10:54 +01:00
|
|
|
import os.path
|
2007-05-07 12:01:51 +01:00
|
|
|
|
|
|
|
|
Params.g_autoconfig = 1
|
|
|
|
|
|
|
|
|
|
# the following two variables are used by the target "waf dist"
|
2007-05-24 17:59:30 +01:00
|
|
|
VERSION = file("VERSION").read().strip()
|
|
|
|
|
APPNAME = 'ns'
|
2007-05-07 12:01:51 +01:00
|
|
|
|
|
|
|
|
# these variables are mandatory ('/' are converted automatically)
|
|
|
|
|
srcdir = '.'
|
|
|
|
|
blddir = 'build'
|
|
|
|
|
|
2007-05-24 19:21:50 +01:00
|
|
|
def dist_hook(srcdir, blddir):
|
|
|
|
|
shutil.rmtree("doc/html")
|
|
|
|
|
shutil.rmtree("doc/latex")
|
2007-05-07 12:01:51 +01:00
|
|
|
|
|
|
|
|
def set_options(opt):
|
2007-06-21 12:26:46 +01:00
|
|
|
|
|
|
|
|
def debug_option_callback(option, opt, value, parser):
|
|
|
|
|
if value == 'debug':
|
|
|
|
|
setattr(parser.values, option.dest, 'ultradebug')
|
|
|
|
|
elif value == 'optimized':
|
|
|
|
|
setattr(parser.values, option.dest, 'optimized')
|
|
|
|
|
else:
|
|
|
|
|
raise optparse.OptionValueError("allowed --debug-level values"
|
|
|
|
|
" are debug, optimized.")
|
|
|
|
|
|
|
|
|
|
opt.add_option('-d', '--debug-level',
|
|
|
|
|
action='callback',
|
|
|
|
|
type=str, dest='debug_level', default='debug',
|
|
|
|
|
help=('Specify the debug level, does nothing if CFLAGS is set'
|
|
|
|
|
' in the environment. [Allowed Values: debug, optimized].'
|
|
|
|
|
' WARNING: this option only has effect '
|
|
|
|
|
'with the configure command.'),
|
|
|
|
|
callback=debug_option_callback)
|
|
|
|
|
|
2007-05-07 12:01:51 +01:00
|
|
|
# options provided by the modules
|
2007-05-17 14:22:27 +01:00
|
|
|
opt.tool_options('compiler_cxx')
|
2007-05-07 12:01:51 +01:00
|
|
|
|
|
|
|
|
opt.add_option('--enable-gcov',
|
2007-06-21 12:26:46 +01:00
|
|
|
help=('Enable code coverage analysis.'
|
|
|
|
|
' WARNING: this option only has effect '
|
|
|
|
|
'with the configure command.'),
|
2007-05-07 12:01:51 +01:00
|
|
|
action="store_true", default=False,
|
|
|
|
|
dest='enable_gcov')
|
|
|
|
|
|
|
|
|
|
opt.add_option('--lcov-report',
|
|
|
|
|
help=('Generate a code coverage report '
|
|
|
|
|
'(use this option at build time, not in configure)'),
|
|
|
|
|
action="store_true", default=False,
|
|
|
|
|
dest='lcov_report')
|
|
|
|
|
|
|
|
|
|
opt.add_option('--doxygen',
|
|
|
|
|
help=('Run doxygen to generate html documentation from source comments'),
|
|
|
|
|
action="store_true", default=False,
|
|
|
|
|
dest='doxygen')
|
|
|
|
|
|
2007-05-23 19:20:54 +01:00
|
|
|
opt.add_option('--run',
|
|
|
|
|
help=('Run a locally built program'),
|
|
|
|
|
type="string", default='', dest='run')
|
|
|
|
|
|
|
|
|
|
opt.add_option('--shell',
|
|
|
|
|
help=('Run a shell with an environment suitably modified to run locally built programs'),
|
|
|
|
|
action="store_true", default=False,
|
|
|
|
|
dest='shell')
|
|
|
|
|
|
2007-05-07 12:01:51 +01:00
|
|
|
# options provided in a script in a subdirectory named "src"
|
|
|
|
|
opt.sub_options('src')
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
def configure(conf):
|
2007-05-17 14:22:27 +01:00
|
|
|
if not conf.check_tool('compiler_cxx'):
|
|
|
|
|
Params.fatal("No suitable compiler found")
|
2007-05-07 12:01:51 +01:00
|
|
|
|
|
|
|
|
# create the second environment, set the variant and set its name
|
|
|
|
|
variant_env = conf.env.copy()
|
2007-06-21 12:26:46 +01:00
|
|
|
debug_level = Params.g_options.debug_level.lower()
|
|
|
|
|
if debug_level == 'ultradebug':
|
|
|
|
|
variant_name = 'debug'
|
|
|
|
|
else:
|
|
|
|
|
variant_name = debug_level
|
2007-05-07 12:01:51 +01:00
|
|
|
|
|
|
|
|
if Params.g_options.enable_gcov:
|
|
|
|
|
variant_name += '-gcov'
|
|
|
|
|
variant_env.append_value('CCFLAGS', '-fprofile-arcs')
|
|
|
|
|
variant_env.append_value('CCFLAGS', '-ftest-coverage')
|
|
|
|
|
variant_env.append_value('CXXFLAGS', '-fprofile-arcs')
|
|
|
|
|
variant_env.append_value('CXXFLAGS', '-ftest-coverage')
|
|
|
|
|
variant_env.append_value('LINKFLAGS', '-fprofile-arcs')
|
|
|
|
|
|
|
|
|
|
conf.env['NS3_ACTIVE_VARIANT'] = variant_name
|
|
|
|
|
variant_env['NS3_ACTIVE_VARIANT'] = variant_name
|
|
|
|
|
variant_env.set_variant(variant_name)
|
|
|
|
|
conf.set_env_name(variant_name, variant_env)
|
|
|
|
|
conf.setenv(variant_name)
|
|
|
|
|
|
|
|
|
|
variant_env.append_value('CXXDEFINES', 'RUN_SELF_TESTS')
|
2007-06-21 11:59:55 +01:00
|
|
|
|
|
|
|
|
if os.path.basename(conf.env['CXX']).startswith("g++"):
|
|
|
|
|
variant_env.append_value('CXXFLAGS', ['-Wall', '-Werror'])
|
|
|
|
|
|
2007-05-07 12:01:51 +01:00
|
|
|
if 'debug' in Params.g_options.debug_level.lower():
|
|
|
|
|
variant_env.append_value('CXXDEFINES', 'NS3_DEBUG_ENABLE')
|
|
|
|
|
variant_env.append_value('CXXDEFINES', 'NS3_ASSERT_ENABLE')
|
|
|
|
|
|
2007-05-17 17:02:03 +01:00
|
|
|
if sys.platform == 'win32':
|
2007-06-21 11:59:55 +01:00
|
|
|
if os.path.basename(conf.env['CXX']).startswith("g++"):
|
|
|
|
|
variant_env.append_value("LINKFLAGS", "-Wl,--enable-runtime-pseudo-reloc")
|
2007-05-17 17:02:03 +01:00
|
|
|
|
2007-05-07 12:01:51 +01:00
|
|
|
conf.sub_config('src')
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
def build(bld):
|
|
|
|
|
variant_name = bld.env_of_name('default')['NS3_ACTIVE_VARIANT']
|
|
|
|
|
variant_env = bld.env_of_name(variant_name)
|
|
|
|
|
bld.m_allenvs['default'] = variant_env # switch to the active variant
|
2007-06-12 19:04:38 +01:00
|
|
|
|
2007-07-05 14:38:39 +01:00
|
|
|
if Params.g_options.shell:
|
2007-06-12 19:04:38 +01:00
|
|
|
run_shell()
|
|
|
|
|
return
|
|
|
|
|
|
2007-05-07 12:01:51 +01:00
|
|
|
# process subfolders from here
|
|
|
|
|
bld.add_subdirs('src')
|
2007-05-13 12:46:18 +01:00
|
|
|
bld.add_subdirs('samples utils examples')
|
2007-05-07 12:01:51 +01:00
|
|
|
|
|
|
|
|
|
|
|
|
|
def shutdown():
|
2007-05-23 19:20:54 +01:00
|
|
|
#import UnitTest
|
|
|
|
|
#ut = UnitTest.unit_test()
|
|
|
|
|
#ut.change_to_testfile_dir = True
|
|
|
|
|
#ut.want_to_see_test_output = True
|
|
|
|
|
#ut.want_to_see_test_error = True
|
|
|
|
|
#ut.run()
|
2007-05-07 12:01:51 +01:00
|
|
|
#ut.print_results()
|
|
|
|
|
|
2007-05-23 19:20:54 +01:00
|
|
|
if Params.g_commands['check']:
|
|
|
|
|
run_program('run-tests')
|
|
|
|
|
|
2007-05-07 12:01:51 +01:00
|
|
|
if Params.g_options.lcov_report:
|
2007-05-23 17:32:32 +01:00
|
|
|
lcov_report()
|
2007-05-07 12:01:51 +01:00
|
|
|
|
|
|
|
|
if Params.g_options.doxygen:
|
2007-05-23 17:32:32 +01:00
|
|
|
doxygen()
|
|
|
|
|
|
2007-07-05 14:38:39 +01:00
|
|
|
if Params.g_options.run:
|
|
|
|
|
run_program(Params.g_options.run)
|
|
|
|
|
|
2007-07-07 18:10:54 +01:00
|
|
|
def _find_program(program_name, env):
|
|
|
|
|
launch_dir = os.path.abspath(Params.g_cwd_launch)
|
2007-07-05 14:44:00 +01:00
|
|
|
found_programs = []
|
2007-05-23 19:20:54 +01:00
|
|
|
for obj in Object.g_allobjs:
|
2007-07-05 14:44:00 +01:00
|
|
|
if obj.m_type != 'program' or not obj.target:
|
|
|
|
|
continue
|
2007-07-07 18:10:54 +01:00
|
|
|
|
|
|
|
|
## filter out programs not in the subtree starting at the launch dir
|
|
|
|
|
if not (obj.path.abspath().startswith(launch_dir)
|
|
|
|
|
or obj.path.abspath(env).startswith(launch_dir)):
|
|
|
|
|
continue
|
|
|
|
|
|
2007-07-05 14:44:00 +01:00
|
|
|
found_programs.append(obj.target)
|
2007-05-23 19:20:54 +01:00
|
|
|
if obj.target == program_name:
|
|
|
|
|
return obj
|
2007-07-05 14:48:21 +01:00
|
|
|
raise ValueError("program '%s' not found; available programs are: %r"
|
2007-07-05 14:44:00 +01:00
|
|
|
% (program_name, found_programs))
|
2007-05-23 19:20:54 +01:00
|
|
|
|
|
|
|
|
def _run_argv(argv):
|
|
|
|
|
env = Params.g_build.env_of_name('default')
|
|
|
|
|
if sys.platform == 'linux2':
|
|
|
|
|
pathvar = 'LD_LIBRARY_PATH'
|
|
|
|
|
pathsep = ':'
|
|
|
|
|
elif sys.platform == 'darwin':
|
|
|
|
|
pathvar = 'DYLD_LIBRARY_PATH'
|
|
|
|
|
pathsep = ':'
|
|
|
|
|
elif sys.platform == 'win32':
|
|
|
|
|
pathvar = 'PATH'
|
|
|
|
|
pathsep = ';'
|
2007-06-21 00:38:58 +01:00
|
|
|
elif sys.platform == 'cygwin':
|
|
|
|
|
pathvar = 'PATH'
|
|
|
|
|
pathsep = ':'
|
2007-05-23 19:20:54 +01:00
|
|
|
else:
|
|
|
|
|
Params.warning(("Don't know how to configure "
|
|
|
|
|
"dynamic library path for the platform '%s'") % (sys.platform,))
|
|
|
|
|
pathvar = None
|
|
|
|
|
pathsep = None
|
|
|
|
|
|
|
|
|
|
os_env = dict(os.environ)
|
|
|
|
|
if pathvar is not None:
|
|
|
|
|
if pathvar in os_env:
|
|
|
|
|
os_env[pathvar] = pathsep.join([os_env[pathvar]] + list(env['NS3_MODULE_PATH']))
|
|
|
|
|
else:
|
|
|
|
|
os_env[pathvar] = pathsep.join(list(env['NS3_MODULE_PATH']))
|
|
|
|
|
|
|
|
|
|
retval = subprocess.Popen(argv, env=os_env).wait()
|
|
|
|
|
if retval:
|
|
|
|
|
raise SystemExit(retval)
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
def run_program(program_string):
|
|
|
|
|
env = Params.g_build.env_of_name('default')
|
|
|
|
|
argv = shlex.split(program_string)
|
|
|
|
|
program_name = argv[0]
|
|
|
|
|
|
|
|
|
|
try:
|
2007-07-07 18:10:54 +01:00
|
|
|
program_obj = _find_program(program_name, env)
|
2007-07-05 14:44:00 +01:00
|
|
|
except ValueError, ex:
|
|
|
|
|
Params.fatal(str(ex))
|
2007-05-23 19:20:54 +01:00
|
|
|
|
|
|
|
|
try:
|
|
|
|
|
program_node, = program_obj.m_linktask.m_outputs
|
|
|
|
|
except AttributeError:
|
|
|
|
|
Params.fatal("%s does not appear to be a program" % (program_name,))
|
|
|
|
|
|
|
|
|
|
execvec = [program_node.abspath(env)] + argv[1:]
|
2007-07-08 12:24:22 +01:00
|
|
|
|
|
|
|
|
former_cwd = os.getcwd()
|
|
|
|
|
os.chdir(Params.g_cwd_launch)
|
|
|
|
|
try:
|
|
|
|
|
return _run_argv(execvec)
|
|
|
|
|
finally:
|
|
|
|
|
os.chdir(former_cwd)
|
2007-05-23 19:20:54 +01:00
|
|
|
|
|
|
|
|
|
|
|
|
|
def run_shell():
|
|
|
|
|
if sys.platform == 'win32':
|
|
|
|
|
shell = os.environ.get("COMSPEC", "cmd.exe")
|
|
|
|
|
else:
|
|
|
|
|
shell = os.environ.get("SHELL", "/bin/sh")
|
|
|
|
|
_run_argv([shell])
|
|
|
|
|
|
2007-05-23 17:32:32 +01:00
|
|
|
|
|
|
|
|
def doxygen():
|
|
|
|
|
doxygen_config = os.path.join('doc', 'doxygen.conf')
|
|
|
|
|
if subprocess.Popen(['doxygen', doxygen_config]).wait():
|
|
|
|
|
raise SystemExit(1)
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
def lcov_report():
|
|
|
|
|
env = Params.g_build.env_of_name('default')
|
|
|
|
|
variant_name = env['NS3_ACTIVE_VARIANT']
|
|
|
|
|
|
|
|
|
|
if 'gcov' not in variant_name:
|
|
|
|
|
Params.fatal("project not configured for code coverage;"
|
|
|
|
|
" reconfigure with --enable-gcov")
|
|
|
|
|
|
|
|
|
|
os.chdir(blddir)
|
|
|
|
|
try:
|
|
|
|
|
lcov_report_dir = os.path.join(variant_name, 'lcov-report')
|
|
|
|
|
create_dir_command = "rm -rf " + lcov_report_dir
|
|
|
|
|
create_dir_command += " && mkdir " + lcov_report_dir + ";"
|
|
|
|
|
|
|
|
|
|
if subprocess.Popen(create_dir_command, shell=True).wait():
|
|
|
|
|
raise SystemExit(1)
|
|
|
|
|
|
|
|
|
|
info_file = os.path.join(lcov_report_dir, variant_name + '.info')
|
|
|
|
|
lcov_command = "../utils/lcov/lcov -c -d . -o " + info_file
|
|
|
|
|
lcov_command += " --source-dirs=" + os.getcwd()
|
|
|
|
|
lcov_command += ":" + os.path.join(
|
|
|
|
|
os.getcwd(), variant_name, 'include')
|
|
|
|
|
if subprocess.Popen(lcov_command, shell=True).wait():
|
|
|
|
|
raise SystemExit(1)
|
|
|
|
|
|
|
|
|
|
genhtml_command = "../utils/lcov/genhtml -o " + lcov_report_dir
|
|
|
|
|
genhtml_command += " " + info_file
|
|
|
|
|
if subprocess.Popen(genhtml_command, shell=True).wait():
|
2007-05-07 12:01:51 +01:00
|
|
|
raise SystemExit(1)
|
2007-05-23 17:32:32 +01:00
|
|
|
finally:
|
|
|
|
|
os.chdir("..")
|
2007-05-07 12:01:51 +01:00
|
|
|
|