Roll src/third_party/WebKit eac3800:0237a66 (svn 202606:202607)
[chromium-blink-merge.git] / net / spdy / spdy_frame_builder.cc
blob5015c9a49a632f511f877fa818aa1f17f048f5ef
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(SpdyConstants::IsValidFrameType(
67 version_, 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(
105 version_, SpdyConstants::SerializeFrameType(version_, type)));
106 DCHECK_EQ(0u, stream_id & ~kStreamIdMask);
107 DCHECK_GT(framer.protocol_version(), SPDY3);
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 &= WriteUInt24(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::WriteStringPiece16(const base::StringPiece& value) {
134 if (value.size() > 0xffff) {
135 DCHECK(false) << "Tried to write string with length > 16bit.";
136 return false;
139 if (!WriteUInt16(static_cast<uint16>(value.size()))) {
140 return false;
143 return WriteBytes(value.data(), static_cast<uint16>(value.size()));
146 bool SpdyFrameBuilder::WriteStringPiece32(const base::StringPiece& value) {
147 if (!WriteUInt32(value.size())) {
148 return false;
151 return WriteBytes(value.data(), value.size());
154 bool SpdyFrameBuilder::WriteBytes(const void* data, uint32 data_len) {
155 if (!CanWrite(data_len)) {
156 return false;
159 char* dest = GetWritableBuffer(data_len);
160 memcpy(dest, data, data_len);
161 Seek(data_len);
162 return true;
165 bool SpdyFrameBuilder::RewriteLength(const SpdyFramer& framer) {
166 return OverwriteLength(framer,
167 length_ - framer.GetControlFrameHeaderSize());
170 bool SpdyFrameBuilder::OverwriteLength(const SpdyFramer& framer,
171 size_t length) {
172 if (version_ < HTTP2) {
173 DCHECK_LE(length,
174 SpdyConstants::GetFrameMaximumSize(version_) -
175 framer.GetFrameMinimumSize());
176 } else {
177 DCHECK_LE(length, SpdyConstants::GetFrameMaximumSize(version_));
179 bool success = false;
180 const size_t old_length = length_;
182 if (version_ < HTTP2) {
183 FlagsAndLength flags_length = CreateFlagsAndLength(
184 0, // We're not writing over the flags value anyway.
185 length);
187 // Write into the correct location by temporarily faking the offset.
188 length_ = 5; // Offset at which the length field occurs.
189 success = WriteBytes(reinterpret_cast<char*>(&flags_length) + 1,
190 sizeof(flags_length) - 1);
191 } else {
192 length_ = 0;
193 success = WriteUInt24(length);
196 length_ = old_length;
197 return success;
200 bool SpdyFrameBuilder::OverwriteFlags(const SpdyFramer& framer,
201 uint8 flags) {
202 DCHECK_GT(framer.protocol_version(), SPDY3);
203 bool success = false;
204 const size_t old_length = length_;
205 // Flags are the fifth octet in the frame prefix.
206 length_ = 4;
207 success = WriteUInt8(flags);
208 length_ = old_length;
209 return success;
212 bool SpdyFrameBuilder::CanWrite(size_t length) const {
213 if (length > kLengthMask) {
214 DCHECK(false);
215 return false;
218 if (offset_ + length_ + length > capacity_) {
219 DCHECK(false);
220 return false;
223 return true;
226 } // namespace net