- 首先是*.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层的记录了。当然前提是你开发的协议必须是有这些层的。