many_tcp.tcl
上传用户:rrhhcc
上传日期:2015-12-11
资源大小:54129k
文件大小:18k
源码类别:

通讯编程

开发平台:

Visual C++

  1. #
  2. # many_tcp.tcl
  3. # $Id: many_tcp.tcl,v 1.18 2000/09/14 18:19:26 haoboy Exp $
  4. #
  5. # Copyright (c) 1998 University of Southern California.
  6. # All rights reserved.                                            
  7. #                                                                
  8. # Redistribution and use in source and binary forms are permitted
  9. # provided that the above copyright notice and this paragraph are
  10. # duplicated in all such forms and that any documentation, advertising
  11. # materials, and other materials related to such distribution and use
  12. # acknowledge that the software was developed by the University of
  13. # Southern California, Information Sciences Institute.  The name of the
  14. # University may not be used to endorse or promote products derived from
  15. # this software without specific prior written permission.
  16. # THIS SOFTWARE IS PROVIDED "AS IS" AND WITHOUT ANY EXPRESS OR IMPLIED
  17. # WARRANTIES, INCLUDING, WITHOUT LIMITATION, THE IMPLIED WARRANTIES OF
  18. # MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE.
  19. proc usage {} {
  20. puts stderr {usage: ns rbp_simulation.tcl [options]
  21. This simulation demonstrates large numbers of TCP flows.
  22. Comments/bugs to John Heidemann <johnh@isi.edu>.
  23. Currently this simulation appears to be memory limited for most
  24. practical purposes.  Each active flow consumes ~100KB of memory (very
  25. roughly).
  26. Why is a flow so expensive?  Primarly because We allocate a separate
  27. pair of nodes and links per flow.  (I don't know how to do otherwise
  28. and preserve some reasonable statement about the independence of
  29. randomly selected client RTTs).
  30. How can performance be improved?
  31. - use multiple concurrent flows per node (should give much better
  32. memory performance at the cost of node RTT independence)
  33. - use session-level simulation techniques (see ``Enabling Large-scale
  34. simulations: selective abstraction approach to the study of multicast
  35. protocols'' by Huang, Estrin, and Heidemann in MASCOTS '98).
  36. - improve the efficiency of the underlying representations (there are
  37. some ways to optimize some otcl common cases)  (this work is in progress)
  38. - use higher level representations of TCP (in progress)
  39. Options (specify with -OPTIONNAME OPTIONVALUE):
  40. }
  41. global raw_opt_info
  42. puts stderr $raw_opt_info
  43. exit 1
  44. }
  45. global raw_opt_info
  46. set raw_opt_info {
  47. # how long to run the sim?
  48. duration 30
  49. # initilization: just start n clients at time 0
  50. # NEEDSWORK:  a more realistic ramp-up model
  51. # or some kind of autodetection on when we've reached
  52. # steady state would be nice.
  53. initial-client-count 10
  54. #
  55. # BASIC TOPOLOGY:
  56. #
  57. # (The basic n clients on the left and right going through a
  58. # bottleneck.)
  59. #
  60. # cl_1                                                     cr_1
  61. # ...     ---- bottleneck_left ---- bottleneck_right ---   ...
  62. # cl_n                                                     cr_n
  63. #
  64. # node-number 0 specifies a new pair of nodes
  65. # on the left and right for each new client
  66. node-number 0
  67. # NEEDSWORK:
  68. # The number of agents attached to a node cannot exceed
  69. # the port-field length (255 bits).  There is currently
  70. # no check or warning message for this.
  71. #
  72. # Currently all data traffic flows left-to-right.
  73. # NEEDSWORK: relax this assumption (but Poduri and Nichols
  74. # I-D suggests that relaxing it won't change things much).
  75. #
  76. #
  77. # CLIENT TRAFFIC MODEL:
  78. #
  79. # arrival rate per second (arrival is poisson)
  80. client-arrival-rate 1
  81. # Currently clients are either mice or elephants.
  82. # NEEDSWORK:  should better model http-like traffic patterns.
  83. # In particular, netscape's 4-connection model makes
  84. # a *big* difference in traffic patterns
  85. # and is not currently modeled at all.
  86. client-mouse-chance 90
  87. client-mouse-packets 10
  88. client-elephant-packets 100
  89. # For traffic in the reverse direction.
  90. client-reverse-chance 0
  91. # Pkt size in bytes.
  92. # NEEDSWORK:  should check that everything is uniformly
  93. # specified (router queues are in packets of 1000B length?).
  94. client-pkt-size 576
  95. #
  96. # CLIENT NETWORK CONNECTION:
  97. #
  98. client-bw 56kb
  99. # client-server rtt is uniform over this range (currently)
  100. # NEEDSWORK:  does this need to be modeled more accurately?
  101. client-delay random
  102. client-delay-range 100ms
  103. client-queue-method DropTail
  104. # Insure that client routers are never a bottleneck.
  105. client-queue-length 100
  106. #
  107. # CLIENT/SERVER TCP IMPLEMENTATION:
  108. #
  109. # NEEDSWORK: should add HTTP model over TCP.
  110. source-tcp-method TCP/Reno
  111. sink-ack-method TCPSink/DelAck
  112. # Set init-win to 1 for initial windows of size 1.
  113. # Set init-win to 10 for initial windows of 10 packets.
  114. # Set init-win to 0 for initial windows per internet-draft.
  115. init-win 1
  116. #
  117. # BOTTLENECK LINK MODEL:
  118. #
  119. bottle-bw 10Mb
  120. bottle-delay 4ms
  121. bottle-queue-method RED
  122. # bottle-queue-length is either in packets or
  123. # is "bw-delay-product" which does the currently
  124. # expected thing.
  125. bottle-queue-length bw-delay-product
  126. #
  127. # OUTPUT OPTIONS:
  128. #
  129. graph-results 0
  130. # Set graph-scale to 2 for "rows" for each flow.
  131. graph-scale 1
  132. graph-join-queueing 1
  133. gen-map 0
  134. mem-trace 0
  135. print-drop-rate 0
  136. debug 1
  137. title none
  138. # set test-suite to write the graph to opts(test-suite-file)
  139. test-suite 0
  140. test-suite-file temp.rands
  141.    
  142. # Random number seed; default is 0, so ns will give a 
  143. # diff. one on each invocation.
  144. ns-random-seed 0
  145. # Animation options; complete traces are useful
  146. # for nam only, so do those only when a tracefile
  147. # is being used for nam
  148. # Set trace-filename to "none" for no tracefile.
  149. trace-filename out
  150. trace-all 0
  151. namtrace-some 0
  152. namtrace-all 0
  153. # Switch to generate the nam tcl file from here
  154. # itself
  155. nam-generate-cmdfile 0
  156. }
  157. Class Main
  158. Main instproc default_options {} {
  159. global opts opt_wants_arg raw_opt_info
  160. set cooked_opt_info $raw_opt_info
  161. while {$cooked_opt_info != ""} {
  162. if {![regexp "^[^n]*n" $cooked_opt_info line]} {
  163. break
  164. }
  165. regsub "^[^n]*n" $cooked_opt_info {} cooked_opt_info
  166. set line [string trim $line]
  167. if {[regexp "^[ t]*#" $line]} {
  168. continue
  169. }
  170. if {$line == ""} {
  171. continue
  172. } elseif [regexp {^([^ ]+)[ ]+([^ ]+)$} $line dummy key value] {
  173. set opts($key) $value
  174. set opt_wants_arg($key) 1
  175. } else {
  176. set opt_wants_arg($key) 0
  177. # die "unknown stuff in raw_opt_infon"
  178. }
  179. }
  180. }
  181. Main instproc process_args {av} {
  182. global opts opt_wants_arg
  183. $self default_options
  184. for {set i 0} {$i < [llength $av]} {incr i} {
  185. set key [lindex $av $i]
  186. if {$key == "-?" || $key == "--help" || $key == "-help" || $key == "-h"} {
  187. usage
  188. }
  189. regsub {^-} $key {} key
  190. if {![info exists opt_wants_arg($key)]} {
  191. puts stderr "unknown option $key";
  192. usage
  193. }
  194. if {$opt_wants_arg($key)} {
  195. incr i
  196. set opts($key) [lindex $av $i]
  197. } else {
  198. set opts($key) [expr !opts($key)]
  199. }
  200. }
  201. }
  202. proc my-duplex-link {ns n1 n2 bw delay queue_method queue_length} {
  203. global opts 
  204. $ns duplex-link $n1 $n2 $bw $delay $queue_method
  205. $ns queue-limit $n1 $n2 $queue_length
  206. $ns queue-limit $n2 $n1 $queue_length
  207. }
  208. Main instproc init_network {} {
  209. global opts fmon
  210. # nodes
  211. # build right to left
  212. $self instvar bottle_l_ bottle_r_ cs_l_ cs_r_ ns_ cs_count_ ns_ clients_started_ clients_finished_ 
  213. #
  214. # Figure supported load.
  215. #
  216. set expected_load_per_client_in_bps [expr ($opts(client-mouse-chance)/100.0)*$opts(client-mouse-packets)*$opts(client-pkt-size)*8 + (1.0-$opts(client-mouse-chance)/100.0)*$opts(client-elephant-packets)*$opts(client-pkt-size)*8]
  217. if {$opts(debug)} {
  218. set max_clients_per_second [expr [$ns_ bw_parse $opts(bottle-bw)]/$expected_load_per_client_in_bps]
  219. puts [format "maximum clients per second: %.3f" $max_clients_per_second]
  220. }
  221. # Compute optimal (?) bottleneck queue size
  222. # as the bw-delay product.
  223. if {$opts(bottle-queue-length) == "bw-delay-product"} {
  224. set opts(bottle-queue-length) [expr ([$ns_ bw_parse $opts(bottle-bw)] * ([$ns_ delay_parse $opts(bottle-delay)] + [$ns_ delay_parse $opts(client-delay-range)]) + $opts(client-pkt-size)*8 - 1)/ ($opts(client-pkt-size) * 8)]
  225. puts "optimal bw queue size: $opts(bottle-queue-length)"
  226. }
  227. # Do our own routing with expanded addresses (21 bits nodes).
  228. # (Basic routing limits us to 128 nodes == 64 clients).
  229. $ns_ rtproto Manual
  230. $ns_ set-address-format expanded
  231. # set up the bottleneck
  232. set bottle_l_ [$ns_ node]
  233. set bottle_r_ [$ns_ node]
  234. my-duplex-link $ns_ $bottle_l_ $bottle_r_ $opts(bottle-bw) $opts(bottle-delay) $opts(bottle-queue-method) $opts(bottle-queue-length)
  235.         if {$opts(print-drop-rate)} {
  236. set slink [$ns_ link $bottle_l_ $bottle_r_]
  237. set fmon [$ns_ makeflowmon Fid]
  238. $ns_ attach-fmon $slink $fmon
  239. }
  240. # Bottlenecks need large routing tables.
  241. # [$bottle_l_ set classifier_] resize 511
  242. # [$bottle_r_ set classifier_] resize 511
  243. # Default routes to the other.
  244. [$bottle_l_ get-module "Manual"] add-route-to-adj-node -default $bottle_r_
  245. [$bottle_r_ get-module "Manual"] add-route-to-adj-node -default $bottle_l_
  246. # Clients are built dynamically.
  247. set cs_count_ 0
  248. set clients_started_ 0
  249. set clients_finished_ 0
  250. }
  251. # create a new pair of end nodes
  252. Main instproc create_client_nodes {node} {
  253. global opts 
  254. $self instvar bottle_l_ bottle_r_ cs_l_ cs_r_ sources_ cs_count_ ns_ rng_
  255. set now [$ns_ now]
  256. set cs_l_($node) [$ns_ node]
  257. set cs_r_($node) [$ns_ node]
  258. # Set delay.
  259. set delay $opts(client-delay)
  260. if {$delay == "random"} {
  261. set delay [$rng_ exponential [$ns_ delay_parse $opts(client-delay-range)]]
  262. }
  263. # Now divide the delay into the two haves and set up the network.
  264. set ldelay [$rng_ uniform 0 $delay]
  265. set rdelay [expr $delay - $ldelay]
  266. my-duplex-link $ns_ $cs_l_($node) $bottle_l_ $opts(client-bw) $ldelay $opts(client-queue-method) $opts(client-queue-length)
  267. my-duplex-link $ns_ $cs_r_($node) $bottle_r_ $opts(client-bw) $rdelay $opts(client-queue-method) $opts(client-queue-length)
  268. # Add routing in all directions
  269. [$cs_l_($node) get-module "Manual"] add-route-to-adj-node -default $bottle_l_
  270. [$cs_r_($node) get-module "Manual"] add-route-to-adj-node -default $bottle_r_
  271. [$bottle_l_ get-module "Manual"] add-route-to-adj-node $cs_l_($node)
  272. [$bottle_r_ get-module "Manual"] add-route-to-adj-node $cs_r_($node)
  273. if {$opts(debug)} {
  274.  # puts "t=[format %.3f $now]: node pair $node created"
  275.  # puts "delay $delay ldelay $ldelay"
  276. }
  277. }
  278. # Get the number of the node pair
  279. Main instproc get_node_number { client_number } {
  280.         global opts
  281.         if {$opts(node-number) > 0} {
  282.                 set node [expr $client_number % $opts(node-number)]
  283.         } else {
  284.                 set node $client_number
  285.         }
  286.         return $node
  287. }
  288. # return the client index
  289. Main instproc create_a_client {} {
  290. global opts
  291. $self instvar cs_l_ cs_r_ sources_ cs_count_ ns_ rng_
  292. # Get the client number for the new client.
  293. set now [$ns_ now]
  294. set i $cs_count_
  295. incr cs_count_
  296. set node $i
  297. if {[expr $i % 100] == 0} {
  298. puts "t=[format %.3f $now]: client $i created"
  299. }
  300. # Get the source and sink nodes.
  301. if {$opts(node-number) > 0} {
  302. if {$node < $opts(node-number) } {
  303. $self create_client_nodes $node
  304. } else {
  305. set node [$self get_node_number $i]
  306. }
  307. } else {
  308. $self create_client_nodes $node
  309. }
  310. if {$opts(debug)} {
  311. # puts "t=[format %.3f $now]: client $i uses node pair $node"
  312. }
  313. # create sources and sinks in both directions
  314. # (actually, only one source per connection, for now)
  315. if {[$rng_ integer 100] < $opts(client-reverse-chance)} {
  316. set sources_($i) [$ns_ create-connection-list $opts(source-tcp-method) $cs_r_($node) $opts(sink-ack-method) $cs_l_($node) $i]
  317. } else {
  318. set sources_($i) [$ns_ create-connection-list $opts(source-tcp-method) $cs_l_($node) $opts(sink-ack-method) $cs_r_($node) $i]
  319. }
  320. [lindex $sources_($i) 0] set maxpkts_ 0
  321. [lindex $sources_($i) 0] set packetSize_ $opts(client-pkt-size)
  322. # Set up a callback when this client ends.
  323. [lindex $sources_($i) 0] proc done {} "$self finish_a_client $i"
  324. if {$opts(debug)} {
  325. # puts "t=[$ns_ now]: client $i created"
  326. }
  327. return $i
  328. }
  329. #
  330. # Make a batch of clients to amortize the cost of routing recomputation
  331. # (actually no longer improtant).
  332. #
  333. Main instproc create_some_clients {} {
  334. global opts
  335. $self instvar idle_clients_ ns_ cs_count_
  336. set now [$ns_ now]
  337. set step 16
  338. if {$opts(debug)} {
  339. puts "t=[format %.3f $now]: creating clients $cs_count_ to [expr $cs_count_ + $step - 1]"
  340. }
  341. for {set i 0} {$i < $step} {incr i} {
  342. lappend idle_clients_ [$self create_a_client]
  343. }
  344. # debugging:
  345. # puts "after client_create:"
  346. # $ns_ gen-map
  347. # $self instvar bottle_l_ bottle_r_
  348. # puts "bottle_l_ classifier_:"
  349. # [$bottle_l_ set classifier_] dump
  350. # puts "bottle_r_ classifier_:"
  351. # [$bottle_r_ set classifier_] dump
  352. }
  353. Main instproc start_a_client {} {
  354. global opts
  355. $self instvar idle_clients_ ns_ sources_ rng_ source_start_ source_size_ clients_started_
  356. set i ""
  357. set now [$ns_ now]
  358. # can we reuse a dead client?
  359. if {![info exists idle_clients_]} {
  360. set idle_clients_ ""
  361. }
  362. while {$idle_clients_ == ""} {
  363. $self create_some_clients
  364. }
  365. set i [lindex $idle_clients_ 0]
  366. set idle_clients_ [lrange $idle_clients_ 1 end]
  367. # Reset the connection.
  368. [lindex $sources_($i) 0] reset
  369. [lindex $sources_($i) 1] reset 
  370. # Start traffic for that client.
  371. if {[$rng_ integer 100] < $opts(client-mouse-chance)} {
  372. set len $opts(client-mouse-packets)
  373. } else {
  374. set len $opts(client-elephant-packets)
  375. }
  376. [lindex $sources_($i) 0] advanceby $len
  377. set source_start_($i) $now
  378. set source_size_($i) $len
  379. if {$opts(debug)} {
  380.  # puts "t=[$ns_ now]: client $i started, ldelay=[format %.6f $ldelay], rdelay=[format %.6f $rdelay]"
  381. puts "t=[format %.3f $now]: client $i started"
  382. }
  383. incr clients_started_
  384. }
  385. Main instproc finish_a_client {i} {
  386. global opts
  387. $self instvar ns_ idle_clients_ source_start_ source_size_ clients_finished_
  388. set now [$ns_ now]
  389. if {$opts(debug)} {
  390. set delta [expr $now - $source_start_($i)]
  391. puts "t=[format %.3f $now]: client $i finished ($source_size_($i) pkts, $delta s)"
  392. }
  393. lappend idle_clients_ $i
  394. incr clients_finished_
  395. }
  396. Main instproc schedule_continuing_traffic {} {
  397. global opts
  398. $self instvar ns_ rng_
  399. $self start_a_client
  400. # schedule the next one
  401. set next [expr [$ns_ now]+([$rng_ exponential]/$opts(client-arrival-rate))]
  402. if {$opts(debug)} {
  403. # puts "t=[$ns_ now]: next continuing traffic at $next"
  404. }
  405. $ns_ at $next "$self schedule_continuing_traffic"
  406. }
  407. Main instproc schedule_initial_traffic {} {
  408. global opts
  409. # Start with no pending clients.
  410. $self instvar idle_clients_
  411. # Start initial clients.
  412. for {set i 0} {$i < $opts(initial-client-count)} {incr i} {
  413. $self start_a_client
  414. }
  415. }
  416. Main instproc open_trace { stop_time } {
  417. global opts
  418. $self instvar ns_ trace_file_ nam_trace_file_ trace_filename_
  419. set trace_filename_ $opts(trace-filename)
  420. exec rm -f "$trace_filename_.tr"
  421. set trace_file_ [open "$trace_filename_.tr" w]
  422. set stop_actions "close $trace_file_"
  423. if {$opts(namtrace-some) || $opts(namtrace-all)} {
  424. exec rm -f "$trace_filename_.nam"
  425. set nam_trace_file_ [open "$trace_filename_.nam" w]
  426. set $stop_actions "$stop_actions; close $nam_trace_file_"
  427. } else {
  428. set nam_trace_file_ ""
  429. }
  430. $ns_ at $stop_time "$stop_actions; $self finish"
  431. return "$trace_file_ $nam_trace_file_"
  432. }
  433. # There seems to be a problem with the foll function, so quit plotting 
  434. # with -a -q, use just -a.
  435. Main instproc finish {} {
  436.         global opts fmon PERL
  437. $self instvar trace_filename_ ns_ cs_count_ clients_started_ clients_finished_
  438. puts "total clients started: $clients_started_"
  439. puts "total clients finished: $clients_finished_"
  440. if {$opts(print-drop-rate)} {
  441. set drops [$fmon set pdrops_]
  442. set pkts [$fmon set parrivals_]
  443. puts "total_drops $drops total_packets $pkts"
  444. set droprate [expr 100.0*$drops / $pkts ]
  445. puts [format "drop_percentage %7.4f" $droprate]
  446. }
  447.         if {$opts(trace-filename) != "none"} {
  448. set title $opts(title)
  449.                 set flow_factor 1
  450.                 if {$opts(graph-scale) == "2"} {
  451.                         set flow_factor 100
  452.                 }
  453. # Make sure that we run in place even without raw2xg in our path
  454. # (for the test suites).
  455. set raw2xg raw2xg
  456. if [file exists ../../bin/raw2xg] {
  457. set raw2xg ../../bin/raw2xg
  458. }
  459. set raw2xg_opts ""
  460. if {$opts(graph-join-queueing)} {
  461. set raw2xg_opts "$raw2xg_opts -q"
  462. }
  463. # always run raw2xg because maybe we need the output
  464. set cmd "$raw2xg -a $raw2xg_opts -n $flow_factor < $trace_filename_.tr >$trace_filename_.xg"
  465. eval "exec $PERL $cmd"
  466. if {$opts(graph-results)} {
  467. if {$opts(graph-join-queueing)} {
  468. exec xgraph -t $title  < $trace_filename_.xg &
  469. } else {
  470. exec xgraph -tk -nl -m -bb -t $title < $trace_filename_.xg &
  471. }
  472. }
  473. if {$opts(test-suite)} {
  474. exec cp $trace_filename_.xg $opts(test-suite-file)
  475. }
  476. # exec raw2xg -a < out.tr | xgraph -t "$opts(server-tcp-method)" &
  477. }
  478. if {$opts(mem-trace)} {
  479. $ns_ clearMemTrace
  480. }
  481. exit 0
  482. }
  483. Main instproc trace_stuff {} {
  484. global opts
  485. $self instvar bottle_l_ bottle_r_ ns_ trace_file_ nam_trace_file_
  486. $self open_trace $opts(duration)
  487. if {$opts(trace-all)} {
  488. $ns_ trace-all $trace_file_
  489. }
  490. if {$opts(namtrace-all)} {
  491. $ns_ namtrace-all $nam_trace_file_
  492. } elseif {$opts(namtrace-some)} {
  493. # xxx
  494. $bottle_l_ dump-namconfig
  495. $bottle_r_ dump-namconfig
  496. [$ns_ link $bottle_l_ $bottle_r_] dump-namconfig
  497. $ns_ namtrace-queue $bottle_l_ $bottle_r_ $nam_trace_file_
  498. $ns_ namtrace-queue $bottle_r_ $bottle_l_ $nam_trace_file_
  499. }
  500. # regular tracing.
  501. # trace left-to-right only
  502.         $ns_ trace-queue $bottle_l_ $bottle_r_ $trace_file_
  503.         $ns_ trace-queue $bottle_r_ $bottle_l_ $trace_file_
  504. # Currently tracing is somewhat broken because
  505. # of how the plumbing happens.
  506. }
  507. Main instproc init {av} {
  508. global opts
  509. $self process_args $av
  510. $self instvar ns_ 
  511. set ns_ [new Simulator]
  512.         # Seed random no. generator; ns-random with arg of 0 heuristically
  513.         # chooses a random number that changes on each invocation.
  514. $self instvar rng_
  515. set rng_ [new RNG]
  516. $rng_ seed $opts(ns-random-seed)
  517. $rng_ next-random
  518. $self init_network
  519. if {$opts(trace-filename) == "none"} {
  520. $ns_ at $opts(duration) "$self finish"
  521. } else {
  522. $self trace_stuff
  523. }
  524. # xxx: hack (next line)
  525. # $self create_some_clients
  526. $ns_ at 0 "$self schedule_initial_traffic"
  527. if {$opts(client-arrival-rate) != 0} {
  528. $ns_ at 0 "$self schedule_continuing_traffic"
  529. }
  530. if {$opts(gen-map)} {
  531. $ns_ gen-map
  532. }       
  533. Agent/TCP set syn_ true
  534. Agent/TCP set delay_growth_ true
  535. Agent/TCP set windowInit_ 1
  536. Agent/TCP set windowInitOption_ 1
  537. if {$opts(init-win) == "0"} {
  538. Agent/TCP set windowInitOption_ 2
  539. } elseif {$opts(init-win) == "10"} {
  540. Agent/TCP set windowInitOption_ 1
  541. Agent/TCP set windowInit_ 10
  542. } elseif {$opts(init-win) == "20"} {
  543. Agent/TCP set windowInitOption_ 1
  544. Agent/TCP set windowInit_ 20
  545. puts "init-win 20"
  546. }
  547. $ns_ run
  548. }
  549. global in_test_suite
  550. if {![info exists in_test_suite]} {
  551. global $argv
  552. new Main $argv
  553. }