- 首先是*.h/cc文件:
- ./trace/basetrace.h, cc: baseTrace基类
- ./trace/trace.h,cc: Trace类实现
- ./trace/cmu-trace.h,cc: 特别是cmu-trace.cc中的nam_format函数.
注: 以上三个类是继承关系,其中后继的类都包含一个Basetrace *pt_ 指针,用来实现对包的跟踪分析。这个类很重要许多场合,特另是C++代码实现中都要先检测其是否为空。如在Cmu-trace
- common/mobilenode.cc: 该文件是实现节点移动的主要类,其中有一个函数set_destination会记录下所有节点走过的轨迹。如下:
{
。。。。。
if (namChan _ != 0) {
double v = speed_ * sqrt( (dX_ * dX_) + (dY_ * dY_));
sprintf(nwrk_,
"n -t %f -s %d -x %f -y %f -U %f -V %f -T %f",
Scheduler::instance().clock(),
nodeid_,
X_, Y_,
speed_ * dX_, speed_ * dY_,
( v != 0) ? len / v : 0. );
namdump();
}
}
我正是因为忽略了这一点,所以在自定义协议时,没有将namChan_与节点绑定,所以生成的nam文件总是显示不出节点的移动过程。而相应的,其它的仿真脚本,如tcl/ex/wireless.tcl文件中却可以。追踪到半夜,才发现问题。
实现上述绑定过程在tcl/lib/ns-lib.tcl文件中Simulator instproc create-wireless-node中实现,详细可以参见具体代码。而该函数会在生成节点时自动调用,因为我创建节点并没有调用该函数,所以未绑定。
解决办法 :在***-create-mobile-node中加上以下这段代码即可。
set namtracefd [$ns_ get-nam-traceall]
if {$namtracefd != "" } {
$node namattach $namtracefd
}
- *.tcl 文件:
- tcl/lib/ns-lib.tcl,主要包括Simulator级别的支持;
该文件中包含了一个initial_node_pos函数,即初始化节点的位置及大小,这个主要是用于nam中显示用。
Simulator instproc initial_node_pos {nodep size}
同时会将相应的信息输出到nam文件中,主要包括几个方面的信息:节点ID由-s指定,位置(x,y,z),大小-z,标记-v, 颜色-black
n -t * -s 0 -x 68.594620455333327 -y 93.540876681702628 -Z 0 -z 5 -v circle -c black
- tcl/lib/ns-namsupp.tcl ,主要包括节点Node级的nam支持,包含以下可用函数:
Node instproc shape { shape } 设置节点形状
Node instproc color { color } 设置节点颜色
Node instproc label { str} 设置节点标号
Node instproc label-color { str} 设置节点标号颜色
Node instproc label-at { str } 设置节点标号方向
注意:以上函数只在节点初始时设置有效,在运行过程中设置无效的。
- tcl/lib/ns-nam.tcl:主要包括与nam simulator相关的函数,作用不大。
最后:通过nam -p可以显示nam的参数及类型。
最近碰到一个问题是,在我仿真生成的nam文件中,只有节点移动的记录,但是没有数据包的记录。查看了一下CC文件,知道应该是cmu-trace.cc文件中的format没有调用,进一步确定问题出现在下面这一行:
if (pt_->namchannel())
nam_format(p, offset);
GDB跟踪后,发现调用到这个地方if条件不满足,即pt_(见前面,这是个BaseTrace基类对象)的namChan_为空。
进一步检查,发现必须将数据包的各层与Trace对象绑定是在ns-lib.tcl中的Simulator instproc create-wireless-node函数中通过以下方式实现的:
#
# This Trace Target is used to log changes in direction
# and velocity for the mobile node.
#
set tracefd [$self get-ns-traceall]
if {$tracefd != "" } {
$node nodetrace $tracefd
$node agenttrace $tracefd
}
set namtracefd [$self get-nam-traceall]
if {$namtracefd != "" } {
$node namattach $namtracefd
}
上面红色是关键,除了上面,还需要打开各层Trace的开关,这个是通过调用
Simulator instproc node-config实现。奇怪的是,我直接如下设置却不行:
$ns_ macTrace $MacTrace
必须用:
$ns_ node-config -adhocRouting $opt(rp) /
-llType $opt(ll) /
-macType $opt(mac) /
-ifqType $opt(ifq) /
-ifqLen $opt(ifqlen) /
-antType $opt(ant) /
-propType $opt(prop) /
-phyType $opt(netif) /
-topoInstance $topo /
-agentTrace ON /
-routerTrace ON /
-macTrace ON /
-movementTrace OFF /
-channel $chan
这样,就可以在nam文件中生成相应mac, router, agent层的记录了。当然前提是你开发的协议必须是有这些层的。