xref: /aosp_15_r20/external/tink/cc/util/secret_proto.h (revision e7b1675dde1b92d52ec075b0a92829627f2c52a5)
1*e7b1675dSTing-Kang Chang // Copyright 2021 Google LLC
2*e7b1675dSTing-Kang Chang //
3*e7b1675dSTing-Kang Chang // Licensed under the Apache License, Version 2.0 (the "License");
4*e7b1675dSTing-Kang Chang // you may not use this file except in compliance with the License.
5*e7b1675dSTing-Kang Chang // You may obtain a copy of the License at
6*e7b1675dSTing-Kang Chang //
7*e7b1675dSTing-Kang Chang //     http://www.apache.org/licenses/LICENSE-2.0
8*e7b1675dSTing-Kang Chang //
9*e7b1675dSTing-Kang Chang // Unless required by applicable law or agreed to in writing, software
10*e7b1675dSTing-Kang Chang // distributed under the License is distributed on an "AS IS" BASIS,
11*e7b1675dSTing-Kang Chang // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
12*e7b1675dSTing-Kang Chang // See the License for the specific language governing permissions and
13*e7b1675dSTing-Kang Chang // limitations under the License.
14*e7b1675dSTing-Kang Chang //
15*e7b1675dSTing-Kang Chang ///////////////////////////////////////////////////////////////////////////////
16*e7b1675dSTing-Kang Chang 
17*e7b1675dSTing-Kang Chang #ifndef TINK_UTIL_SECRET_PROTO_H_
18*e7b1675dSTing-Kang Chang #define TINK_UTIL_SECRET_PROTO_H_
19*e7b1675dSTing-Kang Chang 
20*e7b1675dSTing-Kang Chang #include <memory>
21*e7b1675dSTing-Kang Chang #include <utility>
22*e7b1675dSTing-Kang Chang 
23*e7b1675dSTing-Kang Chang #include "google/protobuf/arena.h"
24*e7b1675dSTing-Kang Chang #include "absl/memory/memory.h"
25*e7b1675dSTing-Kang Chang #include "tink/util/secret_data.h"
26*e7b1675dSTing-Kang Chang #include "tink/util/status.h"
27*e7b1675dSTing-Kang Chang #include "tink/util/statusor.h"
28*e7b1675dSTing-Kang Chang 
29*e7b1675dSTing-Kang Chang namespace crypto {
30*e7b1675dSTing-Kang Chang namespace tink {
31*e7b1675dSTing-Kang Chang namespace util {
32*e7b1675dSTing-Kang Chang 
33*e7b1675dSTing-Kang Chang namespace internal {
34*e7b1675dSTing-Kang Chang 
SecretArenaOptions()35*e7b1675dSTing-Kang Chang inline google::protobuf::ArenaOptions SecretArenaOptions() {
36*e7b1675dSTing-Kang Chang   google::protobuf::ArenaOptions options;
37*e7b1675dSTing-Kang Chang   options.block_alloc = [](size_t sz) {
38*e7b1675dSTing-Kang Chang     return SanitizingAllocator<void>().allocate(sz);
39*e7b1675dSTing-Kang Chang   };
40*e7b1675dSTing-Kang Chang   options.block_dealloc = [](void* ptr, size_t sz) {
41*e7b1675dSTing-Kang Chang     return SanitizingAllocator<void>().deallocate(ptr, sz);
42*e7b1675dSTing-Kang Chang   };
43*e7b1675dSTing-Kang Chang   return options;
44*e7b1675dSTing-Kang Chang }
45*e7b1675dSTing-Kang Chang 
46*e7b1675dSTing-Kang Chang }  // namespace internal
47*e7b1675dSTing-Kang Chang 
48*e7b1675dSTing-Kang Chang // Stores secret (sensitive) protobuf and makes sure it's marked as such and
49*e7b1675dSTing-Kang Chang // destroyed in a safe way.
50*e7b1675dSTing-Kang Chang //
51*e7b1675dSTing-Kang Chang // Note: Currently does not protect fields of type "string" and "bytes"
52*e7b1675dSTing-Kang Chang // (depends on https://github.com/protocolbuffers/protobuf/issues/1896)
53*e7b1675dSTing-Kang Chang template <typename T>
54*e7b1675dSTing-Kang Chang class SecretProto {
55*e7b1675dSTing-Kang Chang  public:
ParseFromSecretData(const SecretData & data)56*e7b1675dSTing-Kang Chang   static StatusOr<SecretProto<T>> ParseFromSecretData(const SecretData& data) {
57*e7b1675dSTing-Kang Chang     SecretProto<T> proto;
58*e7b1675dSTing-Kang Chang     if (!proto->ParseFromArray(data.data(), data.size())) {
59*e7b1675dSTing-Kang Chang       return Status(absl::StatusCode::kInternal, "Could not parse proto");
60*e7b1675dSTing-Kang Chang     }
61*e7b1675dSTing-Kang Chang     return proto;
62*e7b1675dSTing-Kang Chang   }
63*e7b1675dSTing-Kang Chang 
64*e7b1675dSTing-Kang Chang   SecretProto() = default;
65*e7b1675dSTing-Kang Chang 
SecretProto(const SecretProto & other)66*e7b1675dSTing-Kang Chang   SecretProto(const SecretProto& other) { *value_ = *other.value_; }
67*e7b1675dSTing-Kang Chang 
SecretProto(SecretProto && other)68*e7b1675dSTing-Kang Chang   SecretProto(SecretProto&& other) { *this = std::move(other); }
69*e7b1675dSTing-Kang Chang 
SecretProto(const T & value)70*e7b1675dSTing-Kang Chang   explicit SecretProto(const T& value) { *value_ = value; }
71*e7b1675dSTing-Kang Chang 
72*e7b1675dSTing-Kang Chang   SecretProto& operator=(const SecretProto& other) {
73*e7b1675dSTing-Kang Chang     *value_ = *other.value_;
74*e7b1675dSTing-Kang Chang     return *this;
75*e7b1675dSTing-Kang Chang   }
76*e7b1675dSTing-Kang Chang 
77*e7b1675dSTing-Kang Chang   SecretProto& operator=(SecretProto&& other) {
78*e7b1675dSTing-Kang Chang     using std::swap;
79*e7b1675dSTing-Kang Chang     swap(arena_, other.arena_);
80*e7b1675dSTing-Kang Chang     swap(value_, other.value_);
81*e7b1675dSTing-Kang Chang     return *this;
82*e7b1675dSTing-Kang Chang   }
83*e7b1675dSTing-Kang Chang 
get()84*e7b1675dSTing-Kang Chang   inline T* get() { return value_; }
85*e7b1675dSTing-Kang Chang 
86*e7b1675dSTing-Kang Chang   // Accessors to the underlying message.
87*e7b1675dSTing-Kang Chang   inline T* operator->() { return value_; }
88*e7b1675dSTing-Kang Chang   inline const T* operator->() const { return value_; }
89*e7b1675dSTing-Kang Chang 
90*e7b1675dSTing-Kang Chang   inline T& operator*() { return *value_; }
91*e7b1675dSTing-Kang Chang   inline const T& operator*() const { return *value_; }
92*e7b1675dSTing-Kang Chang 
SerializeAsSecretData()93*e7b1675dSTing-Kang Chang   StatusOr<SecretData> SerializeAsSecretData() const {
94*e7b1675dSTing-Kang Chang     SecretData data(value_->ByteSizeLong());
95*e7b1675dSTing-Kang Chang     if (!value_->SerializeToArray(data.data(), data.size())) {
96*e7b1675dSTing-Kang Chang       return Status(absl::StatusCode::kInternal, "Could not serialize proto");
97*e7b1675dSTing-Kang Chang     }
98*e7b1675dSTing-Kang Chang     return data;
99*e7b1675dSTing-Kang Chang   }
100*e7b1675dSTing-Kang Chang 
101*e7b1675dSTing-Kang Chang  private:
102*e7b1675dSTing-Kang Chang   std::unique_ptr<google::protobuf::Arena> arena_ =
103*e7b1675dSTing-Kang Chang       absl::make_unique<google::protobuf::Arena>(internal::SecretArenaOptions());
104*e7b1675dSTing-Kang Chang   T* value_ = google::protobuf::Arena::CreateMessage<T>(arena_.get());
105*e7b1675dSTing-Kang Chang };
106*e7b1675dSTing-Kang Chang 
107*e7b1675dSTing-Kang Chang }  // namespace util
108*e7b1675dSTing-Kang Chang }  // namespace tink
109*e7b1675dSTing-Kang Chang }  // namespace crypto
110*e7b1675dSTing-Kang Chang 
111*e7b1675dSTing-Kang Chang #endif  // TINK_UTIL_SECRET_PROTO_H_
112