OLD | NEW |
(Empty) | |
| 1 // Copyright 2016 The Chromium Authors. All rights reserved. |
| 2 // Use of this source code is governed by a BSD-style license that can be |
| 3 // found in the LICENSE file. |
| 4 |
| 5 #include "extensions/renderer/api/display_source/wifi_display/wifi_display_media
_packetizer.h" |
| 6 |
| 7 #include <utility> |
| 8 |
| 9 #include "base/logging.h" |
| 10 #include "base/rand_util.h" |
| 11 #include "crypto/random.h" |
| 12 #include "extensions/renderer/api/display_source/wifi_display/wifi_display_eleme
ntary_stream_info.h" |
| 13 |
| 14 namespace extensions { |
| 15 namespace { |
| 16 const size_t kMaxTransportStreamPacketCount = 7u; |
| 17 const uint8_t kProtocolPayloadTypeMP2T = 33u; |
| 18 const uint8_t kProtocolVersion = 2u; |
| 19 } // namespace |
| 20 |
| 21 WiFiDisplayMediaDatagramPacket::WiFiDisplayMediaDatagramPacket() = default; |
| 22 |
| 23 WiFiDisplayMediaDatagramPacket::WiFiDisplayMediaDatagramPacket( |
| 24 WiFiDisplayMediaDatagramPacket&&) = default; |
| 25 |
| 26 WiFiDisplayMediaPacketizer::WiFiDisplayMediaPacketizer( |
| 27 const base::TimeDelta& delay_for_unit_time_stamps, |
| 28 std::vector<WiFiDisplayElementaryStreamInfo> stream_infos, |
| 29 const PacketizedCallback& on_packetized) |
| 30 : WiFiDisplayTransportStreamPacketizer(delay_for_unit_time_stamps, |
| 31 std::move(stream_infos)), |
| 32 on_packetized_media_datagram_packet_(on_packetized) { |
| 33 // Sequence numbers are mainly used for detecting lossed packets within one |
| 34 // RTP session. The initial value SHOULD be random (unpredictable) to make |
| 35 // known-plaintext attacks on encryption more difficult, in case the packets |
| 36 // flow through a translator that encrypts them. |
| 37 crypto::RandBytes(&sequence_number_, sizeof(sequence_number_)); |
| 38 base::RandBytes(&synchronization_source_identifier_, |
| 39 sizeof(synchronization_source_identifier_)); |
| 40 } |
| 41 |
| 42 WiFiDisplayMediaPacketizer::~WiFiDisplayMediaPacketizer() {} |
| 43 |
| 44 bool WiFiDisplayMediaPacketizer::OnPacketizedTransportStreamPacket( |
| 45 const WiFiDisplayTransportStreamPacket& transport_stream_packet, |
| 46 bool flush) { |
| 47 DCHECK(CalledOnValidThread()); |
| 48 |
| 49 if (media_datagram_packet_.empty()) { |
| 50 // Convert time to the number of 90 kHz ticks since some epoch. |
| 51 const uint64_t us = static_cast<uint64_t>( |
| 52 (base::TimeTicks::Now() - base::TimeTicks()).InMicroseconds()); |
| 53 const uint64_t time_stamp = (us * 90u + 500u) / 1000u; |
| 54 const uint8_t header_without_identifiers[] = { |
| 55 (kProtocolVersion << 6 | // Version (2 bits) |
| 56 0x0u << 5 | // Padding (no) |
| 57 0x0u << 4 | // Extension (no) |
| 58 0u << 0), // CSRC count (4 bits) |
| 59 (0x0u << 7 | // Marker (no) |
| 60 kProtocolPayloadTypeMP2T << 0), // Payload type (7 bits) |
| 61 sequence_number_ >> 8, // Sequence number (16 bits) |
| 62 sequence_number_ & 0xFFu, // |
| 63 (time_stamp >> 24) & 0xFFu, // Time stamp (32 bits) |
| 64 (time_stamp >> 16) & 0xFFu, // |
| 65 (time_stamp >> 8) & 0xFFu, // |
| 66 time_stamp & 0xFFu}; // |
| 67 ++sequence_number_; |
| 68 media_datagram_packet_.reserve( |
| 69 sizeof(header_without_identifiers) + |
| 70 sizeof(synchronization_source_identifier_) + |
| 71 kMaxTransportStreamPacketCount * |
| 72 WiFiDisplayTransportStreamPacket::kPacketSize); |
| 73 media_datagram_packet_.insert(media_datagram_packet_.end(), |
| 74 std::begin(header_without_identifiers), |
| 75 std::end(header_without_identifiers)); |
| 76 media_datagram_packet_.insert( |
| 77 media_datagram_packet_.end(), |
| 78 std::begin(synchronization_source_identifier_), |
| 79 std::end(synchronization_source_identifier_)); |
| 80 DCHECK_EQ(0u, media_datagram_packet_.size() / |
| 81 WiFiDisplayTransportStreamPacket::kPacketSize); |
| 82 } |
| 83 |
| 84 // Append header and payload data. |
| 85 media_datagram_packet_.insert(media_datagram_packet_.end(), |
| 86 transport_stream_packet.header().begin(), |
| 87 transport_stream_packet.header().end()); |
| 88 media_datagram_packet_.insert(media_datagram_packet_.end(), |
| 89 transport_stream_packet.payload().begin(), |
| 90 transport_stream_packet.payload().end()); |
| 91 media_datagram_packet_.insert(media_datagram_packet_.end(), |
| 92 transport_stream_packet.filler().size(), |
| 93 transport_stream_packet.filler().value()); |
| 94 |
| 95 // Combine multiple transport stream packets into one datagram packet |
| 96 // by delaying delegation whenever possible. |
| 97 if (!flush) { |
| 98 const size_t transport_stream_packet_count = |
| 99 media_datagram_packet_.size() / |
| 100 WiFiDisplayTransportStreamPacket::kPacketSize; |
| 101 if (transport_stream_packet_count < kMaxTransportStreamPacketCount) |
| 102 return true; |
| 103 } |
| 104 |
| 105 WiFiDisplayMediaDatagramPacket packet; |
| 106 packet.swap(media_datagram_packet_); |
| 107 return on_packetized_media_datagram_packet_.Run(std::move(packet)); |
| 108 } |
| 109 |
| 110 } // namespace extensions |
OLD | NEW |