WebRTC RTP Transport

Unofficial Proposal Draft,

More details about this document
This version:
https://github.com/w3c/webrtc-rtptransport
Latest published version:
https://www.w3.org/TR/webrtc-rtptransport/
Feedback:
public-webrtc@w3.org with subject line “[webrtc-rtptransport] … message topic …” (archives)
GitHub
Editors:
(Microsoft Corporation)
(Google)
(Microsoft Corporation)
Participate:
Git Repository.
File an issue.
Version History:
https://github.com/w3c/webrtc-rtptransport/commits

Abstract

The WebRTC-RtpTransport API allows web applications to send and receive RTP/RTCP packets using the protocol defined in [RFC3550].

Status of this document

1. Introduction

This section is non-normative.

This specification uses [RFC3550] to send and receive RTP and RTCP packets.

Note: The API presented in this specification represents a preliminary proposal based on work-in-progress within the W3C WEBRTC WG. As a result, the API is likely to change significantly going forward.

2. Conformance

As well as sections marked as non-normative, all authoring guidelines, diagrams, examples, and notes in this specification are non-normative. Everything else in this specification is normative.

The key words "MUST", "MUST NOT", "REQUIRED", "SHALL", "SHALL NOT", "SHOULD", "SHOULD NOT", "RECOMMENDED", "NOT RECOMMENDED", "MAY", and "OPTIONAL" are to be interpreted as described in [RFC2119] and [RFC8174] when, and only when, they appear in all capitals, as shown here.

This specification defines conformance criteria that apply to a single product: the user agent that implements the interfaces that it contains.

Conformance requirements phrased as algorithms or specific steps may be implemented in any manner, so long as the end result is equivalent. (In particular, the algorithms defined in this specification are intended to be easy to follow, and not intended to be performant.)

Implementations that use ECMAScript to implement the APIs defined in this specification MUST implement them in a manner consistent with the ECMAScript Bindings defined in the Web IDL specification [WEBIDL], as this specification uses that specification and terminology.

3. Terminology

Terminology definitions go here.

4. Interface

[Exposed=(Window)]
interface RTCRtpTransport {
  attribute EventHandler onpacketizedrtpavailable;
  sequence<RTCRtpPacket> readPacketizedRtp(unsigned long );
  // BYOP: Copies the read packets into dstPackets, returning the number of packets written (up to dstPackets.length).
  unsigned long readPacketizedRtp(sequence<RTCRtpPacket> );

  attribute EventHandler onsentrtp;
  // Batch interface to read RTCRtpSent notifications.
  sequence<RTCRtpSent> readSentRtp(long );

  attribute EventHandler onreceivedrtpacks;  // No payload. Use readReceivedRtpAcks
  // Batch interface to read RTCRtpAcks as an alternative to onrtpacksreceived.
  sequence<RTCRtpAcks> readReceivedRtpAcks(long );

  readonly attribute unsigned long bandwidthEstimate;  // bps
  readonly attribute unsigned long allocatedBandwidth;  // bps
  attribute unsigned long customAllocatedBandwidth;  // writable
  // Means "when doing bitrate allocation and rate control, don’t use more than this"
  attribute unsigned long customMaxBandwidth;
  // Means "make each packet smaller by this much so I can put custom stuff in each packet"
  attribute unsigned long customPerPacketOverhead;
};

4.1. Attributes

, of type EventHandler
, of type EventHandler
, of type EventHandler
, of type unsigned long, readonly
, of type unsigned long, readonly
, of type unsigned long
, of type unsigned long
, of type unsigned long

4.2. Methods

5. Objects

5.1. Interface

[Exposed=(Window)]
interface RTCRtpPacket {
  (RTCRtpPacketInit );
  // BYOP: Construct a packet that can be used as destination for writing read packets.
  ();

  attribute boolean marker;
  attribute octet payloadType;
  attribute unsigned short sequenceNumber;
  attribute unsigned long timestamp;
  attribute unsigned long ssrc;
  sequence<unsigned long> getCsrcs();
  undefined setCsrcs(sequence<unsigned long> );
  sequence<RTCRtpHeaderExtension> getHeaderExtensions();
  undefined setHeaderExtensions(sequence<RTCRtpHeaderExtension> );
  attribute unsigned long paddingBytes;

  attribute AllowSharedBufferSource ;

  // OPTIONAL: Extra information that may be useful to know
  attribute DOMHighResTimeStamp? receivedTime;
  attribute unsigned long? sequenceNumberRolloverCount;
};
, of type boolean
, of type octet
, of type unsigned short
, of type unsigned long
, of type unsigned long
, of type unsigned long
, of type DOMHighResTimeStamp, nullable
, of type unsigned long, nullable

5.2. Interface

[Exposed=(Window)]
interface RTCRtpHeaderExtension {
  (RTCRtpHeaderExtensionInit );
  readonly attribute DOMString uri;
  readonly attribute unsigned long valueByteLength;

  // Write the extension’s value to the specified (Shared-)ArrayBuffer/ArrayBufferView,
  // allowing for BYOB. Throws a TypeError if |destination| is smaller than
  // |valueByteLength|.
  undefined copyValueTo(AllowSharedBufferSource );
};
, of type DOMString, readonly
, of type unsigned long, readonly

5.3. Dictionary

dictionary RTCRtpHeaderExtensionInit {
  required DOMString uri;
  required AllowSharedBufferSource value;
};
, of type DOMString
, of type AllowSharedBufferSource

5.4. Dictionary

dictionary RTCRtpPacketInit {
  boolean marker = false;
  required octet payloadType;
  required unsigned long timestamp;
  sequence<unsigned long> csrcs = [];
  // Cannot be MID, RID, or congestion control sequence number
  sequence<RTCRtpHeaderExtensionInit> headerExtensions = [];
  required AllowSharedBufferSource payload;
  // Causes padding bit to be set and padding added when serializing if > 0.
  unsigned long paddingBytes = 0;
};
, of type boolean, defaulting to false
, of type octet
, of type unsigned long
, of type sequence<unsigned long>, defaulting to []
, of type sequence<RTCRtpHeaderExtensionInit>, defaulting to []
, of type AllowSharedBufferSource
, of type unsigned long, defaulting to 0

5.5. Interface

[Exposed=(Window)]
interface RTCRtpAcks {
  sequence<RTCRtpAck> getAcks();
  readonly attribute unsigned long long remoteSendTimestamp;
  readonly attribute DOMHighResTimeStamp receivedTime;
  readonly attribute RTCExplicitCongestionNotification explicitCongestionNotification;  // AKA "ECN"
};
, of type unsigned long long, readonly
, of type DOMHighResTimeStamp, readonly
, of type RTCExplicitCongestionNotification, readonly

5.6.

[Exposed=(Window)]
interface RTCRtpAck {
  // Correlated with RtpSent.ackId
  readonly attribute unsigned long long ackId; 
  readonly attribute unsigned long long remoteReceiveTimestamp;
};
, of type unsigned long long, readonly
, of type unsigned long long, readonly

5.7.

enum RTCExplicitCongestionNotification {
  // ECT = ECN-Capable Transport
  ,  // AKA "Not-ECT";  Bits: 00
  ,  // AKA "ECT(1)" or "Scalable" or "L4S" ; Bits: 01
  , // AKA "ECT(0)" or "Classic" or "not L4S"; Bits: 10
   // AKA "CE" or "ECN-marked" or "marked"; Bits: 11
};

6. Interface

[Exposed=(Window,Worker), Transferable]
interface RTCRtpPacketSender {
  readonly attribute DOMString? mid;
  readonly attribute DOMString? rid;
  readonly attribute unsigned long ssrc;
  readonly attribute unsigned long rtxSsrc;

  attribute EventHandler onpacketizedrtp;
  sequence<RTCRtpPacket> readPacketizedRtp(unsigned long );
  // BYOP: Copies the read packets into dstPackets, returning the number of packets written (up to dstPackets.length).
  unsigned long readPacketizedRtp(sequence<RTCRtpPacket> );

  // https://github.com/w3c/webrtc-rtptransport/issues/32
  undefined sendRtp(RTCRtpPacket );
  Promise<RTCRtpSendResult> sendRtp(RTCRtpPacketInit , optional RTCRtpSendOptions  = {});

  // Amount allocated by the browser
  readonly attribute unsigned long ;
};
, of type DOMString, readonly, nullable
, of type DOMString, readonly, nullable
, of type unsigned long, readonly
, of type unsigned long, readonly
, of type EventHandler

6.1. Interface

[Exposed=(Window)]
interface RTCRtpSendResult {
  readonly attribute RTCRtpSent? sent;
  readonly attribute RTCRtpUnsentReason? unsent;
};
, of type RTCRtpSent, readonly, nullable
, of type RTCRtpUnsentReason, readonly, nullable

6.2. Interface

[Exposed=(Window)]
interface RTCRtpSent {
  readonly attribute DOMHighResTimeStamp time;

  // Can be correlated with acks
  readonly attribute unsigned long long? ackId;
  readonly attribute unsigned long long size;
};
, of type DOMHighResTimeStamp, readonly
, of type unsigned long long, readonly, nullable
, of type unsigned long long, readonly

6.3. Enum

enum RTCRtpUnsentReason {
  ,
  ,
};

6.4. Dictionary

dictionary RTCRtpSendOptions {
  DOMHighResTimeStamp sendTime;
};
, of type DOMHighResTimeStamp

7. Interface

[Exposed=(Window,Worker), Transferable]
interface RTCRtpPacketReceiver {
  readonly attribute DOMString? mid;
  readonly attribute DOMString? rid;  // Unique to RTCRtpPacketReceiver (scoped to MID)
  sequence<unsigned long> getSsrcs();
  sequence<unsigned long> getRtxSsrcs();

  attribute EventHandler onreceivedrtp;
  sequence<RTCRtpPacket> readReceivedRtp(long );
  // BYOP: Copies the read packets into dstPackets, returning the number of packets written (up to dstPackets.length).
  unsigned long readReceivedRtp(sequence<RTCRtpPacket> );

  undefined receiveRtp(RTCRtpPacket );
};
, of type DOMString, readonly, nullable
, of type DOMString, readonly, nullable
, of type EventHandler

8. Interface Extensions

8.1. RTCPeerConnection Interface Extensions

partial interface RTCPeerConnection {
  // There may be an RtpTransport with no RtpSenders and no RtpReceivers.
  readonly attribute RTCRtpTransport? rtpTransport;
};
, of type RTCRtpTransport, readonly, nullable

8.2. RTCRtpSender Interface Extensions

partial interface RTCRtpSender {
  // shared between RTCRtpSenders in the same BUNDLE group
  readonly attribute RTCRtpTransport? rtpTransport;
  Promise<RTCRtpPacketSender> replacePacketSender();
};
, of type RTCRtpTransport, readonly, nullable

8.3. RTCConfiguration Dictionary Extensions

partial dictionary RTCConfiguration {
  // Means "continue to encode and packetize packets, but don’t send them.
  // Instead give them to me via onpacketizedrtpavailable/readPacketizedRtp
  // and I will send them."
  // TODO: Think of a better name
  boolean customPacer;
};
, of type boolean

8.4. RTCRtpReceiver Interface Extensions

partial interface RTCRtpReceiver {
  // shared between RTCRtpSenders in the same BUNDLE group
  readonly attribute RTCRtpTransport? rtpTransport;
  Promise<RTCRtpPacketReceiver> replacePacketReciever();
};
, of type RTCRtpTransport, readonly, nullable

9. Privacy and security considerations

9.1. Confidentiality of Communications

9.2. State Persistence

9.3. Protocol Security

9.4. Fingerprinting and Tracking

10. Examples

See the explainer document.

11. Acknowledgements

The editors wish to thank the WEBRTC Working Group chairs and Team Contact for their support.

Index

Terms defined by this specification

Terms defined by reference

References

Normative References

[HR-TIME-3]
Yoav Weiss. High Resolution Time. URL: https://w3c.github.io/hr-time/
[HTML]
Anne van Kesteren; et al. HTML Standard. Living Standard. URL: https://html.spec.whatwg.org/multipage/
[RFC2119]
S. Bradner. Key words for use in RFCs to Indicate Requirement Levels. March 1997. Best Current Practice. URL: https://datatracker.ietf.org/doc/html/rfc2119
[RFC3550]
H. Schulzrinne; et al. RTP: A Transport Protocol for Real-Time Applications. July 2003. RFC. URL: https://datatracker.ietf.org/doc/html/rfc3550
[RFC8174]
B. Leiba. Ambiguity of Uppercase vs Lowercase in RFC 2119 Key Words. May 2017. Best Current Practice. URL: https://www.rfc-editor.org/rfc/rfc8174
[WEBIDL]
Edgar Chen; Timothy Gu. Web IDL Standard. Living Standard. URL: https://webidl.spec.whatwg.org/
[WEBRTC]
Cullen Jennings; et al. WebRTC: Real-Time Communication in Browsers. URL: https://w3c.github.io/webrtc-pc/

IDL Index

[Exposed=(Window)]
interface RTCRtpTransport {
  attribute EventHandler onpacketizedrtpavailable;
  sequence<RTCRtpPacket> readPacketizedRtp(unsigned long maxNumberOfPackets);
  // BYOP: Copies the read packets into dstPackets, returning the number of packets written (up to dstPackets.length).
  unsigned long readPacketizedRtp(sequence<RTCRtpPacket> dstPackets);

  attribute EventHandler onsentrtp;
  // Batch interface to read RTCRtpSent notifications.
  sequence<RTCRtpSent> readSentRtp(long maxCount);

  attribute EventHandler onreceivedrtpacks;  // No payload. Use readReceivedRtpAcks
  // Batch interface to read RTCRtpAcks as an alternative to onrtpacksreceived.
  sequence<RTCRtpAcks> readReceivedRtpAcks(long maxCount);

  readonly attribute unsigned long bandwidthEstimate;  // bps
  readonly attribute unsigned long allocatedBandwidth;  // bps
  attribute unsigned long customAllocatedBandwidth;  // writable
  // Means "when doing bitrate allocation and rate control, don’t use more than this"
  attribute unsigned long customMaxBandwidth;
  // Means "make each packet smaller by this much so I can put custom stuff in each packet"
  attribute unsigned long customPerPacketOverhead;
};


[Exposed=(Window)]
interface RTCRtpPacket {
  constructor(RTCRtpPacketInit init);
  // BYOP: Construct a packet that can be used as destination for writing read packets.
  constructor();

  attribute boolean marker;
  attribute octet payloadType;
  attribute unsigned short sequenceNumber;
  attribute unsigned long timestamp;
  attribute unsigned long ssrc;
  sequence<unsigned long> getCsrcs();
  undefined setCsrcs(sequence<unsigned long> csrcs);
  sequence<RTCRtpHeaderExtension> getHeaderExtensions();
  undefined setHeaderExtensions(sequence<RTCRtpHeaderExtension> headerExtensions);
  attribute unsigned long paddingBytes;

  attribute AllowSharedBufferSource payload;

  // OPTIONAL: Extra information that may be useful to know
  attribute DOMHighResTimeStamp? receivedTime;
  attribute unsigned long? sequenceNumberRolloverCount;
};

[Exposed=(Window)]
interface RTCRtpHeaderExtension {
  constructor(RTCRtpHeaderExtensionInit init);
  readonly attribute DOMString uri;
  readonly attribute unsigned long valueByteLength;

  // Write the extension’s value to the specified (Shared-)ArrayBuffer/ArrayBufferView,
  // allowing for BYOB. Throws a TypeError if |destination| is smaller than
  // |valueByteLength|.
  undefined copyValueTo(AllowSharedBufferSource destination);
};

dictionary RTCRtpHeaderExtensionInit {
  required DOMString uri;
  required AllowSharedBufferSource value;
};

dictionary RTCRtpPacketInit {
  boolean marker = false;
  required octet payloadType;
  required unsigned long timestamp;
  sequence<unsigned long> csrcs = [];
  // Cannot be MID, RID, or congestion control sequence number
  sequence<RTCRtpHeaderExtensionInit> headerExtensions = [];
  required AllowSharedBufferSource payload;
  // Causes padding bit to be set and padding added when serializing if > 0.
  unsigned long paddingBytes = 0;
};

[Exposed=(Window)]
interface RTCRtpAcks {
  sequence<RTCRtpAck> getAcks();
  readonly attribute unsigned long long remoteSendTimestamp;
  readonly attribute DOMHighResTimeStamp receivedTime;
  readonly attribute RTCExplicitCongestionNotification explicitCongestionNotification;  // AKA "ECN"
};

[Exposed=(Window)]
interface RTCRtpAck {
  // Correlated with RtpSent.ackId
  readonly attribute unsigned long long ackId; 
  readonly attribute unsigned long long remoteReceiveTimestamp;
};

enum RTCExplicitCongestionNotification {
  // ECT = ECN-Capable Transport
  "unset",  // AKA "Not-ECT";  Bits: 00
  "scalable-congestion-not-experienced",  // AKA "ECT(1)" or "Scalable" or "L4S" ; Bits: 01
  "classic-congestion-not-experienced", // AKA "ECT(0)" or "Classic" or "not L4S"; Bits: 10
  "congestion-experienced" // AKA "CE" or "ECN-marked" or "marked"; Bits: 11
};

[Exposed=(Window,Worker), Transferable]
interface RTCRtpPacketSender {
  readonly attribute DOMString? mid;
  readonly attribute DOMString? rid;
  readonly attribute unsigned long ssrc;
  readonly attribute unsigned long rtxSsrc;

  attribute EventHandler onpacketizedrtp;
  sequence<RTCRtpPacket> readPacketizedRtp(unsigned long maxNumberOfPackets);
  // BYOP: Copies the read packets into dstPackets, returning the number of packets written (up to dstPackets.length).
  unsigned long readPacketizedRtp(sequence<RTCRtpPacket> dstPackets);

  // https://github.com/w3c/webrtc-rtptransport/issues/32
  undefined sendRtp(RTCRtpPacket packet);
  Promise<RTCRtpSendResult> sendRtp(RTCRtpPacketInit packetInit, optional RTCRtpSendOptions options = {});

  // Amount allocated by the browser
  readonly attribute unsigned long allocatedBandwidth;
};

[Exposed=(Window)]
interface RTCRtpSendResult {
  readonly attribute RTCRtpSent? sent;
  readonly attribute RTCRtpUnsentReason? unsent;
};

[Exposed=(Window)]
interface RTCRtpSent {
  readonly attribute DOMHighResTimeStamp time;

  // Can be correlated with acks
  readonly attribute unsigned long long? ackId;
  readonly attribute unsigned long long size;
};

enum RTCRtpUnsentReason {
  "overuse",
  "transport-unavailable",
};

dictionary RTCRtpSendOptions {
  DOMHighResTimeStamp sendTime;
};

[Exposed=(Window,Worker), Transferable]
interface RTCRtpPacketReceiver {
  readonly attribute DOMString? mid;
  readonly attribute DOMString? rid;  // Unique to RTCRtpPacketReceiver (scoped to MID)
  sequence<unsigned long> getSsrcs();
  sequence<unsigned long> getRtxSsrcs();

  attribute EventHandler onreceivedrtp;
  sequence<RTCRtpPacket> readReceivedRtp(long maxNumberOfPackets);
  // BYOP: Copies the read packets into dstPackets, returning the number of packets written (up to dstPackets.length).
  unsigned long readReceivedRtp(sequence<RTCRtpPacket> dstPackets);

  undefined receiveRtp(RTCRtpPacket packet);
};

partial interface RTCPeerConnection {
  // There may be an RtpTransport with no RtpSenders and no RtpReceivers.
  readonly attribute RTCRtpTransport? rtpTransport;
};

partial interface RTCRtpSender {
  // shared between RTCRtpSenders in the same BUNDLE group
  readonly attribute RTCRtpTransport? rtpTransport;
  Promise<RTCRtpPacketSender> replacePacketSender();
};

partial dictionary RTCConfiguration {
  // Means "continue to encode and packetize packets, but don’t send them.
  // Instead give them to me via onpacketizedrtpavailable/readPacketizedRtp
  // and I will send them."
  // TODO: Think of a better name
  boolean customPacer;
};

partial interface RTCRtpReceiver {
  // shared between RTCRtpSenders in the same BUNDLE group
  readonly attribute RTCRtpTransport? rtpTransport;
  Promise<RTCRtpPacketReceiver> replacePacketReciever();
};