📄 rtp.cc
字号:
/* -*- Mode:C++; c-basic-offset:8; tab-width:8; indent-tabs-mode:t -*- *//* * Copyright (c) 1997 Regents of the University of California. * All rights reserved. * * Redistribution and use in source and binary forms, with or without * modification, are permitted provided that the following conditions * are met: * 1. Redistributions of source code must retain the above copyright * notice, this list of conditions and the following disclaimer. * 2. Redistributions in binary form must reproduce the above copyright * notice, this list of conditions and the following disclaimer in the * documentation and/or other materials provided with the distribution. * 3. All advertising materials mentioning features or use of this software * must display the following acknowledgement: * This product includes software developed by the MASH Research * Group at the University of California Berkeley. * 4. Neither the name of the University nor of the Research Group may be * used to endorse or promote products derived from this software without * specific prior written permission. * * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE * ARE DISCLAIMED. IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF * SUCH DAMAGE. */#ifndef lintstatic const char rcsid[] = "@(#) $Header: /nfs/jade/vint/CVSROOT/ns-2/apps/rtp.cc,v 1.26 2000/08/18 18:34:01 haoboy Exp $";#endif#include <stdlib.h>#include "config.h"#include "agent.h"#include "random.h"#include "rtp.h"int hdr_rtp::offset_;class RTPHeaderClass : public PacketHeaderClass {public: RTPHeaderClass() : PacketHeaderClass("PacketHeader/RTP", sizeof(hdr_rtp)) { bind_offset(&hdr_rtp::offset_); }} class_rtphdr;static class RTPAgentClass : public TclClass {public: RTPAgentClass() : TclClass("Agent/RTP") {} TclObject* create(int, const char*const*) { return (new RTPAgent()); }} class_rtp_agent;RTPAgent::RTPAgent() : Agent(PT_RTP), session_(0), lastpkttime_(-1e6), running_(0), rtp_timer_(this){ bind("seqno_", &seqno_); bind_time("interval_", &interval_); bind("packetSize_", &size_); bind("maxpkts_", &maxpkts_); bind("random_", &random_);}void RTPAgent::start(){ running_ = 1; sendpkt(); rtp_timer_.resched(interval_);}void RTPAgent::stop(){ rtp_timer_.force_cancel(); finish();}void RTPAgent::sendmsg(int nbytes, const char* /*flags*/){ Packet *p; int n; if (++seqno_ < maxpkts_) { if (size_) n = nbytes / size_; else printf("Error: RTPAgent size = 0\n"); if (nbytes == -1) { start(); return; } while (n-- > 0) { p = allocpkt(); hdr_rtp* rh = hdr_rtp::access(p); rh->seqno() = seqno_; target_->recv(p); } n = nbytes % size_; if (n > 0) { p = allocpkt(); hdr_rtp* rh = hdr_rtp::access(p); rh->seqno() = seqno_; target_->recv(p); } idle(); } else { finish(); // xxx: should we deschedule the timer here? */ };}void RTPAgent::timeout(int) { if (running_) { sendpkt(); if (session_) session_->localsrc_update(size_); double t = interval_; if (random_) /* add some zero-mean white noise */ t += interval_ * Random::uniform(-0.5, 0.5); rtp_timer_.resched(t); }}/* * finish() is called when we must stop (either by request or because * we're out of packets to send. */void RTPAgent::finish(){ running_ = 0; Tcl::instance().evalf("%s done", this->name());}void RTPAgent::advanceby(int delta){ maxpkts_ += delta; if (seqno_ < maxpkts_ && !running_) start();} void RTPAgent::recv(Packet* p, Handler*){ if (session_) session_->recv(p, 0); else Packet::free(p);}int RTPAgent::command(int argc, const char*const* argv){ if (argc == 2) { if (strcmp(argv[1], "rate-change") == 0) { rate_change(); return (TCL_OK); } else if (strcmp(argv[1], "start") == 0) { start(); return (TCL_OK); } else if (strcmp(argv[1], "stop") == 0) { stop(); return (TCL_OK); } } else if (argc == 3) { if (strcmp(argv[1], "session") == 0) { session_ = (RTPSession*)TclObject::lookup(argv[2]); return (TCL_OK); } else if (strcmp(argv[1], "advance") == 0) { int newseq = atoi(argv[2]); advanceby(newseq - seqno_); return (TCL_OK); } else if (strcmp(argv[1], "advanceby") == 0) { advanceby(atoi(argv[2])); return (TCL_OK); } } return (Agent::command(argc, argv));}/* * We modify the rate in this way to get a faster reaction to the a rate * change since a rate change from a very low rate to a very fast rate may * take an undesireably long time if we have to wait for timeout at the old * rate before we can send at the new (faster) rate. */void RTPAgent::rate_change(){ rtp_timer_.force_cancel(); double t = lastpkttime_ + interval_; double now = Scheduler::instance().clock(); if ( t > now) rtp_timer_.resched(t - now); else { sendpkt(); rtp_timer_.resched(interval_); }}void RTPAgent::sendpkt(){ Packet* p = allocpkt(); lastpkttime_ = Scheduler::instance().clock(); makepkt(p); target_->recv(p, (Handler*)0);}void RTPAgent::makepkt(Packet* p){ hdr_rtp *rh = hdr_rtp::access(p); /* Fill in srcid_ and seqno */ rh->seqno() = seqno_++; rh->srcid() = session_ ? session_->srcid() : 0;}void RTPTimer::expire(Event* /*e*/) { a_->timeout(0);}
⌨️ 快捷键说明
复制代码
Ctrl + C
搜索代码
Ctrl + F
全屏模式
F11
切换主题
Ctrl + Shift + D
显示快捷键
?
增大字号
Ctrl + =
减小字号
Ctrl + -