OLD | NEW |
1 // Copyright 2014 The Chromium Authors. All rights reserved. | 1 // Copyright 2014 The Chromium Authors. All rights reserved. |
2 // Use of this source code is governed by a BSD-style license that can be | 2 // Use of this source code is governed by a BSD-style license that can be |
3 // found in the LICENSE file. | 3 // found in the LICENSE file. |
4 | 4 |
5 #include "media/cast/net/rtp/frame_buffer.h" | 5 #include "media/cast/net/rtp/frame_buffer.h" |
6 | 6 |
7 #include "base/logging.h" | 7 #include "base/logging.h" |
8 | 8 |
9 namespace media { | 9 namespace media { |
10 namespace cast { | 10 namespace cast { |
11 | 11 |
12 FrameBuffer::FrameBuffer() | 12 FrameBuffer::FrameBuffer() |
13 : frame_id_(0), | 13 : frame_id_(0), |
14 max_packet_id_(0), | 14 max_packet_id_(0), |
15 num_packets_received_(0), | 15 num_packets_received_(0), |
| 16 max_seen_packet_id_(0), |
16 new_playout_delay_ms_(0), | 17 new_playout_delay_ms_(0), |
17 is_key_frame_(false), | 18 is_key_frame_(false), |
18 total_data_size_(0), | 19 total_data_size_(0), |
19 last_referenced_frame_id_(0), | 20 last_referenced_frame_id_(0), |
20 packets_() {} | 21 packets_() {} |
21 | 22 |
22 FrameBuffer::~FrameBuffer() {} | 23 FrameBuffer::~FrameBuffer() {} |
23 | 24 |
24 void FrameBuffer::InsertPacket(const uint8* payload_data, | 25 bool FrameBuffer::InsertPacket(const uint8* payload_data, |
25 size_t payload_size, | 26 size_t payload_size, |
26 const RtpCastHeader& rtp_header) { | 27 const RtpCastHeader& rtp_header) { |
27 // Is this the first packet in the frame? | 28 // Is this the first packet in the frame? |
28 if (packets_.empty()) { | 29 if (packets_.empty()) { |
29 frame_id_ = rtp_header.frame_id; | 30 frame_id_ = rtp_header.frame_id; |
30 max_packet_id_ = rtp_header.max_packet_id; | 31 max_packet_id_ = rtp_header.max_packet_id; |
31 is_key_frame_ = rtp_header.is_key_frame; | 32 is_key_frame_ = rtp_header.is_key_frame; |
32 new_playout_delay_ms_ = rtp_header.new_playout_delay_ms; | 33 new_playout_delay_ms_ = rtp_header.new_playout_delay_ms; |
33 if (is_key_frame_) | 34 if (is_key_frame_) |
34 DCHECK_EQ(rtp_header.frame_id, rtp_header.reference_frame_id); | 35 DCHECK_EQ(rtp_header.frame_id, rtp_header.reference_frame_id); |
35 last_referenced_frame_id_ = rtp_header.reference_frame_id; | 36 last_referenced_frame_id_ = rtp_header.reference_frame_id; |
36 rtp_timestamp_ = rtp_header.rtp_timestamp; | 37 rtp_timestamp_ = rtp_header.rtp_timestamp; |
37 } | 38 } |
38 // Is this the correct frame? | 39 // Is this the correct frame? |
39 if (rtp_header.frame_id != frame_id_) | 40 if (rtp_header.frame_id != frame_id_) |
40 return; | 41 return false; |
41 | 42 |
42 // Insert every packet only once. | 43 // Insert every packet only once. |
43 if (packets_.find(rtp_header.packet_id) != packets_.end()) { | 44 if (packets_.find(rtp_header.packet_id) != packets_.end()) { |
44 return; | 45 return false; |
45 } | 46 } |
46 | 47 |
47 std::vector<uint8> data; | 48 std::vector<uint8> data; |
48 std::pair<PacketMap::iterator, bool> retval = | 49 std::pair<PacketMap::iterator, bool> retval = |
49 packets_.insert(make_pair(rtp_header.packet_id, data)); | 50 packets_.insert(make_pair(rtp_header.packet_id, data)); |
50 | 51 |
51 // Insert the packet. | 52 // Insert the packet. |
52 retval.first->second.resize(payload_size); | 53 retval.first->second.resize(payload_size); |
53 std::copy( | 54 std::copy( |
54 payload_data, payload_data + payload_size, retval.first->second.begin()); | 55 payload_data, payload_data + payload_size, retval.first->second.begin()); |
55 | 56 |
56 ++num_packets_received_; | 57 ++num_packets_received_; |
| 58 max_seen_packet_id_ = std::max(max_seen_packet_id_, rtp_header.packet_id); |
57 total_data_size_ += payload_size; | 59 total_data_size_ += payload_size; |
| 60 return true; |
58 } | 61 } |
59 | 62 |
60 bool FrameBuffer::Complete() const { | 63 bool FrameBuffer::Complete() const { |
61 return num_packets_received_ - 1 == max_packet_id_; | 64 return num_packets_received_ - 1 == max_packet_id_; |
62 } | 65 } |
63 | 66 |
64 bool FrameBuffer::AssembleEncodedFrame(EncodedFrame* frame) const { | 67 bool FrameBuffer::AssembleEncodedFrame(EncodedFrame* frame) const { |
65 if (!Complete()) | 68 if (!Complete()) |
66 return false; | 69 return false; |
67 | 70 |
(...skipping 11 matching lines...) Expand all Loading... |
79 | 82 |
80 // Build the data vector. | 83 // Build the data vector. |
81 frame->data.clear(); | 84 frame->data.clear(); |
82 frame->data.reserve(total_data_size_); | 85 frame->data.reserve(total_data_size_); |
83 PacketMap::const_iterator it; | 86 PacketMap::const_iterator it; |
84 for (it = packets_.begin(); it != packets_.end(); ++it) | 87 for (it = packets_.begin(); it != packets_.end(); ++it) |
85 frame->data.insert(frame->data.end(), it->second.begin(), it->second.end()); | 88 frame->data.insert(frame->data.end(), it->second.begin(), it->second.end()); |
86 return true; | 89 return true; |
87 } | 90 } |
88 | 91 |
| 92 void FrameBuffer::GetMissingPackets(bool newest_frame, |
| 93 PacketIdSet* missing_packets) const { |
| 94 // Missing packets capped by max_seen_packet_id_. |
| 95 // (Iff it's the latest frame) |
| 96 int maximum = newest_frame ? max_seen_packet_id_ : max_packet_id_; |
| 97 int packet = 0; |
| 98 for (PacketMap::const_iterator i = packets_.begin(); |
| 99 i != packets_.end() && packet <= maximum; |
| 100 ++i) { |
| 101 int end = std::min<int>(i->first, maximum + 1); |
| 102 while (packet < end) { |
| 103 missing_packets->insert(packet); |
| 104 packet++; |
| 105 } |
| 106 packet++; |
| 107 } |
| 108 while (packet <= maximum) { |
| 109 missing_packets->insert(packet); |
| 110 packet++; |
| 111 } |
| 112 } |
| 113 |
| 114 |
89 } // namespace cast | 115 } // namespace cast |
90 } // namespace media | 116 } // namespace media |
OLD | NEW |