/* * Copyright (c) 1994 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 University of * California, Berkeley and the Network Research Group at * Lawrence Berkeley Laboratory. * 4. Neither the name of the University nor of the Laboratory 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. * * @(#) $Header: /cs/research/mice/starship/src/local/CVS_repository/vic/rtp/source.h,v 1.3 1999/12/10 16:34:13 piers Exp $ (LBL) */ #ifndef vic_source_h #define vic_source_h #include #include "sys-time.h" #include "config.h" #include "timer.h" #include "inet.h" #include "vic_tcl.h" #include "rtp.h" #include "mbus.h" #include "pktbuf-rtp.h" #include "net.h" //placement problems? class SourceManager; class pktbuf; /* Added as a variation of Isidor's approach * for the elastic playout buffer. It actually * incorporates the playout.h & playout.cc files * in source.h adn source.cc p */ class MBusHandler; class SourceManager; #define SOURCE_HASH 1024 #define SHASH(a) ((int)((((a) >> 20) ^ ((a) >> 10) ^ (a)) & (SOURCE_HASH-1))) class MtuAlloc { public: MtuAlloc(); ~MtuAlloc(); inline char *new_blk(); inline void delete_blk(char *bp); protected: static char *blk_list_; }; inline char *MtuAlloc::new_blk() { char *bp; if (blk_list_) { bp = blk_list_; blk_list_ = *((char**)bp); } else bp = new char[2 * RTP_MTU]; return (bp); } inline void MtuAlloc::delete_blk(char *bp) { *((char**)bp) = blk_list_; blk_list_ = bp; } class PacketData : public MtuAlloc { public: PacketData(u_char *pkt, struct rtphdr* rh, u_char *vh, int len, u_int playout); inline void construct(u_char *pkt, struct rtphdr* rh, u_char *vh, int len, u_int playout); ~PacketData(); inline const struct rtphdr *rtp_hdr() const { return (rh_); } inline const u_char *video_data() const { return (vh_); } inline const int data_length() const { return (len_); } inline const int playout() const { return (playout_); } private: u_char *pkt_; struct rtphdr *rh_; u_char *vh_; int len_; u_int playout_; public: PacketData *next_; }; inline void PacketData::construct(u_char *pkt, struct rtphdr* rh, u_char *vh, int len, u_int playout) { pkt_ = pkt; rh_ = rh; vh_ = vh; len_ = len; playout_ = playout; next_ = 0; } //#include "net.h" class PacketHandler : public TclObject { public: virtual ~PacketHandler(); inline PacketHandler(int hdrlen) : hdrlen_(hdrlen), delvar_(0) { } inline int hdrlen() const { return (hdrlen_); } // virtual void recv(const struct rtphdr*, // const u_char* data, int len) = 0; virtual void recv(pktbuf*) = 0; inline u_int32_t delvar() const { return (delvar_); } inline void delvar(u_int32_t v) { delvar_ = v; } protected: int hdrlen_; int delvar_; }; class Source : public TclObject, public Timer { public: Source(u_int32_t srcid, u_int32_t ssrc, Address & addr); virtual ~Source(); PacketHandler* activate(int format); PacketHandler* change_format(int format); void deactivate(); virtual int command(int argc, const char*const* argv); inline PacketHandler* handler() const { return (handler_); } void lts_done(const timeval& now) { lts_done_ = now; } // void lts_data(const timeval& now) { lts_data_ = now; } void action() { if (!busy_) set_busy(); } //? void action() { if (trigger_) trigger_media(); } /* All moved into Layer subClass; void lts_ctrl(const timeval& now) { lts_ctrl_ = now; } void sts_data(u_int32_t now) { sts_data_ = now; } void sts_ctrl(u_int32_t now) { sts_ctrl_ = now; } */ void rtp_ctrl(u_int32_t now) { rtp_ctrl_ = now; } void map_ntp_time(u_int32_t t) { map_ntp_time_ = t; } void map_rtp_time(u_int32_t t) { map_rtp_time_ = t; } void rtp2ntp(int v) { rtp2ntp_ = v; } // inline const timeval& lts_ctrl() const { return (lts_ctrl_); } // inline const timeval& lts_data() const { return (lts_data_); } inline const timeval& lts_done() const { return (lts_done_); } // inline int sts_ctrl() const { return (sts_ctrl_); } // inline int sts_data() const { return (sts_data_); } inline int rtp_ctrl() const { return (rtp_ctrl_); } inline int rtp2ntp() const { return (rtp2ntp_); } inline int apdelay() const { return (apdelay_); } inline const char* sdes(int t) const { return (sdes_[t]); } void sdes(int t, const char* value); inline Address const & addr() const { return (addr_); } inline void addr(const Address & a) { addr_ = a; } inline u_int32_t srcid() const { return (srcid_); } inline void srcid(u_int32_t s) { srcid_ = s; } inline u_int32_t ssrc() const { return (ssrc_); } inline void ssrc(u_int32_t s) { ssrc_ = s; } inline void format(int v) { format_ = v; } inline int format() const { return (format_); } inline void mute(int v) { mute_ = v; } inline int mute() const { return (mute_); } inline void ismixer(int v) { ismixer_ = v; } inline int ismixer() const { return (ismixer_); } // New - For RLM inline int reportLoss() const { return (reportLoss_); } void clear_counters(); class Layer : public TclObject { public: Layer(); void clear_counters(); /*XXX should start at random values*/ inline u_int32_t nb() const { return (nb_); } inline u_int32_t nf() const { return (nf_); } inline u_int32_t np() const { return (np_); } inline u_int32_t nm() const { return (nm_); } inline u_int32_t ns() const { return (cs_ - fs_); } /* no. expected */ inline u_int32_t ehs() const { return (-(fs_ & 0xffff0000) | cs_); } inline u_int32_t cs() const { return (cs_); } inline u_int32_t snp() const { return (snp_); } inline u_int32_t sns() const { return (sns_); } inline u_int32_t runt() const { return (nrunt_); } inline u_int32_t dups() const { return (ndup_); } inline void nb(int v) { nb_ += v; } inline void nf(int v) { nf_ += v; } inline void np(int v) { np_ += v; } inline void nm(int v) { nm_ += v; } inline void snp(int v) { snp_ = v; } inline void sns(int v) { sns_ = v; } inline void fs(int v) { fs_ = v; } inline void runt(int v) { nrunt_ += v; } int cs(u_int16_t v, Source*); int checkseq(u_int16_t v); inline const timeval& lts_ctrl() const { return (lts_ctrl_); } inline const timeval& lts_data() const { return (lts_data_); } inline int sts_ctrl() const { return (sts_ctrl_); } inline int sts_data() const { return (sts_data_); } void lts_data(const timeval& now) { lts_data_ = now; } void lts_ctrl(const timeval& now) { lts_ctrl_ = now; } void sts_data(u_int32_t now) { sts_data_ = now; } void sts_ctrl(u_int32_t now) { sts_ctrl_ = now; } private: u_int32_t fs_; /* first seq. no received */ u_int32_t cs_; /* current (most recent) seq. no received */ u_int32_t np_; /* no. packets received */ u_int32_t nf_; /* no. video frames received */ u_int32_t nb_; /* no. bytes received */ u_int32_t nm_; /* no. misordered packets detected */ u_int32_t snp_; /* last advertised no. pkts received */ u_int32_t sns_; /* last advertised no. pkts exptected */ u_int32_t ndup_; /* no. of duplicate packets (via RTP seqno) */ u_int32_t nrunt_; /* count of packets too small */ u_int32_t sts_data_; /* sndr ts from last data packet (net order) */ u_int32_t sts_ctrl_; /* sndr ts from last control packet */ timeval lts_data_; /* local unix time for last data packet */ timeval lts_ctrl_; /* local unix time for last ctrl packet */ #define SOURCE_NSEQ 64 u_int16_t seqno_[SOURCE_NSEQ]; } *layer_[NLAYER]; inline Layer& layer(int i) const { return (*layer_[i]); } int nlayer_;/*XXX*/ void notify(Source::Layer* layer) {UNUSED(layer);}; inline int late() const { return (late_); } inline u_int32_t badsesslen() const { return (badsesslen_); } inline u_int32_t badsessver() const { return (badsessver_); } inline u_int32_t badsessopt() const { return (badsessopt_); } inline u_int32_t badsdes() const { return (badsdes_); } inline u_int32_t badbye() const { return (badbye_); } inline void late(int v) { late_ += v; } inline void apdelay(int v) { apdelay_ = v; } inline void pending(int v) { pending_ = v; } inline void badsesslen(int v) { badsesslen_ += v; } inline void badsessver(int v) { badsessver_ += v; } inline void badsessopt(int v) { badsessopt_ += v; } inline void badsdes(int v) { badsdes_ += v; } inline void badbye(int v) { badbye_ += v; } int cs(u_int16_t v); int checkseq(u_int16_t v); void lost(int); // Added from MASH int nb() const; int nf() const; int np() const; int nm() const; int ns() const; int runt() const; int dups() const; int missing() const; inline void sync(int v) { sync_ = v; } inline int sync() const { return (sync_); } Source* next_; /* link for SourceManager source list */ Source* hlink_; /* link for SourceManager hash table */ // DMIRAS: This bit for the elastic buffer u_int32_t convert_time(u_int32_t ts); void add(u_char *pckt, struct rtphdr* rh, u_char *vh, int len, u_int playout); void process(struct rtphdr* rh, u_char *vh, int len); inline void elastic(int v) { elastic_ = v; } inline int elastic() { return (elastic_); } // DMIRAS: This bit related to elastic buffer void adapt(u_int32_t arr_ts, u_int32_t src_ts, int flag); inline void skew(u_int s) { skew_ = s; } inline u_int skew() const { return (skew_); } inline double dvar() const { return (dvar_); } void recv(u_char *pkt, struct rtphdr* rh, u_char *vh, int cc); inline void mbus(MBusHandler *m) { mbus_ = m; } protected: char* stats(char* cp) const; void set_busy(); //? void trigger_media(); PacketHandler* handler_; u_int32_t srcid_; /* rtp global src id (CSRC), net order */ u_int32_t ssrc_; /* rtp global sync src id (SSRC), net order) */ Address & addr_; /* address of sender (net order) */ int rtp2ntp_; /* true if we've received a SR report */ u_int32_t sts_data_; /* sndr ts from last data packet (net order) */ u_int32_t sts_ctrl_; /* sndr ts from last control packet */ u_int32_t rtp_ctrl_; /* sndr rtp ts from last control packet */ u_int32_t map_rtp_time_;/* local rtp time of last control packet */ u_int32_t map_ntp_time_;/* local ntp time of last control packet */ timeval lts_done_; /* local unix time for bye packet */ /* following errors are from session (rtcp) processing */ u_int32_t badsesslen_; /* bad header length field */ u_int32_t badsessver_; /* bad header version number */ u_int32_t badsessopt_; /* unrecognized option */ u_int32_t badsdes_; /* sdes cnt > available data */ u_int32_t badbye_; /* bye cnt > available data */ int format_; /* RTP packet type */ int mute_; /* true if source muted */ int lost_; /* true when signal lost */ int busy_; /* nonzero. during talk spurt */ int ismixer_; /* true if source has acted as a 'mixer' */ int reportLoss_; #define SOURCE_NSEQ 64 u_int16_t seqno_[SOURCE_NSEQ]; char* sdes_[RTCP_SDES_MAX + 1]; // DMIRAS: This bit the protected for elastic buffer PacketData *head_; PacketData *tail_; PacketData *free_; u_int32_t now_; float dtskew_; int elastic_; // DMIRAS: adaptive bit of the elastic buffer u_int skew_; /* clock skew */ int delay_; /* pkt delay */ double dvar_; /* delay variance */ int delta_; /* last delay time */ int pdelay_; /* playout delay for this frame */ int apdelay_; /* audio playout delay as sent via mbus */ int adapt_init_; /* Initialised adapt? */ int late_; /* num of late pkts */ int count_; int pending_; /* if set, then have to send a mbus msg */ int sync_; /* set if audio-video synchronisation is set */ void schedule(); virtual void timeout(); /* pointer to the mbus object, so that source can send to it */ MBusHandler *mbus_; }; class SourceManager : public TclObject { public: SourceManager(); static inline SourceManager& instance() { return (instance_); } void init(u_int32_t localid, Address & localaddr); virtual int command(int argc, const char*const* argv); Source* lookup(char *cname); Source* lookup(u_int32_t srcid, u_int32_t ssrc, Address & addr); Source* demux(u_int32_t srcid, Address & addr, u_int16_t seq, int layer); Source* consult(u_int32_t srcid); inline int nsources() const { return (nsources_); } inline Source* sources() const { return (sources_); } void CheckActiveSources(double msgint); void ListSources(); u_int32_t clock() const { return (clock_); } inline Source* localsrc() const { return (localsrc_); } void sortactive(char*) const; void remove(Source*); protected: static int compare(const void*, const void*); Source* enter(Source* s); void remove_from_hashtable(Source* s); Source* lookup_duplicate(u_int32_t srcid, Address & addr); int nsources_; Source* sources_; u_int32_t clock_; int keep_sites_; u_int site_drop_time_; Source* localsrc_; Source* generator_; Source* hashtab_[SOURCE_HASH]; static SourceManager instance_; }; //inline int Source::checkseq(u_int16_t v) inline int Source::Layer::checkseq(u_int16_t v) { int k = v & (SOURCE_NSEQ-1); if (seqno_[k] != v) { seqno_[k] = v; return (0); } else { ++ndup_; return (-1); } } //inline int Source::cs(u_int16_t v) inline int Source::Layer::cs(u_int16_t v, Source* s) { /* * This routine updates a 32 bit sequence number based on * the rtp packet's 16 bit seq. no. */ register int c = cs_; register int d = v - c; if (d < -1024 || d > 1024) { cs_ = v; if (v < 512 && c > 0x10000-512) { /* * seq no. wrapped - subtract 64k from fs to * account for it. */ fs_ -= 0x10000; } else { /* * the seq. no made a very large jump. assume * that the other side restarted without telling * us about it so just re-sync (i.e., pretend * this was the first packet). */ fs_ = v - 1; np_ = 0; nf_ = 0; nb_ = 0; snp_ = 0; nm_ = 0; } } else if (d > 0) { /* d <= 0 means duplicate or reordered packet - ignore */ cs_ = v; if (d < 0) /* out of order */ ++nm_; /*XXX call notifier on loss -- need a more robust check * this assumes in-order deliver. */ if (d != 1 && s->reportLoss()) s->notify(this); } return (checkseq(v)); } #endif