📄 rtprecv.c
字号:
/*------------------------------------------------------------------------- * rtprecv.c - rtprecv *------------------------------------------------------------------------- */#include <stdlib.h>#include <sys/types.h>#include <sys/socket.h>#include <rtp.h>#include <unistd.h>#include <hash.h>/*------------------------------------------------------------------------ * rtprecv - receive and process an incoming RTP packet *------------------------------------------------------------------------ */intrtprecv(struct session *psn){ struct rtpln *pln; struct sockaddr_in from; int fromlen, len; struct rtp *prtp; struct stream *pstm; bool head; int rv; pln = (struct rtpln *) bufpoolgetbuf(&psn->sn_bpool); if (pln == NULL) { rtppostevent(psn, EVENT_NOBUFS, 0, NULL, 0); return ERROR; } fromlen = sizeof(from); len = recvfrom(psn->sn_rtpfd, (char *) &pln->rln_rtp, psn->sn_bpool.bp_size - sizeof(struct rtpln), 0, (struct sockaddr *) &from, &fromlen); if (len < 1) { bufpoolfreebuf(pln); return ERROR; } pln->rln_len = len; /* * Extract source address as RTCP destination for unicast session * if the RTCP destination has not already been learned. */ if (psn->sn_rtcpto.sin_addr.s_addr == RTP_INADDRUNINITIALIZED) { psn->sn_rtcpto.sin_addr = from.sin_addr; } /* * Get pointer to RTP header. */ prtp = &pln->rln_rtp; /* * Convert header byte order and check version. */ rtpntoh(prtp); if (prtp->rtp_ver != RTP_CURRVERS) { bufpoolfreebuf(pln); return ERROR; } /* * Lookup stream by SSRC. */ pstm = rtpgetstream(psn, prtp->rtp_ssrc); /* * Create new stream if newly discovered source. */ if (pstm == NULL) { if ((pstm = rtpnewstream(psn, prtp->rtp_ssrc)) == NULL) { bufpoolfreebuf(pln); return ERROR; } } else { if (RTP_INACTIVE(pstm->stm_inactive)) { rtpreleasestream(psn, pstm); bufpoolfreebuf(pln); return ERROR; } } if (pstm->stm_ip.sin_addr.s_addr == RTP_INADDRUNINITIALIZED) pstm->stm_ip = from; /* * Update stream's statistics and check if stream is still * on probation. */ if (rtpupdate(psn, pstm, prtp) == ERROR) { rtpreleasestream(psn, pstm); bufpoolfreebuf(pln); return ERROR; } /* * Enqueue the packet if stream is ``on''. */ if (pstm->stm_enqueue == TRUE) { /* * Tag packet with *extended* sequence number. */ pln->rln_seq = ((pstm->stm_roll) << 16) | prtp->rtp_seq; pthread_mutex_lock(&pstm->stm_queue.rq_mutex); rv = _rtpqinsert(&pstm->stm_queue, pln, &head); pthread_mutex_unlock(&pstm->stm_queue.rq_mutex); /* * Post event if packet added at front of queue. * Doing so will notify a thread waiting on an * empty queue. */ if (rv == OK) { rtppostevent(psn, (head == TRUE ? EVENT_RTP_HEAD : EVENT_RTP), pstm->stm_ssrc, NULL, 0); rtpreleasestream(psn, pstm); return OK; } else { /* * Free packet and return error if * enqueue operation failed. */ rtpreleasestream(psn, pstm); bufpoolfreebuf(pln); return ERROR; } } rtpreleasestream(psn, pstm); /* * Post an RTP-packet-received event. */ rtppostevent(psn, EVENT_RTP, prtp->rtp_ssrc, NULL, 0); bufpoolfreebuf(pln); return OK;}
⌨️ 快捷键说明
复制代码
Ctrl + C
搜索代码
Ctrl + F
全屏模式
F11
切换主题
Ctrl + Shift + D
显示快捷键
?
增大字号
Ctrl + =
减小字号
Ctrl + -