1*61c4878aSAndroid Build Coastguard Worker // Copyright 2021 The Pigweed Authors
2*61c4878aSAndroid Build Coastguard Worker //
3*61c4878aSAndroid Build Coastguard Worker // Licensed under the Apache License, Version 2.0 (the "License"); you may not
4*61c4878aSAndroid Build Coastguard Worker // use this file except in compliance with the License. You may obtain a copy of
5*61c4878aSAndroid Build Coastguard Worker // the License at
6*61c4878aSAndroid Build Coastguard Worker //
7*61c4878aSAndroid Build Coastguard Worker // https://www.apache.org/licenses/LICENSE-2.0
8*61c4878aSAndroid Build Coastguard Worker //
9*61c4878aSAndroid Build Coastguard Worker // Unless required by applicable law or agreed to in writing, software
10*61c4878aSAndroid Build Coastguard Worker // distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
11*61c4878aSAndroid Build Coastguard Worker // WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
12*61c4878aSAndroid Build Coastguard Worker // License for the specific language governing permissions and limitations under
13*61c4878aSAndroid Build Coastguard Worker // the License.
14*61c4878aSAndroid Build Coastguard Worker
15*61c4878aSAndroid Build Coastguard Worker #include "pw_protobuf/map_utils.h"
16*61c4878aSAndroid Build Coastguard Worker
17*61c4878aSAndroid Build Coastguard Worker #include <cstddef>
18*61c4878aSAndroid Build Coastguard Worker
19*61c4878aSAndroid Build Coastguard Worker #include "pw_bytes/span.h"
20*61c4878aSAndroid Build Coastguard Worker #include "pw_protobuf/encoder.h"
21*61c4878aSAndroid Build Coastguard Worker #include "pw_protobuf/serialized_size.h"
22*61c4878aSAndroid Build Coastguard Worker #include "pw_stream/stream.h"
23*61c4878aSAndroid Build Coastguard Worker
24*61c4878aSAndroid Build Coastguard Worker namespace pw::protobuf {
25*61c4878aSAndroid Build Coastguard Worker
26*61c4878aSAndroid Build Coastguard Worker // Note that a map<string, bytes> is essentially
27*61c4878aSAndroid Build Coastguard Worker //
28*61c4878aSAndroid Build Coastguard Worker // message Entry {
29*61c4878aSAndroid Build Coastguard Worker // string key = 1;
30*61c4878aSAndroid Build Coastguard Worker // bytes value = 2;
31*61c4878aSAndroid Build Coastguard Worker // }
32*61c4878aSAndroid Build Coastguard Worker //
33*61c4878aSAndroid Build Coastguard Worker // message Msg {
34*61c4878aSAndroid Build Coastguard Worker // repeated Entry map_field = <field_number>;
35*61c4878aSAndroid Build Coastguard Worker // }
WriteProtoStringToBytesMapEntry(uint32_t field_number,stream::Reader & key,size_t key_size,stream::Reader & value,size_t value_size,ByteSpan stream_pipe_buffer,stream::Writer & writer)36*61c4878aSAndroid Build Coastguard Worker Status WriteProtoStringToBytesMapEntry(uint32_t field_number,
37*61c4878aSAndroid Build Coastguard Worker stream::Reader& key,
38*61c4878aSAndroid Build Coastguard Worker size_t key_size,
39*61c4878aSAndroid Build Coastguard Worker stream::Reader& value,
40*61c4878aSAndroid Build Coastguard Worker size_t value_size,
41*61c4878aSAndroid Build Coastguard Worker ByteSpan stream_pipe_buffer,
42*61c4878aSAndroid Build Coastguard Worker stream::Writer& writer) {
43*61c4878aSAndroid Build Coastguard Worker constexpr uint32_t kMapKeyFieldNumber = 1;
44*61c4878aSAndroid Build Coastguard Worker constexpr uint32_t kMapValueFieldNumber = 2;
45*61c4878aSAndroid Build Coastguard Worker
46*61c4878aSAndroid Build Coastguard Worker if (!protobuf::ValidFieldNumber(field_number) ||
47*61c4878aSAndroid Build Coastguard Worker key_size >= std::numeric_limits<uint32_t>::max() ||
48*61c4878aSAndroid Build Coastguard Worker value_size >= std::numeric_limits<uint32_t>::max()) {
49*61c4878aSAndroid Build Coastguard Worker return Status::InvalidArgument();
50*61c4878aSAndroid Build Coastguard Worker }
51*61c4878aSAndroid Build Coastguard Worker
52*61c4878aSAndroid Build Coastguard Worker Result<size_t> key_field_size = protobuf::SizeOfField(
53*61c4878aSAndroid Build Coastguard Worker kMapKeyFieldNumber, protobuf::WireType::kDelimited, key_size);
54*61c4878aSAndroid Build Coastguard Worker PW_TRY(key_field_size.status());
55*61c4878aSAndroid Build Coastguard Worker
56*61c4878aSAndroid Build Coastguard Worker Result<size_t> value_field_size = protobuf::SizeOfField(
57*61c4878aSAndroid Build Coastguard Worker kMapValueFieldNumber, protobuf::WireType::kDelimited, value_size);
58*61c4878aSAndroid Build Coastguard Worker PW_TRY(value_field_size.status());
59*61c4878aSAndroid Build Coastguard Worker
60*61c4878aSAndroid Build Coastguard Worker size_t entry_payload_total_size =
61*61c4878aSAndroid Build Coastguard Worker key_field_size.value() + value_field_size.value();
62*61c4878aSAndroid Build Coastguard Worker
63*61c4878aSAndroid Build Coastguard Worker Result<size_t> entry_field_total_size = protobuf::SizeOfField(
64*61c4878aSAndroid Build Coastguard Worker field_number, protobuf::WireType::kDelimited, entry_payload_total_size);
65*61c4878aSAndroid Build Coastguard Worker PW_TRY(entry_field_total_size.status());
66*61c4878aSAndroid Build Coastguard Worker
67*61c4878aSAndroid Build Coastguard Worker if (entry_field_total_size.value() > writer.ConservativeWriteLimit()) {
68*61c4878aSAndroid Build Coastguard Worker return Status::ResourceExhausted();
69*61c4878aSAndroid Build Coastguard Worker }
70*61c4878aSAndroid Build Coastguard Worker
71*61c4878aSAndroid Build Coastguard Worker // Write field key and length prefix for nested message `Entry`
72*61c4878aSAndroid Build Coastguard Worker PW_TRY(protobuf::WriteLengthDelimitedKeyAndLengthPrefix(
73*61c4878aSAndroid Build Coastguard Worker field_number, entry_payload_total_size, writer));
74*61c4878aSAndroid Build Coastguard Worker
75*61c4878aSAndroid Build Coastguard Worker protobuf::StreamEncoder encoder(writer, {});
76*61c4878aSAndroid Build Coastguard Worker
77*61c4878aSAndroid Build Coastguard Worker // Write Entry::key
78*61c4878aSAndroid Build Coastguard Worker PW_TRY(encoder.WriteStringFromStream(
79*61c4878aSAndroid Build Coastguard Worker kMapKeyFieldNumber, key, key_size, stream_pipe_buffer));
80*61c4878aSAndroid Build Coastguard Worker // Write Entry::value
81*61c4878aSAndroid Build Coastguard Worker PW_TRY(encoder.WriteBytesFromStream(
82*61c4878aSAndroid Build Coastguard Worker kMapValueFieldNumber, value, value_size, stream_pipe_buffer));
83*61c4878aSAndroid Build Coastguard Worker
84*61c4878aSAndroid Build Coastguard Worker return OkStatus();
85*61c4878aSAndroid Build Coastguard Worker }
86*61c4878aSAndroid Build Coastguard Worker
87*61c4878aSAndroid Build Coastguard Worker } // namespace pw::protobuf
88