QUICHE team | 82dee2f | 2019-01-18 12:35:12 -0500 | [diff] [blame] | 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. |
| 4 | |
| 5 | #ifndef QUICHE_SPDY_CORE_SPDY_HEADER_BLOCK_H_ |
| 6 | #define QUICHE_SPDY_CORE_SPDY_HEADER_BLOCK_H_ |
| 7 | |
| 8 | #include <stddef.h> |
| 9 | |
QUICHE team | bbce603 | 2020-01-03 07:42:10 -0800 | [diff] [blame] | 10 | #include <functional> |
QUICHE team | 82dee2f | 2019-01-18 12:35:12 -0500 | [diff] [blame] | 11 | #include <list> |
bnc | 4471291 | 2019-08-15 18:58:14 -0700 | [diff] [blame] | 12 | #include <string> |
QUICHE team | 82dee2f | 2019-01-18 12:35:12 -0500 | [diff] [blame] | 13 | #include <utility> |
| 14 | #include <vector> |
| 15 | |
vasilvv | 50465ab | 2021-02-08 05:50:01 -0800 | [diff] [blame] | 16 | #include "absl/base/attributes.h" |
bnc | 60cef58 | 2021-03-24 10:25:37 -0700 | [diff] [blame] | 17 | #include "absl/hash/hash.h" |
| 18 | #include "absl/strings/ascii.h" |
| 19 | #include "absl/strings/match.h" |
vasilvv | c8ccdb2 | 2020-10-12 16:42:34 -0700 | [diff] [blame] | 20 | #include "absl/strings/string_view.h" |
QUICHE team | 5be974e | 2020-12-29 18:35:24 -0500 | [diff] [blame] | 21 | #include "common/platform/api/quiche_export.h" |
vasilvv | ed4f308 | 2021-02-01 14:29:40 -0800 | [diff] [blame] | 22 | #include "common/platform/api/quiche_logging.h" |
QUICHE team | 5be974e | 2020-12-29 18:35:24 -0500 | [diff] [blame] | 23 | #include "spdy/core/spdy_header_storage.h" |
| 24 | #include "spdy/platform/api/spdy_containers.h" |
QUICHE team | 82dee2f | 2019-01-18 12:35:12 -0500 | [diff] [blame] | 25 | |
| 26 | namespace spdy { |
| 27 | |
| 28 | namespace test { |
QUICHE team | 1bd18d1 | 2020-10-21 08:36:48 -0700 | [diff] [blame] | 29 | class Http2HeaderBlockPeer; |
QUICHE team | 82dee2f | 2019-01-18 12:35:12 -0500 | [diff] [blame] | 30 | class ValueProxyPeer; |
| 31 | } // namespace test |
| 32 | |
dschinazi | b2b2615 | 2020-07-01 10:17:25 -0700 | [diff] [blame] | 33 | #ifndef SPDY_HEADER_DEBUG |
| 34 | #if !defined(NDEBUG) || defined(ADDRESS_SANITIZER) |
| 35 | #define SPDY_HEADER_DEBUG 1 |
| 36 | #else // !defined(NDEBUG) || defined(ADDRESS_SANITIZER) |
| 37 | #define SPDY_HEADER_DEBUG 0 |
| 38 | #endif // !defined(NDEBUG) || defined(ADDRESS_SANITIZER) |
| 39 | #endif // SPDY_HEADER_DEBUG |
| 40 | |
QUICHE team | 82dee2f | 2019-01-18 12:35:12 -0500 | [diff] [blame] | 41 | // This class provides a key-value map that can be used to store SPDY header |
| 42 | // names and values. This data structure preserves insertion order. |
| 43 | // |
| 44 | // Under the hood, this data structure uses large, contiguous blocks of memory |
vasilvv | c8ccdb2 | 2020-10-12 16:42:34 -0700 | [diff] [blame] | 45 | // to store names and values. Lookups may be performed with absl::string_view |
| 46 | // keys, and values are returned as absl::string_views (via ValueProxy, below). |
QUICHE team | 557dfbc | 2020-10-20 10:52:30 -0700 | [diff] [blame] | 47 | // Value absl::string_views are valid as long as the Http2HeaderBlock exists; |
| 48 | // allocated memory is never freed until Http2HeaderBlock's destruction. |
QUICHE team | 82dee2f | 2019-01-18 12:35:12 -0500 | [diff] [blame] | 49 | // |
| 50 | // This implementation does not make much of an effort to minimize wasted space. |
QUICHE team | 557dfbc | 2020-10-20 10:52:30 -0700 | [diff] [blame] | 51 | // It's expected that keys are rarely deleted from a Http2HeaderBlock. |
| 52 | class QUICHE_EXPORT_PRIVATE Http2HeaderBlock { |
QUICHE team | 82dee2f | 2019-01-18 12:35:12 -0500 | [diff] [blame] | 53 | private: |
QUICHE team | 82dee2f | 2019-01-18 12:35:12 -0500 | [diff] [blame] | 54 | // Stores a list of value fragments that can be joined later with a |
| 55 | // key-dependent separator. |
bnc | 45ccf4b | 2020-01-21 19:05:27 -0800 | [diff] [blame] | 56 | class QUICHE_EXPORT_PRIVATE HeaderValue { |
QUICHE team | 82dee2f | 2019-01-18 12:35:12 -0500 | [diff] [blame] | 57 | public: |
QUICHE team | 6b29700 | 2019-12-09 09:37:56 -0800 | [diff] [blame] | 58 | HeaderValue(SpdyHeaderStorage* storage, |
vasilvv | c8ccdb2 | 2020-10-12 16:42:34 -0700 | [diff] [blame] | 59 | absl::string_view key, |
| 60 | absl::string_view initial_value); |
QUICHE team | 82dee2f | 2019-01-18 12:35:12 -0500 | [diff] [blame] | 61 | |
| 62 | // Moves are allowed. |
| 63 | HeaderValue(HeaderValue&& other); |
| 64 | HeaderValue& operator=(HeaderValue&& other); |
| 65 | |
QUICHE team | 0b74550 | 2019-12-10 11:02:08 -0800 | [diff] [blame] | 66 | void set_storage(SpdyHeaderStorage* storage); |
| 67 | |
QUICHE team | 82dee2f | 2019-01-18 12:35:12 -0500 | [diff] [blame] | 68 | // Copies are not. |
| 69 | HeaderValue(const HeaderValue& other) = delete; |
| 70 | HeaderValue& operator=(const HeaderValue& other) = delete; |
| 71 | |
| 72 | ~HeaderValue(); |
| 73 | |
| 74 | // Consumes at most |fragment.size()| bytes of memory. |
vasilvv | c8ccdb2 | 2020-10-12 16:42:34 -0700 | [diff] [blame] | 75 | void Append(absl::string_view fragment); |
QUICHE team | 82dee2f | 2019-01-18 12:35:12 -0500 | [diff] [blame] | 76 | |
vasilvv | c8ccdb2 | 2020-10-12 16:42:34 -0700 | [diff] [blame] | 77 | absl::string_view value() const { return as_pair().second; } |
| 78 | const std::pair<absl::string_view, absl::string_view>& as_pair() const; |
QUICHE team | 82dee2f | 2019-01-18 12:35:12 -0500 | [diff] [blame] | 79 | |
| 80 | // Size estimate including separators. Used when keys are erased from |
QUICHE team | 557dfbc | 2020-10-20 10:52:30 -0700 | [diff] [blame] | 81 | // Http2HeaderBlock. |
QUICHE team | 82dee2f | 2019-01-18 12:35:12 -0500 | [diff] [blame] | 82 | size_t SizeEstimate() const { return size_; } |
| 83 | |
| 84 | private: |
| 85 | // May allocate a large contiguous region of memory to hold the concatenated |
| 86 | // fragments and separators. |
vasilvv | c8ccdb2 | 2020-10-12 16:42:34 -0700 | [diff] [blame] | 87 | absl::string_view ConsolidatedValue() const; |
QUICHE team | 82dee2f | 2019-01-18 12:35:12 -0500 | [diff] [blame] | 88 | |
QUICHE team | 6b29700 | 2019-12-09 09:37:56 -0800 | [diff] [blame] | 89 | mutable SpdyHeaderStorage* storage_; |
vasilvv | c8ccdb2 | 2020-10-12 16:42:34 -0700 | [diff] [blame] | 90 | mutable std::vector<absl::string_view> fragments_; |
QUICHE team | 82dee2f | 2019-01-18 12:35:12 -0500 | [diff] [blame] | 91 | // The first element is the key; the second is the consolidated value. |
vasilvv | c8ccdb2 | 2020-10-12 16:42:34 -0700 | [diff] [blame] | 92 | mutable std::pair<absl::string_view, absl::string_view> pair_; |
QUICHE team | 82dee2f | 2019-01-18 12:35:12 -0500 | [diff] [blame] | 93 | size_t size_ = 0; |
| 94 | size_t separator_size_ = 0; |
| 95 | }; |
| 96 | |
bnc | 60cef58 | 2021-03-24 10:25:37 -0700 | [diff] [blame] | 97 | struct StringPieceCaseHash { |
| 98 | size_t operator()(absl::string_view data) const { |
| 99 | std::string lower = absl::AsciiStrToLower(data); |
| 100 | absl::Hash<absl::string_view> hasher; |
| 101 | return hasher(lower); |
| 102 | } |
| 103 | }; |
| 104 | |
| 105 | struct StringPieceCaseEqual { |
| 106 | bool operator()(absl::string_view piece1, absl::string_view piece2) const { |
| 107 | return absl::EqualsIgnoreCase(piece1, piece2); |
| 108 | } |
| 109 | }; |
| 110 | |
vasilvv | c8ccdb2 | 2020-10-12 16:42:34 -0700 | [diff] [blame] | 111 | typedef SpdyLinkedHashMap<absl::string_view, |
QUICHE team | bbce603 | 2020-01-03 07:42:10 -0800 | [diff] [blame] | 112 | HeaderValue, |
bnc | 60cef58 | 2021-03-24 10:25:37 -0700 | [diff] [blame] | 113 | StringPieceCaseHash, |
| 114 | StringPieceCaseEqual> |
QUICHE team | 82dee2f | 2019-01-18 12:35:12 -0500 | [diff] [blame] | 115 | MapType; |
| 116 | |
| 117 | public: |
vasilvv | c8ccdb2 | 2020-10-12 16:42:34 -0700 | [diff] [blame] | 118 | typedef std::pair<absl::string_view, absl::string_view> value_type; |
QUICHE team | 82dee2f | 2019-01-18 12:35:12 -0500 | [diff] [blame] | 119 | |
vasilvv | c8ccdb2 | 2020-10-12 16:42:34 -0700 | [diff] [blame] | 120 | // Provides iteration over a sequence of std::pair<absl::string_view, |
| 121 | // absl::string_view>, even though the underlying MapType::value_type is |
QUICHE team | 82dee2f | 2019-01-18 12:35:12 -0500 | [diff] [blame] | 122 | // different. Dereferencing the iterator will result in memory allocation for |
| 123 | // multi-value headers. |
bnc | 45ccf4b | 2020-01-21 19:05:27 -0800 | [diff] [blame] | 124 | class QUICHE_EXPORT_PRIVATE iterator { |
QUICHE team | 82dee2f | 2019-01-18 12:35:12 -0500 | [diff] [blame] | 125 | public: |
| 126 | // The following type definitions fulfill the requirements for iterator |
| 127 | // implementations. |
vasilvv | c8ccdb2 | 2020-10-12 16:42:34 -0700 | [diff] [blame] | 128 | typedef std::pair<absl::string_view, absl::string_view> value_type; |
QUICHE team | 82dee2f | 2019-01-18 12:35:12 -0500 | [diff] [blame] | 129 | typedef value_type& reference; |
| 130 | typedef value_type* pointer; |
| 131 | typedef std::forward_iterator_tag iterator_category; |
| 132 | typedef MapType::iterator::difference_type difference_type; |
| 133 | |
| 134 | // In practice, this iterator only offers access to const value_type. |
| 135 | typedef const value_type& const_reference; |
| 136 | typedef const value_type* const_pointer; |
| 137 | |
| 138 | explicit iterator(MapType::const_iterator it); |
| 139 | iterator(const iterator& other); |
| 140 | ~iterator(); |
| 141 | |
| 142 | // This will result in memory allocation if the value consists of multiple |
| 143 | // fragments. |
dschinazi | b2b2615 | 2020-07-01 10:17:25 -0700 | [diff] [blame] | 144 | const_reference operator*() const { |
| 145 | #if SPDY_HEADER_DEBUG |
vasilvv | ed4f308 | 2021-02-01 14:29:40 -0800 | [diff] [blame] | 146 | QUICHE_CHECK(!dereference_forbidden_); |
dschinazi | b2b2615 | 2020-07-01 10:17:25 -0700 | [diff] [blame] | 147 | #endif // SPDY_HEADER_DEBUG |
| 148 | return it_->second.as_pair(); |
| 149 | } |
QUICHE team | 82dee2f | 2019-01-18 12:35:12 -0500 | [diff] [blame] | 150 | |
| 151 | const_pointer operator->() const { return &(this->operator*()); } |
| 152 | bool operator==(const iterator& it) const { return it_ == it.it_; } |
| 153 | bool operator!=(const iterator& it) const { return !(*this == it); } |
| 154 | |
| 155 | iterator& operator++() { |
| 156 | it_++; |
| 157 | return *this; |
| 158 | } |
| 159 | |
| 160 | iterator operator++(int) { |
| 161 | auto ret = *this; |
| 162 | this->operator++(); |
| 163 | return ret; |
| 164 | } |
| 165 | |
dschinazi | b2b2615 | 2020-07-01 10:17:25 -0700 | [diff] [blame] | 166 | #if SPDY_HEADER_DEBUG |
| 167 | void forbid_dereference() { dereference_forbidden_ = true; } |
| 168 | #endif // SPDY_HEADER_DEBUG |
| 169 | |
QUICHE team | 82dee2f | 2019-01-18 12:35:12 -0500 | [diff] [blame] | 170 | private: |
| 171 | MapType::const_iterator it_; |
dschinazi | b2b2615 | 2020-07-01 10:17:25 -0700 | [diff] [blame] | 172 | #if SPDY_HEADER_DEBUG |
| 173 | bool dereference_forbidden_ = false; |
| 174 | #endif // SPDY_HEADER_DEBUG |
QUICHE team | 82dee2f | 2019-01-18 12:35:12 -0500 | [diff] [blame] | 175 | }; |
| 176 | typedef iterator const_iterator; |
| 177 | |
QUICHE team | 557dfbc | 2020-10-20 10:52:30 -0700 | [diff] [blame] | 178 | Http2HeaderBlock(); |
| 179 | Http2HeaderBlock(const Http2HeaderBlock& other) = delete; |
| 180 | Http2HeaderBlock(Http2HeaderBlock&& other); |
| 181 | ~Http2HeaderBlock(); |
QUICHE team | 82dee2f | 2019-01-18 12:35:12 -0500 | [diff] [blame] | 182 | |
QUICHE team | 557dfbc | 2020-10-20 10:52:30 -0700 | [diff] [blame] | 183 | Http2HeaderBlock& operator=(const Http2HeaderBlock& other) = delete; |
| 184 | Http2HeaderBlock& operator=(Http2HeaderBlock&& other); |
| 185 | Http2HeaderBlock Clone() const; |
QUICHE team | 82dee2f | 2019-01-18 12:35:12 -0500 | [diff] [blame] | 186 | |
QUICHE team | 557dfbc | 2020-10-20 10:52:30 -0700 | [diff] [blame] | 187 | bool operator==(const Http2HeaderBlock& other) const; |
| 188 | bool operator!=(const Http2HeaderBlock& other) const; |
QUICHE team | 82dee2f | 2019-01-18 12:35:12 -0500 | [diff] [blame] | 189 | |
| 190 | // Provides a human readable multi-line representation of the stored header |
| 191 | // keys and values. |
bnc | 4471291 | 2019-08-15 18:58:14 -0700 | [diff] [blame] | 192 | std::string DebugString() const; |
QUICHE team | 82dee2f | 2019-01-18 12:35:12 -0500 | [diff] [blame] | 193 | |
dschinazi | b2b2615 | 2020-07-01 10:17:25 -0700 | [diff] [blame] | 194 | iterator begin() { return wrap_iterator(map_.begin()); } |
| 195 | iterator end() { return wrap_iterator(map_.end()); } |
| 196 | const_iterator begin() const { return wrap_const_iterator(map_.begin()); } |
| 197 | const_iterator end() const { return wrap_const_iterator(map_.end()); } |
QUICHE team | 8e3bb9d | 2019-12-06 16:03:22 -0800 | [diff] [blame] | 198 | bool empty() const { return map_.empty(); } |
| 199 | size_t size() const { return map_.size(); } |
vasilvv | c8ccdb2 | 2020-10-12 16:42:34 -0700 | [diff] [blame] | 200 | iterator find(absl::string_view key) { return wrap_iterator(map_.find(key)); } |
| 201 | const_iterator find(absl::string_view key) const { |
dschinazi | b2b2615 | 2020-07-01 10:17:25 -0700 | [diff] [blame] | 202 | return wrap_const_iterator(map_.find(key)); |
QUICHE team | 82dee2f | 2019-01-18 12:35:12 -0500 | [diff] [blame] | 203 | } |
vasilvv | c8ccdb2 | 2020-10-12 16:42:34 -0700 | [diff] [blame] | 204 | void erase(absl::string_view key); |
QUICHE team | 82dee2f | 2019-01-18 12:35:12 -0500 | [diff] [blame] | 205 | |
| 206 | // Clears both our MapType member and the memory used to hold headers. |
| 207 | void clear(); |
| 208 | |
| 209 | // The next few methods copy data into our backing storage. |
| 210 | |
| 211 | // If key already exists in the block, replaces the value of that key. Else |
| 212 | // adds a new header to the end of the block. |
| 213 | void insert(const value_type& value); |
| 214 | |
| 215 | // If a header with the key is already present, then append the value to the |
| 216 | // existing header value, NUL ("\0") separated unless the key is cookie, in |
| 217 | // which case the separator is "; ". |
| 218 | // If there is no such key, a new header with the key and value is added. |
vasilvv | c8ccdb2 | 2020-10-12 16:42:34 -0700 | [diff] [blame] | 219 | void AppendValueOrAddHeader(const absl::string_view key, |
| 220 | const absl::string_view value); |
QUICHE team | 82dee2f | 2019-01-18 12:35:12 -0500 | [diff] [blame] | 221 | |
QUICHE team | 557dfbc | 2020-10-20 10:52:30 -0700 | [diff] [blame] | 222 | // This object provides automatic conversions that allow Http2HeaderBlock to |
| 223 | // be nearly a drop-in replacement for |
bnc | 4471291 | 2019-08-15 18:58:14 -0700 | [diff] [blame] | 224 | // SpdyLinkedHashMap<std::string, std::string>. |
QUICHE team | 6b29700 | 2019-12-09 09:37:56 -0800 | [diff] [blame] | 225 | // It reads data from or writes data to a SpdyHeaderStorage. |
bnc | 45ccf4b | 2020-01-21 19:05:27 -0800 | [diff] [blame] | 226 | class QUICHE_EXPORT_PRIVATE ValueProxy { |
QUICHE team | 82dee2f | 2019-01-18 12:35:12 -0500 | [diff] [blame] | 227 | public: |
| 228 | ~ValueProxy(); |
| 229 | |
| 230 | // Moves are allowed. |
| 231 | ValueProxy(ValueProxy&& other); |
| 232 | ValueProxy& operator=(ValueProxy&& other); |
| 233 | |
| 234 | // Copies are not. |
| 235 | ValueProxy(const ValueProxy& other) = delete; |
| 236 | ValueProxy& operator=(const ValueProxy& other) = delete; |
| 237 | |
QUICHE team | 557dfbc | 2020-10-20 10:52:30 -0700 | [diff] [blame] | 238 | // Assignment modifies the underlying Http2HeaderBlock. |
vasilvv | c8ccdb2 | 2020-10-12 16:42:34 -0700 | [diff] [blame] | 239 | ValueProxy& operator=(absl::string_view value); |
QUICHE team | 3f57c75 | 2019-12-16 14:09:53 -0800 | [diff] [blame] | 240 | |
vasilvv | c8ccdb2 | 2020-10-12 16:42:34 -0700 | [diff] [blame] | 241 | // Provides easy comparison against absl::string_view. |
| 242 | bool operator==(absl::string_view value) const; |
QUICHE team | 82dee2f | 2019-01-18 12:35:12 -0500 | [diff] [blame] | 243 | |
bnc | 4471291 | 2019-08-15 18:58:14 -0700 | [diff] [blame] | 244 | std::string as_string() const; |
QUICHE team | 82dee2f | 2019-01-18 12:35:12 -0500 | [diff] [blame] | 245 | |
| 246 | private: |
QUICHE team | 557dfbc | 2020-10-20 10:52:30 -0700 | [diff] [blame] | 247 | friend class Http2HeaderBlock; |
QUICHE team | 82dee2f | 2019-01-18 12:35:12 -0500 | [diff] [blame] | 248 | friend class test::ValueProxyPeer; |
| 249 | |
QUICHE team | 557dfbc | 2020-10-20 10:52:30 -0700 | [diff] [blame] | 250 | ValueProxy(Http2HeaderBlock* block, |
| 251 | Http2HeaderBlock::MapType::iterator lookup_result, |
vasilvv | c8ccdb2 | 2020-10-12 16:42:34 -0700 | [diff] [blame] | 252 | const absl::string_view key, |
QUICHE team | 82dee2f | 2019-01-18 12:35:12 -0500 | [diff] [blame] | 253 | size_t* spdy_header_block_value_size); |
| 254 | |
QUICHE team | 557dfbc | 2020-10-20 10:52:30 -0700 | [diff] [blame] | 255 | Http2HeaderBlock* block_; |
| 256 | Http2HeaderBlock::MapType::iterator lookup_result_; |
vasilvv | c8ccdb2 | 2020-10-12 16:42:34 -0700 | [diff] [blame] | 257 | absl::string_view key_; |
QUICHE team | 82dee2f | 2019-01-18 12:35:12 -0500 | [diff] [blame] | 258 | size_t* spdy_header_block_value_size_; |
| 259 | bool valid_; |
| 260 | }; |
| 261 | |
QUICHE team | 8e3bb9d | 2019-12-06 16:03:22 -0800 | [diff] [blame] | 262 | // Allows either lookup or mutation of the value associated with a key. |
vasilvv | 50465ab | 2021-02-08 05:50:01 -0800 | [diff] [blame] | 263 | ABSL_MUST_USE_RESULT ValueProxy operator[](const absl::string_view key); |
QUICHE team | 8e3bb9d | 2019-12-06 16:03:22 -0800 | [diff] [blame] | 264 | |
QUICHE team | 82dee2f | 2019-01-18 12:35:12 -0500 | [diff] [blame] | 265 | // Returns the estimate of dynamically allocated memory in bytes. |
| 266 | size_t EstimateMemoryUsage() const; |
| 267 | |
| 268 | size_t TotalBytesUsed() const { return key_size_ + value_size_; } |
| 269 | |
| 270 | private: |
QUICHE team | 1bd18d1 | 2020-10-21 08:36:48 -0700 | [diff] [blame] | 271 | friend class test::Http2HeaderBlockPeer; |
QUICHE team | 82dee2f | 2019-01-18 12:35:12 -0500 | [diff] [blame] | 272 | |
dschinazi | b2b2615 | 2020-07-01 10:17:25 -0700 | [diff] [blame] | 273 | inline iterator wrap_iterator(MapType::const_iterator inner_iterator) const { |
| 274 | #if SPDY_HEADER_DEBUG |
| 275 | iterator outer_iterator(inner_iterator); |
| 276 | if (inner_iterator == map_.end()) { |
| 277 | outer_iterator.forbid_dereference(); |
| 278 | } |
| 279 | return outer_iterator; |
| 280 | #else // SPDY_HEADER_DEBUG |
| 281 | return iterator(inner_iterator); |
| 282 | #endif // SPDY_HEADER_DEBUG |
| 283 | } |
| 284 | |
| 285 | inline const_iterator wrap_const_iterator( |
| 286 | MapType::const_iterator inner_iterator) const { |
| 287 | #if SPDY_HEADER_DEBUG |
| 288 | const_iterator outer_iterator(inner_iterator); |
| 289 | if (inner_iterator == map_.end()) { |
| 290 | outer_iterator.forbid_dereference(); |
| 291 | } |
| 292 | return outer_iterator; |
| 293 | #else // SPDY_HEADER_DEBUG |
| 294 | return iterator(inner_iterator); |
| 295 | #endif // SPDY_HEADER_DEBUG |
| 296 | } |
| 297 | |
vasilvv | c8ccdb2 | 2020-10-12 16:42:34 -0700 | [diff] [blame] | 298 | void AppendHeader(const absl::string_view key, const absl::string_view value); |
| 299 | absl::string_view WriteKey(const absl::string_view key); |
QUICHE team | 82dee2f | 2019-01-18 12:35:12 -0500 | [diff] [blame] | 300 | size_t bytes_allocated() const; |
| 301 | |
vasilvv | c8ccdb2 | 2020-10-12 16:42:34 -0700 | [diff] [blame] | 302 | // absl::string_views held by |map_| point to memory owned by |storage_|. |
QUICHE team | 8e3bb9d | 2019-12-06 16:03:22 -0800 | [diff] [blame] | 303 | MapType map_; |
QUICHE team | 0b74550 | 2019-12-10 11:02:08 -0800 | [diff] [blame] | 304 | SpdyHeaderStorage storage_; |
QUICHE team | 82dee2f | 2019-01-18 12:35:12 -0500 | [diff] [blame] | 305 | |
| 306 | size_t key_size_ = 0; |
| 307 | size_t value_size_ = 0; |
| 308 | }; |
| 309 | |
QUICHE team | 557dfbc | 2020-10-20 10:52:30 -0700 | [diff] [blame] | 310 | // TODO(b/156770486): Remove this alias when the rename is complete. |
| 311 | using SpdyHeaderBlock = Http2HeaderBlock; |
| 312 | |
QUICHE team | 82dee2f | 2019-01-18 12:35:12 -0500 | [diff] [blame] | 313 | } // namespace spdy |
| 314 | |
| 315 | #endif // QUICHE_SPDY_CORE_SPDY_HEADER_BLOCK_H_ |