Ignore:
Timestamp:
Sep 8, 2011 6:47:28 AM (13 years ago)
Author:
bennylp
Message:

Slightly more clever RTP transport remote address switch: transmission won't switch as long as we're receiving RTP/RTCP packets from the "correct" remote address, to avoid constantly switching to a new source address. Also when the remote peer address is in probation, RT(C)P packets from that address will be silently discarded, to avoid playback confusion. This closes #1366.

File:
1 edited

Legend:

Unmodified
Added
Removed
  • pjproject/branches/1.x/pjmedia/src/pjmedia/transport_ice.c

    r3553 r3745  
    7373    pj_sockaddr          rtcp_src_addr; /**< Actual source RTCP address.    */ 
    7474    unsigned             rtp_src_cnt;   /**< How many pkt from this addr.   */ 
     75    unsigned             rtcp_src_cnt;  /**< How many pkt from this addr.   */ 
    7576 
    7677    unsigned             tx_drop_pct;   /**< Percent of tx pkts to drop.    */ 
     
    15001501    if (tp_ice->use_ice || tp_ice->rtp_src_cnt) { 
    15011502        info->src_rtp_name  = tp_ice->rtp_src_addr; 
     1503    } 
     1504    if (tp_ice->use_ice || tp_ice->rtcp_src_cnt) { 
    15021505        info->src_rtcp_name = tp_ice->rtcp_src_addr; 
    15031506    } 
     
    15651568    tp_ice->rtcp_src_addr = tp_ice->remote_rtcp; 
    15661569    tp_ice->rtp_src_cnt = 0; 
     1570    tp_ice->rtcp_src_cnt = 0; 
    15671571 
    15681572    return PJ_SUCCESS; 
     
    16401644{ 
    16411645    struct transport_ice *tp_ice; 
     1646    pj_bool_t discard = PJ_FALSE; 
    16421647 
    16431648    tp_ice = (struct transport_ice*) pj_ice_strans_get_user_data(ice_st); 
     
    16551660        } 
    16561661 
    1657         (*tp_ice->rtp_cb)(tp_ice->stream, pkt, size); 
    1658  
    16591662        /* See if source address of RTP packet is different than the  
    16601663         * configured address, and switch RTP remote address to  
     
    16631666         */ 
    16641667        if (!tp_ice->use_ice) { 
    1665  
    1666             /* Increment counter and avoid zero */ 
    1667             if (++tp_ice->rtp_src_cnt == 0)  
    1668                 tp_ice->rtp_src_cnt = 1; 
    1669  
    1670             if (pj_sockaddr_cmp(&tp_ice->remote_rtp, src_addr) != 0) { 
     1668            pj_bool_t enable_switch = 
     1669                    ((tp_ice->options & PJMEDIA_ICE_NO_SRC_ADDR_CHECKING)==0); 
     1670 
     1671            if (!enable_switch || 
     1672                pj_sockaddr_cmp(&tp_ice->remote_rtp, src_addr) == 0) 
     1673            { 
     1674                /* Don't switch while we're receiving from remote_rtp */ 
     1675                tp_ice->rtp_src_cnt = 0; 
     1676            } else { 
     1677 
     1678                ++tp_ice->rtp_src_cnt; 
    16711679 
    16721680                /* Check if the source address is recognized. */ 
     
    16761684                    /* Reset counter */ 
    16771685                    tp_ice->rtp_src_cnt = 0; 
     1686                    discard = PJ_TRUE; 
    16781687                } 
    16791688 
    1680                 if ((tp_ice->options & PJMEDIA_ICE_NO_SRC_ADDR_CHECKING)==0 && 
    1681                     tp_ice->rtp_src_cnt >= PJMEDIA_RTP_NAT_PROBATION_CNT)  
    1682                 { 
     1689                if (tp_ice->rtp_src_cnt < PJMEDIA_RTP_NAT_PROBATION_CNT) { 
     1690                    discard = PJ_TRUE; 
     1691                } else { 
    16831692                    char addr_text[80]; 
    16841693 
     
    17091718 
    17101719                        PJ_LOG(4,(tp_ice->base.name, 
    1711                                   "Remote RTCP address switched to %s", 
     1720                                  "Remote RTCP address switched to predicted " 
     1721                                  "address %s", 
    17121722                                  pj_sockaddr_print(&tp_ice->remote_rtcp,  
    17131723                                                    addr_text, 
     
    17171727            } 
    17181728        } 
     1729 
     1730        if (!discard) 
     1731            (*tp_ice->rtp_cb)(tp_ice->stream, pkt, size); 
     1732 
    17191733    } else if (comp_id==2 && tp_ice->rtcp_cb) { 
    1720         (*tp_ice->rtcp_cb)(tp_ice->stream, pkt, size); 
    17211734 
    17221735        /* Check if RTCP source address is the same as the configured 
     
    17251738         */ 
    17261739        if (!tp_ice->use_ice && 
    1727             pj_sockaddr_cmp(&tp_ice->remote_rtcp, src_addr) != 0) 
     1740            (tp_ice->options & PJMEDIA_ICE_NO_SRC_ADDR_CHECKING)==0) 
    17281741        { 
    1729             pj_sockaddr_cp(&tp_ice->rtcp_src_addr, src_addr); 
    1730  
    1731             if ((tp_ice->options & PJMEDIA_ICE_NO_SRC_ADDR_CHECKING)==0) { 
     1742            if (pj_sockaddr_cmp(&tp_ice->remote_rtcp, src_addr) == 0) { 
     1743                tp_ice->rtcp_src_cnt = 0; 
     1744            } else { 
    17321745                char addr_text[80]; 
    17331746 
    1734                 pj_sockaddr_cp(&tp_ice->remote_rtcp, src_addr); 
    1735  
    1736                 pj_assert(tp_ice->addr_len == pj_sockaddr_get_len(src_addr)); 
    1737  
    1738                 PJ_LOG(4,(tp_ice->base.name, 
    1739                           "Remote RTCP address switched to %s", 
    1740                           pj_sockaddr_print(&tp_ice->remote_rtcp, addr_text, 
    1741                                             sizeof(addr_text), 3))); 
     1747                ++tp_ice->rtcp_src_cnt; 
     1748                if (tp_ice->rtcp_src_cnt < PJMEDIA_RTCP_NAT_PROBATION_CNT) { 
     1749                    discard = PJ_TRUE; 
     1750                } else { 
     1751                    tp_ice->rtcp_src_cnt = 0; 
     1752                    pj_sockaddr_cp(&tp_ice->rtcp_src_addr, src_addr); 
     1753                    pj_sockaddr_cp(&tp_ice->remote_rtcp, src_addr); 
     1754 
     1755                    pj_assert(tp_ice->addr_len==pj_sockaddr_get_len(src_addr)); 
     1756 
     1757                    PJ_LOG(4,(tp_ice->base.name, 
     1758                              "Remote RTCP address switched to %s", 
     1759                              pj_sockaddr_print(&tp_ice->remote_rtcp, 
     1760                                                addr_text, sizeof(addr_text), 
     1761                                                3))); 
     1762                } 
    17421763            } 
    17431764        } 
     1765 
     1766        if (!discard) 
     1767            (*tp_ice->rtcp_cb)(tp_ice->stream, pkt, size); 
    17441768    } 
    17451769 
Note: See TracChangeset for help on using the changeset viewer.