OLD | NEW |
1 // Copyright (c) 2012 The Chromium Authors. All rights reserved. | 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 | 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 "net/spdy/spdy_frame_builder.h" | 5 #include "net/spdy/spdy_frame_builder.h" |
6 | 6 |
7 #include <limits> | 7 #include <limits> |
8 | 8 |
9 #include "base/logging.h" | 9 #include "base/logging.h" |
10 #include "net/spdy/spdy_framer.h" | 10 #include "net/spdy/spdy_framer.h" |
(...skipping 42 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
53 return false; | 53 return false; |
54 } | 54 } |
55 | 55 |
56 length_ += length; | 56 length_ += length; |
57 return true; | 57 return true; |
58 } | 58 } |
59 | 59 |
60 bool SpdyFrameBuilder::WriteControlFrameHeader(const SpdyFramer& framer, | 60 bool SpdyFrameBuilder::WriteControlFrameHeader(const SpdyFramer& framer, |
61 SpdyFrameType type, | 61 SpdyFrameType type, |
62 uint8 flags) { | 62 uint8 flags) { |
63 DCHECK_GE(type, FIRST_CONTROL_TYPE); | |
64 DCHECK_LE(type, LAST_CONTROL_TYPE); | |
65 DCHECK_GT(4, framer.protocol_version()); | 63 DCHECK_GT(4, framer.protocol_version()); |
| 64 DCHECK_NE(-1, |
| 65 SpdyConstants::SerializeFrameType(framer.protocol_version(), type)); |
66 bool success = true; | 66 bool success = true; |
67 FlagsAndLength flags_length = CreateFlagsAndLength( | 67 FlagsAndLength flags_length = CreateFlagsAndLength( |
68 flags, capacity_ - framer.GetControlFrameHeaderSize()); | 68 flags, capacity_ - framer.GetControlFrameHeaderSize()); |
69 success &= WriteUInt16(kControlFlagMask | framer.protocol_version()); | 69 success &= WriteUInt16(kControlFlagMask | framer.protocol_version()); |
70 success &= WriteUInt16(type); | 70 success &= WriteUInt16( |
| 71 SpdyConstants::SerializeFrameType(framer.protocol_version(), type)); |
71 success &= WriteBytes(&flags_length, sizeof(flags_length)); | 72 success &= WriteBytes(&flags_length, sizeof(flags_length)); |
72 DCHECK_EQ(framer.GetControlFrameHeaderSize(), length()); | 73 DCHECK_EQ(framer.GetControlFrameHeaderSize(), length()); |
73 return success; | 74 return success; |
74 } | 75 } |
75 | 76 |
76 bool SpdyFrameBuilder::WriteDataFrameHeader(const SpdyFramer& framer, | 77 bool SpdyFrameBuilder::WriteDataFrameHeader(const SpdyFramer& framer, |
77 SpdyStreamId stream_id, | 78 SpdyStreamId stream_id, |
78 SpdyDataFlags flags) { | 79 uint8 flags) { |
79 if (framer.protocol_version() >= 4) { | 80 if (framer.protocol_version() >= 4) { |
80 return WriteFramePrefix(framer, DATA, flags, stream_id); | 81 return WriteFramePrefix(framer, DATA, flags, stream_id); |
81 } | 82 } |
82 DCHECK_EQ(0u, stream_id & ~kStreamIdMask); | 83 DCHECK_EQ(0u, stream_id & ~kStreamIdMask); |
83 bool success = true; | 84 bool success = true; |
84 success &= WriteUInt32(stream_id); | 85 success &= WriteUInt32(stream_id); |
85 size_t length_field = capacity_ - framer.GetDataFrameMinimumSize(); | 86 size_t length_field = capacity_ - framer.GetDataFrameMinimumSize(); |
86 DCHECK_EQ(0u, length_field & ~static_cast<size_t>(kLengthMask)); | 87 DCHECK_EQ(0u, length_field & ~static_cast<size_t>(kLengthMask)); |
87 FlagsAndLength flags_length; | 88 FlagsAndLength flags_length; |
88 flags_length.length_ = htonl(length_field); | 89 flags_length.length_ = htonl(length_field); |
89 DCHECK_EQ(0, flags & ~kDataFlagsMask); | 90 DCHECK_EQ(0, flags & ~kDataFlagsMask); |
90 flags_length.flags_[0] = flags; | 91 flags_length.flags_[0] = flags; |
91 success &= WriteBytes(&flags_length, sizeof(flags_length)); | 92 success &= WriteBytes(&flags_length, sizeof(flags_length)); |
92 DCHECK_EQ(framer.GetDataFrameMinimumSize(), length()); | 93 DCHECK_EQ(framer.GetDataFrameMinimumSize(), length()); |
93 return success; | 94 return success; |
94 } | 95 } |
95 | 96 |
96 bool SpdyFrameBuilder::WriteFramePrefix(const SpdyFramer& framer, | 97 bool SpdyFrameBuilder::WriteFramePrefix(const SpdyFramer& framer, |
97 SpdyFrameType type, | 98 SpdyFrameType type, |
98 uint8 flags, | 99 uint8 flags, |
99 SpdyStreamId stream_id) { | 100 SpdyStreamId stream_id) { |
100 DCHECK_LE(DATA, type); | 101 DCHECK_NE(-1, |
101 DCHECK_GE(LAST_CONTROL_TYPE, type); | 102 SpdyConstants::SerializeFrameType(framer.protocol_version(), type)); |
102 DCHECK_EQ(0u, stream_id & ~kStreamIdMask); | 103 DCHECK_EQ(0u, stream_id & ~kStreamIdMask); |
103 DCHECK_LE(4, framer.protocol_version()); | 104 DCHECK_LE(4, framer.protocol_version()); |
104 bool success = true; | 105 bool success = true; |
105 // Upstream DCHECK's that capacity_ is under the maximum frame size at this | 106 // Upstream DCHECK's that capacity_ is under the maximum frame size at this |
106 // point. Chromium does not, because of the large additional zlib inflation | 107 // point. Chromium does not, because of the large additional zlib inflation |
107 // factor we use. (Frame size is is still checked by OverwriteLength() below). | 108 // factor we use. (Frame size is is still checked by OverwriteLength() below). |
108 success &= WriteUInt16(capacity_); | 109 if (type != DATA) { |
109 success &= WriteUInt8(type); | 110 success &= WriteUInt16(capacity_ - framer.GetControlFrameHeaderSize()); |
| 111 } else { |
| 112 success &= WriteUInt16(capacity_ - framer.GetDataFrameMinimumSize()); |
| 113 } |
| 114 success &= WriteUInt8( |
| 115 SpdyConstants::SerializeFrameType(framer.protocol_version(), type)); |
110 success &= WriteUInt8(flags); | 116 success &= WriteUInt8(flags); |
111 success &= WriteUInt32(stream_id); | 117 success &= WriteUInt32(stream_id); |
112 DCHECK_EQ(framer.GetDataFrameMinimumSize(), length()); | 118 DCHECK_EQ(framer.GetDataFrameMinimumSize(), length()); |
113 return success; | 119 return success; |
114 } | 120 } |
115 | 121 |
116 bool SpdyFrameBuilder::WriteString(const std::string& value) { | 122 bool SpdyFrameBuilder::WriteString(const std::string& value) { |
117 if (value.size() > 0xffff) { | 123 if (value.size() > 0xffff) { |
118 DCHECK(false) << "Tried to write string with length > 16bit."; | 124 DCHECK(false) << "Tried to write string with length > 16bit."; |
119 return false; | 125 return false; |
(...skipping 18 matching lines...) Expand all Loading... |
138 return false; | 144 return false; |
139 } | 145 } |
140 | 146 |
141 char* dest = GetWritableBuffer(data_len); | 147 char* dest = GetWritableBuffer(data_len); |
142 memcpy(dest, data, data_len); | 148 memcpy(dest, data, data_len); |
143 Seek(data_len); | 149 Seek(data_len); |
144 return true; | 150 return true; |
145 } | 151 } |
146 | 152 |
147 bool SpdyFrameBuilder::RewriteLength(const SpdyFramer& framer) { | 153 bool SpdyFrameBuilder::RewriteLength(const SpdyFramer& framer) { |
148 if (framer.protocol_version() < 4) { | 154 return OverwriteLength(framer, |
149 return OverwriteLength(framer, | 155 length_ - framer.GetControlFrameHeaderSize()); |
150 length_ - framer.GetControlFrameHeaderSize()); | |
151 } else { | |
152 return OverwriteLength(framer, length_); | |
153 } | |
154 } | 156 } |
155 | 157 |
156 bool SpdyFrameBuilder::OverwriteLength(const SpdyFramer& framer, | 158 bool SpdyFrameBuilder::OverwriteLength(const SpdyFramer& framer, |
157 size_t length) { | 159 size_t length) { |
158 if (framer.protocol_version() < 4) { | 160 if (framer.protocol_version() < 4) { |
159 DCHECK_GT(framer.GetFrameMaximumSize() - framer.GetFrameMinimumSize(), | 161 DCHECK_GT(framer.GetFrameMaximumSize() - framer.GetFrameMinimumSize(), |
160 length); | 162 length); |
161 } else { | 163 } else { |
162 DCHECK_GE(framer.GetFrameMaximumSize(), length); | 164 DCHECK_GE(framer.GetFrameMaximumSize(), length); |
163 } | 165 } |
(...skipping 11 matching lines...) Expand all Loading... |
175 sizeof(flags_length) - 1); | 177 sizeof(flags_length) - 1); |
176 } else { | 178 } else { |
177 length_ = 0; | 179 length_ = 0; |
178 success = WriteUInt16(length); | 180 success = WriteUInt16(length); |
179 } | 181 } |
180 | 182 |
181 length_ = old_length; | 183 length_ = old_length; |
182 return success; | 184 return success; |
183 } | 185 } |
184 | 186 |
| 187 bool SpdyFrameBuilder::OverwriteFlags(const SpdyFramer& framer, |
| 188 uint8 flags) { |
| 189 DCHECK_LE(SPDY4, framer.protocol_version()); |
| 190 bool success = false; |
| 191 const size_t old_length = length_; |
| 192 // Flags are the fourth octet in the frame prefix. |
| 193 length_ = 3; |
| 194 success = WriteUInt8(flags); |
| 195 length_ = old_length; |
| 196 return success; |
| 197 } |
| 198 |
185 bool SpdyFrameBuilder::CanWrite(size_t length) const { | 199 bool SpdyFrameBuilder::CanWrite(size_t length) const { |
186 if (length > kLengthMask) { | 200 if (length > kLengthMask) { |
187 DCHECK(false); | 201 DCHECK(false); |
188 return false; | 202 return false; |
189 } | 203 } |
190 | 204 |
191 if (length_ + length > capacity_) { | 205 if (length_ + length > capacity_) { |
192 DCHECK(false); | 206 DCHECK(false); |
193 return false; | 207 return false; |
194 } | 208 } |
195 | 209 |
196 return true; | 210 return true; |
197 } | 211 } |
198 | 212 |
199 } // namespace net | 213 } // namespace net |
OLD | NEW |