⭐ 欢迎来到虫虫下载站! | 📦 资源下载 📁 资源专辑 ℹ️ 关于我们
⭐ 虫虫下载站

📄 rtpparse.c

📁 mediastreamer2是开源的网络传输媒体流的库
💻 C
字号:
/*  The oRTP library is an RTP (Realtime Transport Protocol - rfc3550) stack.  Copyright (C) 2001  Simon MORLAT simon.morlat@linphone.org  This library is free software; you can redistribute it and/or  modify it under the terms of the GNU Lesser General Public  License as published by the Free Software Foundation; either  version 2.1 of the License, or (at your option) any later version.  This library is distributed in the hope that it will be useful,  but WITHOUT ANY WARRANTY; without even the implied warranty of  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU  Lesser General Public License for more details.  You should have received a copy of the GNU Lesser General Public  License along with this library; if not, write to the Free Software  Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA*/#include <ortp/ortp.h>#include "jitterctl.h"#include "utils.h"#include "rtpsession_priv.h"#define SSRC_CHANGED_THRESHOLD 50static void queue_packet(queue_t *q, int maxrqsz, mblk_t *mp, rtp_header_t *rtp, int *discarded){	mblk_t *tmp;	int header_size;	*discarded=0;	header_size=RTP_FIXED_HEADER_SIZE+ (4*rtp->cc);	if ((mp->b_wptr - mp->b_rptr)==header_size){		ortp_debug("Rtp packet contains no data.");		(*discarded)++;		freemsg(mp);		return;	}	/* and then add the packet to the queue */		rtp_putq(q,mp);	/* make some checks: q size must not exceed RtpStream::max_rq_size */	while (q->q_mcount > maxrqsz)	{		/* remove the oldest mblk_t */		tmp=getq(q);		if (mp!=NULL)		{			ortp_debug("rtp_putq: Queue is full. Discarding message with ts=%i",((rtp_header_t*)mp->b_rptr)->timestamp);			freemsg(tmp);			(*discarded)++;		}	}}void rtp_session_rtp_parse(RtpSession *session, mblk_t *mp, uint32_t local_str_ts, struct sockaddr *addr, socklen_t addrlen){	int i;	rtp_header_t *rtp;	int msgsize;	RtpStream *rtpstream=&session->rtp;	rtp_stats_t *stats=&rtpstream->stats;		msgsize=mp->b_wptr-mp->b_rptr;	if (msgsize<RTP_FIXED_HEADER_SIZE){		ortp_warning("Packet too small to be a rtp packet (%i)!",msgsize);		rtpstream->stats.bad++;		ortp_global_stats.bad++;		freemsg(mp);		return;	}	rtp=(rtp_header_t*)mp->b_rptr;	if (rtp->version!=2)	{		/* try to see if it is a STUN packet */		uint16_t stunlen=*((uint16_t*)(mp->b_rptr + sizeof(uint16_t)));		stunlen = ntohs(stunlen);		if (stunlen+20==mp->b_wptr-mp->b_rptr){			/* this looks like a stun packet */			if (session->eventqs!=NULL){				OrtpEvent *ev=ortp_event_new(ORTP_EVENT_STUN_PACKET_RECEIVED);				OrtpEventData *ed=ortp_event_get_data(ev);				ed->packet=mp;				ed->ep=rtp_endpoint_new(addr,addrlen);				rtp_session_dispatch_event(session,ev);				return;			}		}else{			/* discard*/			ortp_debug("Receiving rtp packet with version number !=2...discarded");			stats->bad++;			ortp_global_stats.bad++;			freemsg(mp);			return;		}	}	/* only count non-stun packets. */	ortp_global_stats.packet_recv++;	stats->packet_recv++;	ortp_global_stats.hw_recv+=msgsize;	stats->hw_recv+=msgsize;	session->rtp.hwrcv_since_last_SR++;		/* convert all header data from network order to host order */	rtp->seq_number=ntohs(rtp->seq_number);	rtp->timestamp=ntohl(rtp->timestamp);	rtp->ssrc=ntohl(rtp->ssrc);	/* convert csrc if necessary */	if (rtp->cc*sizeof(uint32_t) > (uint32_t) (msgsize-RTP_FIXED_HEADER_SIZE)){		ortp_debug("Receiving too short rtp packet.");		stats->bad++;		ortp_global_stats.bad++;		freemsg(mp);		return;	}#ifndef PERF	/* Write down the last RTP/RTCP packet reception time. */	gettimeofday(&session->last_recv_time, NULL);#endif	for (i=0;i<rtp->cc;i++)		rtp->csrc[i]=ntohl(rtp->csrc[i]);	/*the goal of the following code is to lock on an incoming SSRC to avoid	receiving "mixed streams"*/	if (session->ssrc_set){		/*the ssrc is set, so we must check it */		if (session->rcv.ssrc!=rtp->ssrc){			if (session->inc_ssrc_candidate==rtp->ssrc){				session->inc_same_ssrc_count++;			}else{				session->inc_same_ssrc_count=0;				session->inc_ssrc_candidate=rtp->ssrc;			}			if (session->inc_same_ssrc_count>SSRC_CHANGED_THRESHOLD){				/* store the sender rtp address to do symmetric RTP */				if (!session->use_connect){					if (session->rtp.socket>0 && session->symmetric_rtp){						/* store the sender rtp address to do symmetric RTP */						memcpy(&session->rtp.rem_addr,addr,addrlen);						session->rtp.rem_addrlen=addrlen;					}				}				session->rcv.ssrc=rtp->ssrc;				rtp_signal_table_emit(&session->on_ssrc_changed);			}else{				/*discard the packet*/				ortp_debug("Receiving packet with unknown ssrc.");				stats->bad++;				ortp_global_stats.bad++;				freemsg(mp);				return;			}		}    else{			/* The SSRC change must not happen if we still receive			ssrc from the initial source. */			session->inc_same_ssrc_count=0;		}	}else{		session->ssrc_set=TRUE;		session->rcv.ssrc=rtp->ssrc;		if (!session->use_connect){			if (session->rtp.socket>0 && session->symmetric_rtp){				/* store the sender rtp address to do symmetric RTP */				memcpy(&session->rtp.rem_addr,addr,addrlen);				session->rtp.rem_addrlen=addrlen;			}		}	}		/* update some statistics */	{		poly32_t *extseq=(poly32_t*)&rtpstream->hwrcv_extseq;		if (rtp->seq_number>extseq->split.lo){			extseq->split.lo=rtp->seq_number;		}else if (rtp->seq_number<200 && extseq->split.lo>((1<<16) - 200)){			/* this is a check for sequence number looping */			extseq->split.lo=rtp->seq_number;			extseq->split.hi++;		}	}		/* check for possible telephone events */	if (rtp->paytype==session->rcv.telephone_events_pt){		queue_packet(&session->rtp.tev_rq,session->rtp.max_rq_size,mp,rtp,&i);		stats->discarded+=i;		ortp_global_stats.discarded+=i;		return;	}		/* check for possible payload type change, in order to update accordingly our clock-rate dependant	parameters */	if (session->hw_recv_pt!=rtp->paytype){		rtp_session_update_payload_type(session,rtp->paytype);	}		jitter_control_new_packet(&session->rtp.jittctl,rtp->timestamp,local_str_ts);	if (session->flags & RTP_SESSION_FIRST_PACKET_DELIVERED) {		/* detect timestamp important jumps in the future, to workaround stupid rtp senders */		if (RTP_TIMESTAMP_IS_NEWER_THAN(rtp->timestamp,session->rtp.rcv_last_ts+session->rtp.ts_jump)){			ortp_debug("rtp_parse: timestamp jump ?");			rtp_signal_table_emit2(&session->on_timestamp_jump,(long)&rtp->timestamp);		}		else if (RTP_TIMESTAMP_IS_STRICTLY_NEWER_THAN(session->rtp.rcv_last_ts,rtp->timestamp)){			/* don't queue packets older than the last returned packet to the application*/			/* Call timstamp jumb in case of			 * large negative Ts jump or if ts is set to 0			*/						if ( RTP_TIMESTAMP_IS_STRICTLY_NEWER_THAN(session->rtp.rcv_last_ts, rtp->timestamp + session->rtp.ts_jump) ){				ortp_warning("rtp_parse: negative timestamp jump");				rtp_signal_table_emit2(&session->on_timestamp_jump,							(long)&rtp->timestamp);			}			ortp_debug("rtp_parse: discarding too old packet (ts=%i)",rtp->timestamp);			freemsg(mp);			stats->outoftime++;			ortp_global_stats.outoftime++;			return;		}	}		queue_packet(&session->rtp.rq,session->rtp.max_rq_size,mp,rtp,&i);	stats->discarded+=i;	ortp_global_stats.discarded+=i;}

⌨️ 快捷键说明

复制代码 Ctrl + C
搜索代码 Ctrl + F
全屏模式 F11
切换主题 Ctrl + Shift + D
显示快捷键 ?
增大字号 Ctrl + =
减小字号 Ctrl + -