Make mojo message building and validation follow the depth-first traversal order.
[chromium-blink-merge.git] / net / spdy / spdy_frame_builder.cc
bloba7eb91d71ce8199d9e68b08831340d5580045670
1 // Copyright (c) 2012 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.
5 #include "net/spdy/spdy_frame_builder.h"
7 #include <limits>
9 #include "base/logging.h"
10 #include "net/spdy/spdy_framer.h"
11 #include "net/spdy/spdy_protocol.h"
13 namespace net {
15 namespace {
17 // A special structure for the 8 bit flags and 24 bit length fields.
18 union FlagsAndLength {
19 uint8 flags_[4]; // 8 bits
20 uint32 length_; // 24 bits
23 // Creates a FlagsAndLength.
24 FlagsAndLength CreateFlagsAndLength(uint8 flags, size_t length) {
25 DCHECK_EQ(0u, length & ~static_cast<size_t>(kLengthMask));
26 FlagsAndLength flags_length;
27 flags_length.length_ = htonl(static_cast<uint32>(length));
28 DCHECK_EQ(0, flags & ~kControlFlagsMask);
29 flags_length.flags_[0] = flags;
30 return flags_length;
33 } // namespace
35 SpdyFrameBuilder::SpdyFrameBuilder(size_t size, SpdyMajorVersion version)
36 : buffer_(new char[size]),
37 capacity_(size),
38 length_(0),
39 offset_(0),
40 version_(version) {
43 SpdyFrameBuilder::~SpdyFrameBuilder() {
46 char* SpdyFrameBuilder::GetWritableBuffer(size_t length) {
47 if (!CanWrite(length)) {
48 return NULL;
50 return buffer_.get() + offset_ + length_;
53 bool SpdyFrameBuilder::Seek(size_t length) {
54 if (!CanWrite(length)) {
55 return false;
58 length_ += length;
59 return true;
62 bool SpdyFrameBuilder::WriteControlFrameHeader(const SpdyFramer& framer,
63 SpdyFrameType type,
64 uint8 flags) {
65 DCHECK_GE(SPDY3, version_);
66 DCHECK_NE(-1,
67 SpdyConstants::SerializeFrameType(version_, type));
68 bool success = true;
69 FlagsAndLength flags_length = CreateFlagsAndLength(
70 flags, capacity_ - framer.GetControlFrameHeaderSize());
71 success &= WriteUInt16(kControlFlagMask |
72 SpdyConstants::SerializeMajorVersion(version_));
73 success &= WriteUInt16(
74 SpdyConstants::SerializeFrameType(framer.protocol_version(), type));
75 success &= WriteBytes(&flags_length, sizeof(flags_length));
76 DCHECK_EQ(framer.GetControlFrameHeaderSize(), length());
77 return success;
80 bool SpdyFrameBuilder::WriteDataFrameHeader(const SpdyFramer& framer,
81 SpdyStreamId stream_id,
82 uint8 flags) {
83 if (version_ > SPDY3) {
84 return BeginNewFrame(framer, DATA, flags, stream_id);
86 DCHECK_EQ(0u, stream_id & ~kStreamIdMask);
87 bool success = true;
88 success &= WriteUInt32(stream_id);
89 size_t length_field = capacity_ - framer.GetDataFrameMinimumSize();
90 DCHECK_EQ(0u, length_field & ~static_cast<size_t>(kLengthMask));
91 FlagsAndLength flags_length;
92 flags_length.length_ = htonl(length_field);
93 DCHECK_EQ(0, flags & ~kDataFlagsMask);
94 flags_length.flags_[0] = flags;
95 success &= WriteBytes(&flags_length, sizeof(flags_length));
96 DCHECK_EQ(framer.GetDataFrameMinimumSize(), length());
97 return success;
100 bool SpdyFrameBuilder::BeginNewFrame(const SpdyFramer& framer,
101 SpdyFrameType type,
102 uint8 flags,
103 SpdyStreamId stream_id) {
104 DCHECK(SpdyConstants::IsValidFrameType(version_,
105 SpdyConstants::SerializeFrameType(version_, type)));
106 DCHECK_EQ(0u, stream_id & ~kStreamIdMask);
107 DCHECK_LT(SPDY3, framer.protocol_version());
108 bool success = true;
109 if (length_ > 0) {
110 // Update length field for previous frame.
111 OverwriteLength(framer, length_ - framer.GetPrefixLength(type));
112 DLOG_IF(DFATAL, SpdyConstants::GetFrameMaximumSize(version_) < length_)
113 << "Frame length " << length_
114 << " is longer than the maximum allowed length.";
117 offset_ += length_;
118 length_ = 0;
120 // Assume all remaining capacity will be used for this frame. If not,
121 // the length will get overwritten when we begin the next frame.
122 // Don't check for length limits here because this may be larger than the
123 // actual frame length.
124 success &= WriteUInt16(capacity_ - offset_ - framer.GetPrefixLength(type));
125 success &= WriteUInt8(
126 SpdyConstants::SerializeFrameType(version_, type));
127 success &= WriteUInt8(flags);
128 success &= WriteUInt32(stream_id);
129 DCHECK_EQ(framer.GetDataFrameMinimumSize(), length_);
130 return success;
133 bool SpdyFrameBuilder::WriteString(const std::string& value) {
134 if (value.size() > 0xffff) {
135 DCHECK(false) << "Tried to write string with length > 16bit.";
136 return false;
139 if (!WriteUInt16(static_cast<int>(value.size())))
140 return false;
142 return WriteBytes(value.data(), static_cast<uint16>(value.size()));
145 bool SpdyFrameBuilder::WriteStringPiece32(const base::StringPiece& value) {
146 if (!WriteUInt32(value.size())) {
147 return false;
150 return WriteBytes(value.data(), value.size());
153 bool SpdyFrameBuilder::WriteBytes(const void* data, uint32 data_len) {
154 if (!CanWrite(data_len)) {
155 return false;
158 char* dest = GetWritableBuffer(data_len);
159 memcpy(dest, data, data_len);
160 Seek(data_len);
161 return true;
164 bool SpdyFrameBuilder::RewriteLength(const SpdyFramer& framer) {
165 return OverwriteLength(framer,
166 length_ - framer.GetControlFrameHeaderSize());
169 bool SpdyFrameBuilder::OverwriteLength(const SpdyFramer& framer,
170 size_t length) {
171 if (version_ <= SPDY3) {
172 DCHECK_GE(SpdyConstants::GetFrameMaximumSize(version_) -
173 framer.GetFrameMinimumSize(),
174 length);
175 } else {
176 DCHECK_GE(SpdyConstants::GetFrameMaximumSize(version_), length);
178 bool success = false;
179 const size_t old_length = length_;
181 if (version_ <= SPDY3) {
182 FlagsAndLength flags_length = CreateFlagsAndLength(
183 0, // We're not writing over the flags value anyway.
184 length);
186 // Write into the correct location by temporarily faking the offset.
187 length_ = 5; // Offset at which the length field occurs.
188 success = WriteBytes(reinterpret_cast<char*>(&flags_length) + 1,
189 sizeof(flags_length) - 1);
190 } else {
191 length_ = 0;
192 success = WriteUInt16(length);
195 length_ = old_length;
196 return success;
199 bool SpdyFrameBuilder::OverwriteFlags(const SpdyFramer& framer,
200 uint8 flags) {
201 DCHECK_LT(SPDY3, framer.protocol_version());
202 bool success = false;
203 const size_t old_length = length_;
204 // Flags are the fourth octet in the frame prefix.
205 length_ = 3;
206 success = WriteUInt8(flags);
207 length_ = old_length;
208 return success;
211 bool SpdyFrameBuilder::CanWrite(size_t length) const {
212 if (length > kLengthMask) {
213 DCHECK(false);
214 return false;
217 if (offset_ + length_ + length > capacity_) {
218 DCHECK(false);
219 return false;
222 return true;
225 } // namespace net