Skip to content
Snippets Groups Projects
RTCPREMBPacket.java 7.76 KiB
Newer Older
/*
 * Jitsi, the OpenSource Java VoIP and Instant Messaging client.
 *
 * Distributable under LGPL license.
 * See terms of license at gnu.org.
 */
package org.jitsi.impl.neomedia.rtcp;

import java.io.*;
import java.util.*;

import net.sf.fmj.media.rtp.*;

import org.jitsi.impl.neomedia.*;

/**
 * Created by gp on 6/24/14.
 *
 * 0                   1                   2                   3
 * 0 1 2 3 4 5 6 7 8 9 0 1 2 3 4 5 6 7 8 9 0 1 2 3 4 5 6 7 8 9 0 1
 * +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
 * |V=2|P| FMT=15  |   PT=206      |             length            |
 * +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
 * |                  SSRC of packet sender                        |
 * +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
 * |                  SSRC of media source                         |
 * +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
 * |  Unique identifier 'R' 'E' 'M' 'B'                            |
 * +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
 * |  Num SSRC     | BR Exp    |  BR Mantissa                      |
 * +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
 * |   SSRC feedback                                               |
 * +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
 * |  ...                                                          |
 */
public class RTCPREMBPacket extends RTCPFBPacket
{
    /**
     * The exponential scaling of the mantissa for the maximum total media
     * bit rate value, ignoring all packet overhead.
     */
    public int exp;

    /**
     * The mantissa of the maximum total media bit rate (ignoring all packet
     * overhead) that the sender of the REMB estimates.  The BR is the estimate
     * of the traveled path for the SSRCs reported in this message.
     */
    public int mantissa;

    /**
     * one or more SSRC entries which this feedback message applies to.
     */
    public long[] dest;

    public RTCPREMBPacket(
            long senderSSRC,
            long mediaSSRC,
            int exp,
            int mantissa,
            long[] dest)
    {
        super(FMT, PSFB, senderSSRC, mediaSSRC);
    public RTCPREMBPacket(
            long senderSSRC,
            long mediaSSRC,
            long bitrate,
            long[] dest)
    {
        super(FMT, PSFB, senderSSRC, mediaSSRC);

        // 6 bit Exp
        // 18 bit mantissa
        this.exp = 0;
        for(int i=0; i<64; i++)
        {
            if(bitrate <= (0x3ffff << i))
            {
                this.exp = i;
                break;
            }
        }

        /* type of bitrate is an unsigned int (32 bits) */
        this.mantissa = (((int)bitrate) >> this.exp);
        this.dest = dest;
    }

    public RTCPREMBPacket(RTCPCompoundPacket base)
    {
        super(base);
    }

/*
    0                   1                   2                   3
    0 1 2 3 4 5 6 7 8 9 0 1 2 3 4 5 6 7 8 9 0 1 2 3 4 5 6 7 8 9 0 1
   +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
   |V=2|P| FMT=15  |   PT=206      |             length            |
   +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
   |                  SSRC of packet sender                        |
   +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
   |                  SSRC of media source                         |
   +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
   |  Unique identifier 'R' 'E' 'M' 'B'                            |
   +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
   |  Num SSRC     | BR Exp    |  BR Mantissa                      |
   +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
   |   SSRC feedback                                               |
   +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
   |  ...                                                          |

 */

    @Override
    public void assemble(DataOutputStream dataoutputstream) throws IOException
    {
        int len = this.calcLength();
        byte[] buf = new byte[len];
        int off = 0;

        /*
         * version (V): (2 bits):   This field identifies the RTP version.  The
         *     current version is 2.
         * padding (P) (1 bit):   If set, the padding bit indicates that the
         *     packet contains additional padding octets at the end that
         *     are not part of the control information but are included
         *     in the length field.  Always 0.
         * Feedback message type (FMT) (5 bits):  This field identifies the type
         *     of the FB message and is interpreted relative to the type
         *     (transport layer, payload- specific, or application layer
         *     feedback).  Always 15, application layer feedback
         *     message.
         */
        buf[off++] = (byte) (0x8F);

        /*
         * Payload type (PT) (8 bits):   This is the RTCP packet type that
         *     identifies the packet as being an RTCP FB message.
         *     Always PSFB (206), Payload-specific FB message.
         */
        buf[off++] = (byte) (0xCE);

        // Length (16 bits):  The length of this packet in 32-bit words minus
        // one, including the header and any padding. This is in
        // line with the definition of the length field used in RTCP
        // sender and receiver reports
        int rtcpPacketLength = len / 4 - 1;
        buf[off++] = (byte) ((rtcpPacketLength & 0xFF00) >> 8);
        buf[off++] = (byte) (rtcpPacketLength & 0x00FF);

        // SSRC of packet sender: 32 bits
        RTCPFeedbackMessagePacket.writeSSRC(senderSSRC, buf, off);
        off += 4;

        //  SSRC of media source (32 bits):  Always 0;
        RTCPFeedbackMessagePacket.writeSSRC(0L, buf, off);
        off += 4;

        // Unique identifier (32 bits):  Always 'R' 'E' 'M' 'B' (4 ASCII
        // characters).
        buf[off++] = (byte) 'R';
        buf[off++] = (byte) 'E';
        buf[off++] = (byte) 'M';
        buf[off++] = (byte) 'B';

        // Num SSRC (8 bits):  Number of SSRCs in this message.
        buf[off++] =
                (byte) ((dest != null && dest.length != 0)
                        ? dest.length
                        : 0);

        // BR Exp (6 bits):   The exponential scaling of the mantissa for the
        // maximum total media bit rate value, ignoring all packet
        // overhead.

        // BR Mantissa (18 bits):   The mantissa of the maximum total media bit
        // rate (ignoring all packet overhead) that the sender of
        // the REMB estimates.
        buf[off++] = (byte) (((exp & 0x3f) << 2) | (mantissa & 0x30000) >> 16);
        buf[off++] = (byte) ((mantissa & 0xff00) >> 8);
        buf[off++] = (byte) (mantissa & 0xff);

        // SSRC feedback (32 bits)  Consists of one or more SSRC entries which
        // this feedback message applies to.
        if (dest != null && dest.length != 0)
            for (int i = 0; i < dest.length; i++)
            {
                RTCPFeedbackMessagePacket.writeSSRC(dest[i], buf, off);
                off += 4;
            }

        dataoutputstream.write(buf, 0, len);
    }

    @Override
    public int calcLength()
    {
        int len = 20; // 20 bytes header + standard data

        if (dest != null)
            len += dest.length * 4;

        return len;
    }

    @Override
    public String toString()
    {
        return "\tRTCP REMB packet from sync source " + senderSSRC
                + "\n\t\tfor sync sources: " + Arrays.toString(dest)
                + "\n\t\tBR Exp: " + exp
                + "\n\t\tBR Mantissa: " + mantissa;
    }
}