Chromium Code Reviews
chromiumcodereview-hr@appspot.gserviceaccount.com (chromiumcodereview-hr) | Please choose your nickname with Settings | Help | Chromium Project | Gerrit Changes | Sign out
(566)

Side by Side Diff: extensions/browser/api/cast_channel/cast_framer.cc

Issue 555283002: Create new class "CastTransport", which encapsulates the message read and write event loops. (Closed) Base URL: https://chromium.googlesource.com/chromium/src.git@master
Patch Set: Added unit tests to CL Created 6 years, 3 months ago
Use n/p to move between diff chunks; N/P to move between comments. Draft comments are only viewable by you.
Jump to:
View unified diff | Download patch
OLDNEW
(Empty)
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
3 // found in the LICENSE file.
4
5 #include "extensions/browser/api/cast_channel/cast_framer.h"
6
7 #include <stdlib.h>
8 #include <string>
9
10 #include "base/numerics/safe_conversions.h"
11 #include "base/strings/string_number_conversions.h"
12 #include "base/sys_byteorder.h"
13 #include "extensions/browser/api/cast_channel/cast_channel.pb.h"
14
15 namespace extensions {
16 namespace core_api {
17 namespace cast_channel {
18 MessageFramer::MessageFramer(scoped_refptr<net::GrowableIOBuffer> input_buffer)
19 : input_buffer_(input_buffer), error_(false) {
20 Reset();
21 }
22
23 MessageFramer::~MessageFramer() {
24 }
25
26 MessageFramer::MessageHeader::MessageHeader() : message_size(0) {
27 }
28
29 void MessageFramer::MessageHeader::SetMessageSize(size_t size) {
30 DCHECK_LT(size, static_cast<size_t>(kuint32max));
31 DCHECK_GT(size, 0U);
32 message_size = size;
33 }
34
35 // TODO(mfoltz): Investigate replacing header serialization with base::Pickle,
36 // if bit-for-bit compatible.
37 void MessageFramer::MessageHeader::PrependToString(std::string* str) {
38 MessageHeader output = *this;
39 output.message_size = base::HostToNet32(message_size);
40 size_t header_size = MessageHeader::header_size();
41 scoped_ptr<char, base::FreeDeleter> char_array(
42 static_cast<char*>(malloc(header_size)));
43 memcpy(char_array.get(), &output, header_size);
44 str->insert(0, char_array.get(), header_size);
45 }
46
47 // TODO(mfoltz): Investigate replacing header deserialization with base::Pickle,
48 // if bit-for-bit compatible.
49 void MessageFramer::MessageHeader::Deserialize(char* data,
50 MessageHeader* header) {
51 uint32 message_size;
52 memcpy(&message_size, data, header_size());
53 header->message_size =
54 base::checked_cast<size_t>(base::NetToHost32(message_size));
55 }
56
57 // static
58 size_t MessageFramer::MessageHeader::header_size() {
59 return sizeof(uint32);
60 }
61
62 // static
63 size_t MessageFramer::MessageHeader::max_message_size() {
64 return 65535;
65 }
66
67 std::string MessageFramer::MessageHeader::ToString() {
68 return "{message_size: " +
69 base::UintToString(static_cast<uint32>(message_size)) + "}";
70 }
71
72 // static
73 bool MessageFramer::Serialize(const CastMessage& message_proto,
74 std::string* message_data) {
75 DCHECK(message_data);
76 message_proto.SerializeToString(message_data);
77 size_t message_size = message_data->size();
78 if (message_size > MessageHeader::max_message_size()) {
79 message_data->clear();
80 return false;
81 }
82 MessageHeader header;
83 header.SetMessageSize(message_size);
84 header.PrependToString(message_data);
85 return true;
86 }
87
88 size_t MessageFramer::BytesRequested() {
89 size_t bytes_left;
90 if (error_) {
91 return 0;
92 }
93
94 switch (current_element_) {
95 case HEADER:
96 bytes_left = MessageHeader::header_size() - message_bytes_received_;
97 DCHECK_LE(bytes_left, MessageHeader::header_size());
98 VLOG(2) << "Bytes needed for header: " << bytes_left;
99 return bytes_left;
100 case BODY:
101 bytes_left =
102 (body_size_ + MessageHeader::header_size()) - message_bytes_received_;
103 DCHECK_LE(
104 bytes_left,
105 MessageHeader::max_message_size() - MessageHeader::header_size());
106 VLOG(2) << "Bytes needed for body: " << bytes_left;
107 return bytes_left;
108 default:
109 NOTREACHED() << "Unhandled packet element type.";
110 return 0;
111 }
112 }
113
114 scoped_ptr<CastMessage> MessageFramer::Ingest(size_t num_bytes,
115 size_t* message_length,
116 ChannelError* error) {
117 DCHECK(error);
118 DCHECK(message_length);
119 if (error_) {
120 *error = CHANNEL_ERROR_INVALID_MESSAGE;
121 return scoped_ptr<CastMessage>();
122 }
123
124 DCHECK_EQ(base::checked_cast<int32>(message_bytes_received_),
125 input_buffer_->offset());
126 CHECK_LE(num_bytes, BytesRequested());
127 message_bytes_received_ += num_bytes;
128 *error = CHANNEL_ERROR_NONE;
129 *message_length = 0;
130 switch (current_element_) {
131 case HEADER:
132 if (BytesRequested() == 0) {
133 MessageHeader header;
134 MessageHeader::Deserialize(input_buffer_.get()->StartOfBuffer(),
135 &header);
136 if (header.message_size > MessageHeader::max_message_size()) {
137 VLOG(1) << "Error parsing header (message size too large).";
138 *error = CHANNEL_ERROR_INVALID_MESSAGE;
139 error_ = true;
140 return scoped_ptr<CastMessage>();
141 }
142 current_element_ = BODY;
143 body_size_ = header.message_size;
144 }
145 break;
146 case BODY:
147 if (BytesRequested() == 0) {
148 scoped_ptr<CastMessage> parsed_message(new CastMessage);
149 if (!parsed_message->ParseFromArray(
150 input_buffer_->StartOfBuffer() + MessageHeader::header_size(),
151 body_size_)) {
152 VLOG(1) << "Error parsing packet body.";
153 *error = CHANNEL_ERROR_INVALID_MESSAGE;
154 error_ = true;
155 return scoped_ptr<CastMessage>();
156 }
157 *message_length = body_size_;
158 Reset();
159 return parsed_message.Pass();
160 }
161 break;
162 default:
163 NOTREACHED() << "Unhandled packet element type.";
164 return scoped_ptr<CastMessage>();
165 }
166
167 input_buffer_->set_offset(message_bytes_received_);
168 return scoped_ptr<CastMessage>();
169 }
170
171 void MessageFramer::Reset() {
172 current_element_ = HEADER;
173 message_bytes_received_ = 0;
174 body_size_ = 0;
175 input_buffer_->set_offset(0);
176 }
177
178 } // namespace cast_channel
179 } // namespace core_api
180 } // namespace extensions
OLDNEW

Powered by Google App Engine
This is Rietveld 408576698