C/C++ API Reference
Loading...
Searching...
No Matches
multibuf_v1.h
1// Copyright 2023 The Pigweed Authors
2//
3// Licensed under the Apache License, Version 2.0 (the "License"); you may not
4// use this file except in compliance with the License. You may obtain a copy of
5// the License at
6//
7// https://www.apache.org/licenses/LICENSE-2.0
8//
9// Unless required by applicable law or agreed to in writing, software
10// distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
11// WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
12// License for the specific language governing permissions and limitations under
13// the License.
14#pragma once
15
16#include <iterator>
17#include <tuple>
18
19#include "pw_multibuf/chunk.h"
20#include "pw_multibuf/config.h"
21#include "pw_preprocessor/compiler.h"
22#include "pw_status/status_with_size.h"
23
24namespace pw::multibuf {
25
27
28class MultiBuf;
29
31class PW_MULTIBUF_DEPRECATED MultiBufChunks {
32 public:
33 using element_type = Chunk;
34 using value_type = Chunk;
35 using pointer = Chunk*;
36 using reference = Chunk&;
37 using const_pointer = const Chunk*;
38 using difference_type = std::ptrdiff_t;
39 using const_reference = const Chunk&;
40 using size_type = std::size_t;
41
43 class iterator {
44 public:
45 using value_type = Chunk;
46 using difference_type = std::ptrdiff_t;
47 using reference = Chunk&;
48 using pointer = Chunk*;
49 using iterator_category = std::forward_iterator_tag;
50
51 constexpr iterator() = default;
52
53 constexpr reference operator*() const { return *chunk_; }
54 constexpr pointer operator->() const { return chunk_; }
55
56 constexpr iterator& operator++() {
57 chunk_ = chunk_->next_in_buf_;
58 return *this;
59 }
60
61 constexpr iterator operator++(int) {
62 iterator tmp = *this;
63 ++(*this);
64 return tmp;
65 }
66
67 constexpr bool operator==(const iterator& other) const {
68 return chunk_ == other.chunk_;
69 }
70
71 constexpr bool operator!=(const iterator& other) const {
72 return chunk_ != other.chunk_;
73 }
74
75 private:
76 friend class MultiBufChunks;
77 friend class MultiBuf;
78
79 constexpr iterator(Chunk* chunk) : chunk_(chunk) {}
80 static constexpr iterator end() { return iterator(nullptr); }
81 Chunk* chunk_ = nullptr;
82 };
83
86 public:
87 using value_type = const Chunk;
88 using difference_type = std::ptrdiff_t;
89 using reference = const Chunk&;
90 using pointer = const Chunk*;
91 using iterator_category = std::forward_iterator_tag;
92
93 constexpr const_iterator() = default;
94
95 constexpr reference operator*() const { return *chunk_; }
96 constexpr pointer operator->() const { return chunk_; }
97
98 constexpr const_iterator& operator++() {
99 chunk_ = chunk_->next_in_buf_;
100 return *this;
101 }
102
103 constexpr const_iterator operator++(int) {
104 const_iterator tmp = *this;
105 ++(*this);
106 return tmp;
107 }
108
109 constexpr bool operator==(const const_iterator& other) const {
110 return chunk_ == other.chunk_;
111 }
112
113 constexpr bool operator!=(const const_iterator& other) const {
114 return chunk_ != other.chunk_;
115 }
116
117 private:
118 friend class MultiBufChunks;
119 friend class MultiBuf;
120
121 constexpr const_iterator(const Chunk* chunk) : chunk_(chunk) {}
122 static constexpr const_iterator end() { return const_iterator(nullptr); }
123 const Chunk* chunk_ = nullptr;
124 };
125
126 MultiBufChunks(const MultiBufChunks&) = delete;
127 MultiBufChunks& operator=(const MultiBufChunks&) = delete;
128
132 constexpr Chunk& front() { return *first_; }
133 constexpr const Chunk& front() const { return *first_; }
134
140 Chunk& back() { return const_cast<Chunk&>(std::as_const(*this).back()); }
141 const Chunk& back() const;
142
143 constexpr iterator begin() { return iterator(first_); }
144 constexpr const_iterator begin() const { return cbegin(); }
145 constexpr const_iterator cbegin() const { return const_iterator(first_); }
146
147 constexpr iterator end() { return iterator::end(); }
148 constexpr const_iterator end() const { return cend(); }
149 constexpr const_iterator cend() const { return const_iterator::end(); }
150
152 size_t size() const {
153 return static_cast<size_t>(std::distance(begin(), end()));
154 }
155
157 size_t size_bytes() const;
158
160 [[nodiscard]] bool empty() const { return first_ == nullptr; }
161
165 void push_front(OwnedChunk&& chunk);
166
170 void push_back(OwnedChunk&& chunk);
171
176
183 //
184 // Implementation note: `Chunks().size()` should be remain relatively small,
185 // but this could be made `O(1)` in the future by adding a `prev` pointer to
186 // the `iterator`.
187 iterator insert(iterator position, OwnedChunk&& chunk);
188
195 //
196 // Implementation note: `Chunks().size()` should be remain relatively small,
197 // but this could be made `O(1)` in the future by adding a `prev` pointer to
198 // the `iterator`.
199 std::tuple<iterator, OwnedChunk> take(iterator position);
200
201 protected:
202 explicit constexpr MultiBufChunks(Chunk* first_chunk) : first_(first_chunk) {}
203
205 ~MultiBufChunks() { Release(); }
206
207 // Disable maybe-uninitialized: this check fails erroneously on Windows GCC.
209 PW_MODIFY_DIAGNOSTIC_GCC(ignored, "-Wmaybe-uninitialized");
210 constexpr MultiBufChunks(MultiBufChunks&& other) noexcept
211 : first_(other.first_) {
212 other.first_ = nullptr;
213 }
215
216 MultiBufChunks& operator=(MultiBufChunks&& other) noexcept {
217 Release();
218 first_ = other.first_;
219 other.first_ = nullptr;
220 return *this;
221 }
222
223 // Releases all chunks in the `MultiBuf`.
224 void Release() noexcept;
225
226 void PushSuffix(MultiBufChunks&& tail);
227
235 Chunk* Previous(Chunk* chunk) const;
236
237 private:
238 Chunk* first_;
239};
240
248class PW_MULTIBUF_DEPRECATED MultiBuf : private MultiBufChunks {
249 public:
250 class iterator;
251 class const_iterator;
252
253 constexpr MultiBuf() : MultiBufChunks(nullptr) {}
254
255 static MultiBuf FromChunk(OwnedChunk&& chunk) {
256 return MultiBuf(std::move(chunk).Take());
257 }
258
259 MultiBuf(const MultiBuf&) = delete;
260 MultiBuf& operator=(const MultiBuf&) = delete;
261
262 constexpr MultiBuf(MultiBuf&& other) noexcept = default;
263
264 MultiBuf& operator=(MultiBuf&& other) noexcept = default;
265
274 void Release() noexcept { MultiBufChunks::Release(); }
275
277 ~MultiBuf() = default;
278
282 [[nodiscard]] size_t size() const { return MultiBufChunks::size_bytes(); }
283
288 [[nodiscard]] bool empty() const;
289
296 [[nodiscard]] bool IsContiguous() const {
297 return ContiguousSpan().has_value();
298 }
299
308 std::optional<ByteSpan> ContiguousSpan() {
309 auto result = std::as_const(*this).ContiguousSpan();
310 if (result.has_value()) {
311 return span(const_cast<std::byte*>(result->data()), result->size());
312 }
313 return std::nullopt;
314 }
315 std::optional<ConstByteSpan> ContiguousSpan() const;
316
318 iterator begin() { return iterator(Chunks().begin().chunk_); }
321 return const_iterator(Chunks().begin().chunk_);
322 }
325 return const_iterator(Chunks().begin().chunk_);
326 }
327
329 iterator end() { return iterator::end(); }
331 const_iterator end() const { return const_iterator::end(); }
333 const_iterator cend() const { return const_iterator::end(); }
334
343 [[nodiscard]] bool ClaimPrefix(size_t bytes_to_claim);
344
353 [[nodiscard]] bool ClaimSuffix(size_t bytes_to_claim);
354
363 void DiscardPrefix(size_t bytes_to_discard);
364
373 void Slice(size_t begin, size_t end);
374
382 void Truncate(size_t len);
383
391
400 std::optional<MultiBuf> TakePrefix(size_t bytes_to_take);
401
410 std::optional<MultiBuf> TakeSuffix(size_t bytes_to_take);
411
415 void PushPrefix(MultiBuf&& front);
416
420 void PushSuffix(MultiBuf&& tail) {
421 return MultiBufChunks::PushSuffix(std::move(tail.Chunks()));
422 }
423
442 StatusWithSize CopyTo(ByteSpan dest, size_t position = 0) const;
443
461 StatusWithSize CopyFrom(ConstByteSpan source, size_t position = 0) {
462 return CopyFromAndOptionallyTruncate(source, position, /*truncate=*/false);
463 }
464
493 size_t position = 0) {
494 return CopyFromAndOptionallyTruncate(source, position, /*truncate=*/true);
495 }
496
498 //--------------------- Chunk manipulation ----------------------//
500
503 MultiBufChunks::push_front(std::move(chunk));
504 }
505
507 void PushBackChunk(OwnedChunk&& chunk) {
508 MultiBufChunks::push_back(std::move(chunk));
509 }
510
512 OwnedChunk TakeFrontChunk() { return MultiBufChunks::take_front(); }
513
516 OwnedChunk&& chunk) {
517 return MultiBufChunks::insert(position, std::move(chunk));
518 }
519
521 std::tuple<MultiBufChunks::iterator, OwnedChunk> TakeChunk(
522 MultiBufChunks::iterator position) {
523 return MultiBufChunks::take(position);
524 }
525
527 constexpr MultiBufChunks& Chunks() { return *this; }
528
530 constexpr const MultiBufChunks& Chunks() const { return *this; }
531
533 constexpr const MultiBufChunks& ConstChunks() const { return *this; }
534
536 //--------------------- Iterator details ------------------------//
538
539 using element_type = std::byte;
540 using value_type = std::byte;
541 using pointer = std::byte*;
542 using const_pointer = const std::byte*;
543 using reference = std::byte&;
544 using const_reference = const std::byte&;
545 using difference_type = std::ptrdiff_t;
546 using size_type = std::size_t;
547
550 public:
551 using value_type = std::byte;
552 using difference_type = std::ptrdiff_t;
553 using reference = const std::byte&;
554 using pointer = const std::byte*;
555 using iterator_category = std::forward_iterator_tag;
556
557 constexpr const_iterator() : chunk_(nullptr), byte_index_(0) {}
558
559 reference operator*() const { return (*chunk_)[byte_index_]; }
560 pointer operator->() const { return &(*chunk_)[byte_index_]; }
561
562 const_iterator& operator++();
563 const_iterator operator++(int) {
564 const_iterator tmp = *this;
565 ++(*this);
566 return tmp;
567 }
568 const_iterator operator+(size_t rhs) const {
569 const_iterator tmp = *this;
570 tmp += rhs;
571 return tmp;
572 }
573 const_iterator& operator+=(size_t advance);
574
575 constexpr bool operator==(const const_iterator& other) const {
576 return chunk_ == other.chunk_ && byte_index_ == other.byte_index_;
577 }
578
579 constexpr bool operator!=(const const_iterator& other) const {
580 return !(*this == other);
581 }
582
584 constexpr const Chunk* chunk() const { return chunk_; }
585
588 constexpr size_t byte_index() const { return byte_index_; }
589
590 private:
591 friend class MultiBuf;
592
593 explicit constexpr const_iterator(const Chunk* chunk, size_t byte_index = 0)
594 : chunk_(chunk), byte_index_(byte_index) {
595 AdvanceToData();
596 }
597
598 static const_iterator end() { return const_iterator(nullptr); }
599
600 constexpr void AdvanceToData() {
601 while (chunk_ != nullptr && chunk_->empty()) {
602 chunk_ = chunk_->next_in_buf_;
603 }
604 }
605
606 const Chunk* chunk_;
607 size_t byte_index_;
608 };
609
611 class iterator {
612 public:
613 using value_type = std::byte;
614 using difference_type = std::ptrdiff_t;
615 using reference = std::byte&;
616 using pointer = std::byte*;
617 using iterator_category = std::forward_iterator_tag;
618
619 constexpr iterator() = default;
620
621 reference operator*() const { return const_cast<std::byte&>(*const_iter_); }
622 pointer operator->() const { return const_cast<std::byte*>(&*const_iter_); }
623
624 iterator& operator++() {
625 const_iter_++;
626 return *this;
627 }
628 iterator operator++(int) {
629 iterator tmp = *this;
630 ++(*this);
631 return tmp;
632 }
633 iterator operator+(size_t rhs) const {
634 iterator tmp = *this;
635 tmp += rhs;
636 return tmp;
637 }
638 iterator& operator+=(size_t rhs) {
639 const_iter_ += rhs;
640 return *this;
641 }
642 constexpr bool operator==(const iterator& other) const {
643 return const_iter_ == other.const_iter_;
644 }
645 constexpr bool operator!=(const iterator& other) const {
646 return const_iter_ != other.const_iter_;
647 }
648
650 constexpr Chunk* chunk() const {
651 return const_cast<Chunk*>(const_iter_.chunk());
652 }
653
656 constexpr size_t byte_index() const { return const_iter_.byte_index(); }
657
658 private:
659 friend class MultiBuf;
660
661 explicit constexpr iterator(Chunk* chunk, size_t byte_index = 0)
662 : const_iter_(chunk, byte_index) {}
663
664 static iterator end() { return iterator(nullptr); }
665
666 const_iterator const_iter_;
667 };
668
669 private:
670 explicit constexpr MultiBuf(Chunk* first_chunk)
671 : MultiBufChunks(first_chunk) {}
672
673 StatusWithSize CopyFromAndOptionallyTruncate(ConstByteSpan source,
674 size_t position,
675 bool truncate);
676};
677
679
680} // namespace pw::multibuf
Definition: status_with_size.h:49
Definition: chunk.h:50
A const std::forward_iterator over the bytes of a MultiBuf.
Definition: multibuf_v1.h:549
constexpr size_t byte_index() const
Definition: multibuf_v1.h:588
constexpr const Chunk * chunk() const
Returns the current Chunk pointed to by this iterator.
Definition: multibuf_v1.h:584
An std::forward_iterator over the bytes of a MultiBuf.
Definition: multibuf_v1.h:611
constexpr size_t byte_index() const
Definition: multibuf_v1.h:656
constexpr Chunk * chunk() const
Returns the current Chunk pointed to by this iterator.
Definition: multibuf_v1.h:650
A const std::forward_iterator over the Chunks of a MultiBuf.
Definition: multibuf_v1.h:85
A std::forward_iterator over the Chunks of a MultiBuf.
Definition: multibuf_v1.h:43
A Chunk-oriented view of a MultiBuf.
Definition: multibuf_v1.h:31
bool empty() const
Returns whether the MultiBuf contains any chunks (size() == 0).
Definition: multibuf_v1.h:160
constexpr Chunk & front()
Definition: multibuf_v1.h:132
std::tuple< iterator, OwnedChunk > take(iterator position)
void push_back(OwnedChunk &&chunk)
void push_front(OwnedChunk &&chunk)
iterator insert(iterator position, OwnedChunk &&chunk)
Chunk & back()
Definition: multibuf_v1.h:140
~MultiBufChunks()
This destructor will acquire a mutex and is not IRQ safe.
Definition: multibuf_v1.h:205
size_t size_bytes() const
Returns the total number of bytes in all Chunks.
size_t size() const
Returns the number of Chunks in this MultiBuf, including empty chunks.
Definition: multibuf_v1.h:152
Definition: multibuf_v1.h:248
~MultiBuf()=default
This destructor will acquire a mutex and is not IRQ safe.
void PushBackChunk(OwnedChunk &&chunk)
Definition: multibuf_v1.h:507
constexpr MultiBufChunks & Chunks()
Returns a Chunk-oriented view of this MultiBuf.
Definition: multibuf_v1.h:527
void Release() noexcept
Definition: multibuf_v1.h:274
MultiBufChunks::iterator InsertChunk(MultiBufChunks::iterator position, OwnedChunk &&chunk)
Definition: multibuf_v1.h:515
const_iterator cbegin() const
Returns a const iterator pointing to the first byte of this MultiBuf.
Definition: multibuf_v1.h:324
void PushSuffix(MultiBuf &&tail)
Definition: multibuf_v1.h:420
std::optional< MultiBuf > TakeSuffix(size_t bytes_to_take)
void TruncateAfter(iterator pos)
void DiscardPrefix(size_t bytes_to_discard)
void PushFrontChunk(OwnedChunk &&chunk)
Definition: multibuf_v1.h:502
const_iterator begin() const
Returns a const iterator pointing to the first byte of this MultiBuf.
Definition: multibuf_v1.h:320
const_iterator end() const
Returns a const iterator pointing to the end of this MultiBuf.
Definition: multibuf_v1.h:331
StatusWithSize CopyTo(ByteSpan dest, size_t position=0) const
iterator begin()
Returns an iterator pointing to the first byte of this MultiBuf.
Definition: multibuf_v1.h:318
StatusWithSize CopyFromAndTruncate(ConstByteSpan source, size_t position=0)
Definition: multibuf_v1.h:492
bool IsContiguous() const
Definition: multibuf_v1.h:296
constexpr const MultiBufChunks & ConstChunks() const
Returns a const Chunk-oriented view of this MultiBuf.
Definition: multibuf_v1.h:533
void Truncate(size_t len)
iterator end()
Returns an iterator pointing to the end of this MultiBuf.
Definition: multibuf_v1.h:329
StatusWithSize CopyFrom(ConstByteSpan source, size_t position=0)
Definition: multibuf_v1.h:461
bool ClaimSuffix(size_t bytes_to_claim)
std::optional< MultiBuf > TakePrefix(size_t bytes_to_take)
std::tuple< MultiBufChunks::iterator, OwnedChunk > TakeChunk(MultiBufChunks::iterator position)
Definition: multibuf_v1.h:521
OwnedChunk TakeFrontChunk()
Definition: multibuf_v1.h:512
bool ClaimPrefix(size_t bytes_to_claim)
constexpr const MultiBufChunks & Chunks() const
Returns a const Chunk-oriented view of this MultiBuf.
Definition: multibuf_v1.h:530
std::optional< ByteSpan > ContiguousSpan()
Definition: multibuf_v1.h:308
void PushPrefix(MultiBuf &&front)
const_iterator cend() const
Returns a const iterator pointing to the end of this MultiBuf.
Definition: multibuf_v1.h:333
size_t size() const
Definition: multibuf_v1.h:282
void Slice(size_t begin, size_t end)
Definition: chunk.h:326
Definition: span_impl.h:235
BasicMultiBuf< MultiBufProperty::kLayerable > MultiBuf
Definition: multibuf_v2.h:71
#define PW_MODIFY_DIAGNOSTICS_POP()
Definition: compiler.h:194
#define PW_MODIFY_DIAGNOSTIC_GCC(kind, option)
Definition: compiler.h:211
#define PW_MODIFY_DIAGNOSTICS_PUSH()
Definition: compiler.h:189