[ < ] | [ > ] | [ << ] | [ Up ] | [ >> ] | [Top] | [Contents] | [Index] | [ ? ] |
The tracing subsystem relies heavily on the ns-3
Callback and Attribute
mechanisms. You should read and understand the corresponding sections of the
manual before attempting to understand the tracing system.
The ns-3 tracing system is built on the concepts of independent tracing sources and tracing sinks; along with a uniform mechanism for connecting sources to sinks.
Trace sources are entities that can signal events that happen in a simulation and provide access to interesting underlying data. For example, a trace source could indicate when a packet is received by a net device and provide access to the packet contents for interested trace sinks. A trace source might also indicate when an interesting state change happens in a model. For example, the congestion window of a TCP model is a prime candidate for a trace source.
Trace sources are not useful by themselves; they must be connected to other pieces of code that actually do something useful with the information provided by the source. The entities that consume trace information are called trace sinks. Trace sources are generators of events and trace sinks are consumers.
This explicit division allows for large numbers of trace sources to be scattered around the system in places which model authors believe might be useful. Unless a user connects a trace sink to one of these sources, nothing is output. This arrangement allows relatively unsophisticated users to attach new types of sinks to existing tracing sources, without requiring editing and recompiling the core or models of the simulator.
There can be zero or more consumers of trace events generated by a trace source. One can think of a trace source as a kind of point-to-multipoint information link.
The “transport protocol” for this conceptual point-to-multipoint link is an
ns-3
Callback
.
Recall from the Callback Section that callback facility is a way to allow two modules in the system to communicate via function calls while at the same time decoupling the calling function from the called class completely. This is the same requirement as outlined above for the tracing system.
Basically, a trace source is a callback to which multiple
functions may be registered. When a trace sink expresses
interest in receiving trace events, it adds a callback to a list of callbacks
held by the trace source. When an interesting event happens, the trace source
invokes its operator()
providing zero or more parameters. This tells
the source to go through its list of callbacks invoking each one in turn. In
this way, the parameter(s) are communicated to the trace sinks, which are just
functions.
[ < ] | [ > ] | [ << ] | [ Up ] | [ >> ] | [Top] | [Contents] | [Index] | [ ? ] |
It will be useful to go walk a quick example just to reinforce what we’ve said.
|
The first thing to do is include the required files. As mentioned above, the
trace system makes heavy use of the Object and Attribute systems. The first
two includes bring in the declarations for those systems. The file,
traced-value.h
brings in the required declarations for tracing data
that obeys value semantics.
In general, value semantics just means that you can pass the object around, not an address. In order to use value semantics at all you have to have an object with an associated copy constructor and assignment operator available. We extend the requirements to talk about the set of operators that are pre-defined for plain-old-data (POD) types. Operator=, operator++, operator–, operator+, operator==, etc.
What this all means is that you will be able to trace changes to an object made using those operators.
|
Since the tracing system is integrated with Attributes, and Attributes work
with Objects, there must be an ns-3
Object
for the trace source
to live in. The two important lines of code are the .AddTraceSource
and
the TracedValue
declaration.
The .AddTraceSource
provides the “hooks” used for connecting the trace
source to the outside world. The TracedValue
declaration provides the
infrastructure that overloads the operators mentioned above and drives the callback
process.
|
This is the definition of the trace sink. It corresponds directly to a callback
function. This function will be called whenever one of the operators of the
TracedValue
is executed.
|
In this snippet, the first thing that needs to be done is to create the object in which the trace source lives.
The next step, the TraceConnectWithoutContext
, forms the connection
between the trace source and the trace sink. Notice the MakeCallback
template function. Recall from the Callback section that this creates the
specialized functor responsible for providing the overloaded operator()
used to “fire” the callback. The overloaded operators (++, –, etc.) will
use this operator()
to actually invoke the callback. The
TraceConnectWithoutContext
, takes a string parameter that provides
the name of the Attribute assigned to the trace source. Let’s ignore the bit
about context for now since it is not important yet.
Finally, the line,
myObject->m_myInt = 1234;
should be interpreted as an invocation of operator=
on the member
variable m_myInt
with the integer 1234
passed as a parameter.
It turns out that this operator is defined (by TracedValue
) to execute
a callback that returns void and takes two integer values as parameters –
an old value and a new value for the integer in question. That is exactly
the function signature for the callback function we provided – IntTrace
.
To summarize, a trace source is, in essence, a variable that holds a list of callbacks. A trace sink is a function used as the target of a callback. The Attribute and object type information systems are used to provide a way to connect trace sources to trace sinks. The act of “hitting” a trace source is executing an operator on the trace source which fires callbacks. This results in the trace sink callbacks registering interest in the source being called with the parameters provided by the source.
[ < ] | [ > ] | [ << ] | [ Up ] | [ >> ] | [Top] | [Contents] | [Index] | [ ? ] |
The TraceConnectWithoutContext
call shown above in the simple example is
actually very rarely used in the system. More typically, the Config
subsystem is used to allow selecting a trace source in the system using what is
called a config path.
For example, one might find something that looks like the following in the system
(taken from examples/tcp-large-transfer.cc
)
|
This should look very familiar. It is the same thing as the previous example,
except that a static member function of class Config
is being called instead
of a method on Object
; and instead of an Attribute
name, a path is
being provided.
The first thing to do is to read the path backward. The last segment of the path
must be an Attribute
of an Object
. In fact, if you had a pointer to
the Object
that has the “CongestionWindow” Attribute
handy (call it
theObject
), you could write this just like the previous example:
|
It turns out that the code for Config::ConnectWithoutContext
does exactly that.
This function takes a path that represents a chain of Object
pointers and follows
them until it gets to the end of the path and interprets the last segment as an
Attribute
on the last object. Let’s walk through what happens.
The leading “/” character in the path refers to a so-called namespace. One of the
predefined namespaces in the config system is “NodeList” which is a list of all of
the nodes in the simulation. Items in the list are referred to by indices into the
list, so “/NodeList/0” refers to the zeroth node in the list of nodes created by
the simulation. This node is actually a Ptr<Node>
and so is a subclass of
an ns3::Object
.
As described in the Object Model section, ns-3
supports an object aggregation
model. The next path segment begins with the “$” character which indicates a
GetObject
call should be made looking for the type that follows. When a
node is initialized by an InternetStackHelper
a number of interfaces are
aggregated to the node. One of these is the TCP level four protocol. The runtime
type of this protocol object is “ns3::TcpL4Protocol”. When the GetObject
is executed, it returns a pointer to the object of this type.
The TcpL4Protocol
class defines an Attribute called “SocketList” which is
a list of sockets. Each socket is actually an ns3::Object
with its own
Attributes
. The items in the list of sockets are referred to by index just
as in the NodeList, so “SocketList/0” refers to the zeroth socket in the list
of sockets on the zeroth node in the NodeList – the first node constructed in the
simulation.
This socket, the type of which turns out to be an ns3::TcpSocketImpl
defines
an attribute called “CongestionWindow” which is a TracedValue<uint32_t>
.
The Config::ConnectWithoutContext
now does a,
|
using the object pointer from “SocketList/0” which makes the connection between
the trace source defined in the socket to the callback – CwndTracer
.
Now, whenever a change is made to the TracedValue<uint32_t>
representing the
congestion window in the TCP socket, the registered callback will be executed and
the function CwndTracer
will be called printing out the old and new values
of the TCP congestion window.
[ < ] | [ > ] | [ << ] | [ Up ] | [ >> ] |
This document was generated on April 21, 2010 using texi2html 1.82.