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