Wednesday, September 26, 2012

How to add a New Module to ns-3 and enhance package with additional features


-->
When you have created a group of related classes, examples, and tests, they can be combined together into an ns-3 module so that they can be used with existing ns-3 modules and by other researchers.

-->

Here is link to the ns-3 documentation, which includes a tutorial: Ns-3 documentation
Maybe this can help you: manual ns3.13
 
This chapter walks you through the steps necessary to add a new module to ns-3.
Step 1 - Familiarize yourself with the module layout
All modules can be found in the src directory. Each module can be found in a directory that has the same name as the module. For example, the spectrum module can be found here:
src/spectrum
A prototypical module has the following directory structure and required files:
src/
module-name/
bindings/
doc/
examples/
wscript
helper/
model/
test/
examples-to-run.py
wscript
Not all directories will be present in each module.
Step 2 - Create your new module based on the template module
A python program is provided in the source directory that will create a skeleton for a new module
src/create-module.py
For the purposes of this discussion we will assume that your new module is called “new-module”. From the src directory, do the following to create the new module:
./create-module.py new-module
Next, cd into new-module; you will find this directory layout:
examples helper model test wscript
We next walk through how to customize this module. All ns-3 modules depend on the ‘core’ module and usually on other modules. This dependency is specified in the wscript file. Let’s assume that ‘new-module’ depends on the internet, mobility, and aodv modules. Then the call to the function that will create this module should look like this before editing:
def build(bld):
module = bld.create_ns3_module('new-module', ['core'])
and after editing:
def build(bld):
module = bld.create_ns3_module('new-module', ['internet', 'mobility', 'aodv'])
Your module will most likely have model source files. Initial skeletons (which will compile successfully) are created in model/new-module.cc andmodel/new-module.h.
If your module will have helper source files, then they will go into the helper/ directory; again, initial skeletons are created in that directory.
Finally, it is good practice to write tests. A skeleton test suite and test case is created in the test/ directory. The below constructor specifies that it will be a unit test named ‘new-module’:
New-moduleTestSuite::New-moduleTestSuite ()
: TestSuite ("new-module", UNIT)
{
AddTestCase (new New-moduleTestCase1);
}
Step 3 - Adding to your module’s source files
If your new module has model and/or helper source files, then they must be specified in your
src/new-module/wscript
file by modifying it with your text editor.
As an example, the source files for the spectrum module are specified in
src/spectrum/wscript
with the following list of source files:
module.source = [
'model/spectrum-model.cc',
'model/spectrum-value.cc',
.
.
.
'model/microwave-oven-spectrum-value-helper.cc',
'helper/spectrum-helper.cc',
'helper/adhoc-aloha-noack-ideal-phy-helper.cc',
'helper/waveform-generator-helper.cc',
'helper/spectrum-analyzer-helper.cc',
]
Step 4 - Specify your module’s header files
If your new module has model and/or helper header files, then they must be specified in your
src/new-module/wscript
file by modifying it with your text editor.
As an example, the header files for the spectrum module are specified in
src/spectrum/wscript
with the following function call, module name, and list of header files. Note that the argument for the function new_task_gen() tells waf to install this module’s headers with the other ns-3 headers:
headers = bld.new_task_gen(features=['ns3header'])

headers.module = 'spectrum'

headers.source = [
'model/spectrum-model.h',
'model/spectrum-value.h',
.
.
.
'model/microwave-oven-spectrum-value-helper.h',
'helper/spectrum-helper.h',
'helper/adhoc-aloha-noack-ideal-phy-helper.h',
'helper/waveform-generator-helper.h',
'helper/spectrum-analyzer-helper.h',
]
Step 5 - Specify your module’s tests
If your new module has tests, then they must be specified in your
src/new-module/wscript
file by modifying it with your text editor.
As an example, the tests for the spectrum module are specified in
src/spectrum/wscript
with the following function call and list of test suites:
module_test = bld.create_ns3_module_test_library('spectrum')

module_test.source = [
'test/spectrum-interference-test.cc',
'test/spectrum-value-test.cc',
]
Step 6 - Specify your module’s examples
If your new module has examples, then they must be specified in your
src/new-module/examples/wscript
file by modifying it with your text editor.
As an example, the examples for the core module are specified in
src/core/examples/wscript
The core module’s C++ examples are specified using the following function calls and source file names. Note that the second argument for the function create_ns3_program() is the list of modules that the program being created depends on:
obj = bld.create_ns3_program('main-callback', ['core'])
obj.source = 'main-callback.cc'

obj = bld.create_ns3_program('sample-simulator', ['core'])
obj.source = 'sample-simulator.cc'
The core module’s Python examples are specified using the following function call. Note that the second argument for the function register_ns3_script() is the list of modules that the Python example depends on:
bld.register_ns3_script('sample-simulator.py', ['core'])
Step 7 - Specify which of your module’s examples should be run as tests
The test framework can also be instrumented to run example programs to try to catch regressions in the examples. However, not all examples are suitable for regression tests. A file called examples-to-run.py that exists in each module’s test directory can control the invocation of the examples when the test framework runs.
As an example, the examples that are run by test.py for the core module are specified in
src/core/test/examples-to-run.py
using the following two lists of C++ and Python examples:
# A list of C++ examples to run in order to ensure that they remain
# buildable and runnable over time. Each tuple in the list contains
#
# (example_name, do_run, do_valgrind_run).
#
# See test.py for more information.
cpp_examples = [
("main-attribute-value", "True", "True"),
("main-callback", "True", "True"),
("sample-simulator", "True", "True"),
("main-ptr", "True", "True"),
("main-random-variable", "True", "True"),
("sample-random-variable", "True", "True"),
]

# A list of Python examples to run in order to ensure that they remain
# runnable over time. Each tuple in the list contains
#
# (example_name, do_run).
#
# See test.py for more information.
python_examples = [
("sample-simulator.py", "True"),
]
Each tuple in the C++ list of examples to run contains
(example_name, do_run, do_valgrind_run)
where example_name is the executable to be run, do_run is a condition under which to run the example, and do_valgrind_run is a condition under which to run the example under valgrind. This is needed because NSC causes illegal instruction crashes with some tests when they are run under valgrind.
Note that the two conditions are Python statements that can depend on waf configuration variables. For example,
("tcp-nsc-lfn", "NSC_ENABLED == True", "NSC_ENABLED == False"),
Each tuple in the Python list of examples to run contains
(example_name, do_run)
where example_name is the Python script to be run and do_run is a condition under which to run the example.
Note that the condition is a Python statement that can depend on waf configuration variables. For example,
("realtime-udp-echo.py", "ENABLE_REAL_TIME == False"),
If your new module has examples, then you must specify which of them should be run in your
src/new-module/test/examples-to-run.py
file by modifying it with your text editor. These examples are run by test.py.
Step 8 - Build and test your new module
You can now build and test your module as normal:
./waf configure --enable-examples --enable-tests
./waf build
./test.py
and look for your new module’s test suite (and example programs, if enabled) in the test output.