1 /* 2 * Copyright 2020 The Android Open Source Project 3 * 4 * Licensed under the Apache License, Version 2.0 (the "License"); 5 * you may not use this file except in compliance with the License. 6 * You may obtain a copy of the License at 7 * 8 * http://www.apache.org/licenses/LICENSE-2.0 9 * 10 * Unless required by applicable law or agreed to in writing, software 11 * distributed under the License is distributed on an "AS IS" BASIS, 12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 13 * See the License for the specific language governing permissions and 14 * limitations under the License. 15 */ 16 17 #pragma once 18 19 #include <vector> 20 21 #include <aidl/android/hardware/security/keymint/BlockMode.h> 22 #include <aidl/android/hardware/security/keymint/Digest.h> 23 #include <aidl/android/hardware/security/keymint/EcCurve.h> 24 #include <aidl/android/hardware/security/keymint/PaddingMode.h> 25 26 #include <keymint_support/keymint_tags.h> 27 28 namespace aidl::android::hardware::security::keymint { 29 30 using std::vector; 31 32 class AuthorizationSetBuilder; 33 34 /** 35 * A collection of KeyParameters. It provides memory ownership and some convenient functionality for 36 * sorting, deduplicating, joining, and subtracting sets of KeyParameters. 37 */ 38 class AuthorizationSet { 39 public: 40 typedef KeyParameter value_type; 41 42 /** 43 * Construct an empty, dynamically-allocated, growable AuthorizationSet. 44 */ AuthorizationSet()45 AuthorizationSet(){}; 46 47 // Copy constructor. AuthorizationSet(const AuthorizationSet & other)48 AuthorizationSet(const AuthorizationSet& other) : data_(other.data_) {} 49 50 // Move constructor. AuthorizationSet(AuthorizationSet && other)51 AuthorizationSet(AuthorizationSet&& other) noexcept : data_(std::move(other.data_)) {} 52 53 // Constructor from vector<KeyParameter> AuthorizationSet(const vector<KeyParameter> & other)54 AuthorizationSet(const vector<KeyParameter>& other) { *this = other; } 55 56 // Copy assignment. 57 AuthorizationSet& operator=(const AuthorizationSet& other) { 58 data_ = other.data_; 59 return *this; 60 } 61 62 // Move assignment. 63 AuthorizationSet& operator=(AuthorizationSet&& other) noexcept { 64 data_ = std::move(other.data_); 65 return *this; 66 } 67 68 AuthorizationSet& operator=(const vector<KeyParameter>& other) { 69 if (other.size() > 0) { 70 data_.resize(other.size()); 71 for (size_t i = 0; i < data_.size(); ++i) { 72 /* This makes a deep copy even of embedded blobs. 73 * See assignment operator/copy constructor of vector.*/ 74 data_[i] = other[i]; 75 } 76 } 77 return *this; 78 } 79 80 /** 81 * Clear existing authorization set data 82 */ 83 void Clear(); 84 85 ~AuthorizationSet() = default; 86 87 /** 88 * Returns the size of the set. 89 */ size()90 size_t size() const { return data_.size(); } 91 92 /** 93 * Returns true if the set is empty. 94 */ empty()95 bool empty() const { return size() == 0; } 96 97 /** 98 * Returns the data in the set, directly. Be careful with this. 99 */ data()100 const KeyParameter* data() const { return data_.data(); } 101 102 /** 103 * Sorts the set 104 */ 105 void Sort(); 106 107 /** 108 * Sorts the set and removes duplicates (inadvertently duplicating tags is easy to do with the 109 * AuthorizationSetBuilder). 110 */ 111 void Deduplicate(); 112 113 /** 114 * Adds all elements from \p set that are not already present in this AuthorizationSet. As a 115 * side-effect, if \p set is not null this AuthorizationSet will end up sorted. 116 */ 117 void Union(const AuthorizationSet& set); 118 119 /** 120 * Removes all elements in \p set from this AuthorizationSet. 121 */ 122 void Subtract(const AuthorizationSet& set); 123 124 /** 125 * Returns the offset of the next entry that matches \p tag, starting from the element after \p 126 * begin. If not found, returns -1. 127 */ 128 int find(Tag tag, int begin = -1) const; 129 130 /** 131 * Removes the entry at the specified index. Returns true if successful, false if the index was 132 * out of bounds. 133 */ 134 bool erase(int index); 135 136 /** 137 * Returns iterator (pointer) to beginning of elems array, to enable STL-style iteration 138 */ begin()139 auto begin() { return data_.begin(); } begin()140 auto begin() const { return data_.begin(); } 141 142 /** 143 * Returns iterator (pointer) one past end of elems array, to enable STL-style iteration 144 */ end()145 auto end() { return data_.end(); } end()146 auto end() const { return data_.end(); } 147 148 /** 149 * Returns the nth element of the set. 150 * Like for std::vector::operator[] there is no range check performed. Use of out of range 151 * indices is undefined. 152 */ 153 KeyParameter& operator[](int n); 154 155 /** 156 * Returns the nth element of the set. 157 * Like for std::vector::operator[] there is no range check performed. Use of out of range 158 * indices is undefined. 159 */ 160 const KeyParameter& operator[](int n) const; 161 162 /** 163 * Returns true if the set contains at least one instance of \p tag 164 */ Contains(Tag tag)165 bool Contains(Tag tag) const { return find(tag) != -1; } 166 167 template <TagType tag_type, Tag tag, typename ValueT> Contains(TypedTag<tag_type,tag> ttag,const ValueT & value)168 bool Contains(TypedTag<tag_type, tag> ttag, const ValueT& value) const { 169 for (const auto& param : data_) { 170 auto entry = authorizationValue(ttag, param); 171 if (entry && static_cast<ValueT>(*entry) == value) return true; 172 } 173 return false; 174 } 175 /** 176 * Returns the number of \p tag entries. 177 */ 178 size_t GetTagCount(Tag tag) const; 179 180 template <typename T> 181 inline auto GetTagValue(T tag) const -> decltype(authorizationValue(tag, KeyParameter())) { 182 auto entry = GetEntry(tag); 183 if (entry) return authorizationValue(tag, *entry); 184 return {}; 185 } 186 push_back(const KeyParameter & param)187 void push_back(const KeyParameter& param) { data_.push_back(param); } push_back(KeyParameter && param)188 void push_back(KeyParameter&& param) { data_.push_back(std::move(param)); } push_back(const AuthorizationSet & set)189 void push_back(const AuthorizationSet& set) { 190 for (auto& entry : set) { 191 push_back(entry); 192 } 193 } push_back(AuthorizationSet && set)194 void push_back(AuthorizationSet&& set) { 195 std::move(set.begin(), set.end(), std::back_inserter(*this)); 196 } 197 198 /** 199 * Append the tag and enumerated value to the set. 200 * "val" may be exactly one parameter unless a boolean parameter is added. 201 * In this case "val" is omitted. This condition is checked at compile time by Authorization() 202 */ 203 template <typename TypedTagT, typename... Value> push_back(TypedTagT tag,Value &&...val)204 void push_back(TypedTagT tag, Value&&... val) { 205 push_back(Authorization(tag, std::forward<Value>(val)...)); 206 } 207 208 template <typename Iterator> append(Iterator begin,Iterator end)209 void append(Iterator begin, Iterator end) { 210 while (begin != end) { 211 push_back(*begin); 212 ++begin; 213 } 214 } 215 vector_data()216 vector<KeyParameter> vector_data() const { 217 vector<KeyParameter> result(begin(), end()); 218 return result; 219 } 220 221 private: 222 std::optional<std::reference_wrapper<const KeyParameter>> GetEntry(Tag tag) const; 223 224 std::vector<KeyParameter> data_; 225 }; 226 227 class AuthorizationSetBuilder : public AuthorizationSet { 228 public: 229 template <typename TagType, typename... ValueType> Authorization(TagType ttag,ValueType &&...value)230 AuthorizationSetBuilder& Authorization(TagType ttag, ValueType&&... value) { 231 push_back(ttag, std::forward<ValueType>(value)...); 232 return *this; 233 } 234 235 template <Tag tag> Authorization(TypedTag<TagType::BYTES,tag> ttag,const uint8_t * data,size_t data_length)236 AuthorizationSetBuilder& Authorization(TypedTag<TagType::BYTES, tag> ttag, const uint8_t* data, 237 size_t data_length) { 238 vector<uint8_t> new_blob(data, data + data_length); 239 push_back(ttag, new_blob); 240 return *this; 241 } 242 243 template <Tag tag> Authorization(TypedTag<TagType::BYTES,tag> ttag,const char * data,size_t data_length)244 AuthorizationSetBuilder& Authorization(TypedTag<TagType::BYTES, tag> ttag, const char* data, 245 size_t data_length) { 246 return Authorization(ttag, reinterpret_cast<const uint8_t*>(data), data_length); 247 } 248 249 template <Tag tag> Authorization(TypedTag<TagType::BYTES,tag> ttag,char * data,size_t data_length)250 AuthorizationSetBuilder& Authorization(TypedTag<TagType::BYTES, tag> ttag, char* data, 251 size_t data_length) { 252 return Authorization(ttag, reinterpret_cast<const uint8_t*>(data), data_length); 253 } 254 255 template <Tag tag, size_t size> Authorization(TypedTag<TagType::BYTES,tag> ttag,const char (& data)[size])256 AuthorizationSetBuilder& Authorization(TypedTag<TagType::BYTES, tag> ttag, 257 const char (&data)[size]) { 258 return Authorization(ttag, reinterpret_cast<const uint8_t*>(&data[0]), 259 size - 1); // drop the terminating '\0' 260 } 261 262 template <Tag tag> Authorization(TypedTag<TagType::BYTES,tag> ttag,const std::string & data)263 AuthorizationSetBuilder& Authorization(TypedTag<TagType::BYTES, tag> ttag, 264 const std::string& data) { 265 return Authorization(ttag, reinterpret_cast<const uint8_t*>(data.data()), data.size()); 266 } 267 Authorizations(const AuthorizationSet & set)268 AuthorizationSetBuilder& Authorizations(const AuthorizationSet& set) { 269 for (const auto& entry : set) { 270 push_back(entry); 271 } 272 return *this; 273 } 274 275 AuthorizationSetBuilder& RsaKey(uint32_t key_size, uint64_t public_exponent); 276 AuthorizationSetBuilder& EcdsaKey(uint32_t key_size); 277 AuthorizationSetBuilder& EcdsaKey(EcCurve curve); 278 AuthorizationSetBuilder& AesKey(uint32_t key_size); 279 AuthorizationSetBuilder& TripleDesKey(uint32_t key_size); 280 AuthorizationSetBuilder& HmacKey(uint32_t key_size); 281 282 AuthorizationSetBuilder& RsaSigningKey(uint32_t key_size, uint64_t public_exponent); 283 AuthorizationSetBuilder& RsaEncryptionKey(uint32_t key_size, uint64_t public_exponent); 284 AuthorizationSetBuilder& EcdsaSigningKey(EcCurve curve); 285 AuthorizationSetBuilder& AesEncryptionKey(uint32_t key_size); 286 AuthorizationSetBuilder& TripleDesEncryptionKey(uint32_t key_size); 287 288 AuthorizationSetBuilder& SigningKey(); 289 AuthorizationSetBuilder& EncryptionKey(); 290 AuthorizationSetBuilder& AttestKey(); 291 292 AuthorizationSetBuilder& NoDigestOrPadding(); 293 294 AuthorizationSetBuilder& EcbMode(); 295 AuthorizationSetBuilder& GcmModeMinMacLen(uint32_t minMacLength); 296 AuthorizationSetBuilder& GcmModeMacLen(uint32_t macLength); 297 298 AuthorizationSetBuilder& BlockMode(std::initializer_list<BlockMode> blockModes); 299 AuthorizationSetBuilder& OaepMGFDigest(const std::vector<Digest>& digests); 300 AuthorizationSetBuilder& Digest(std::vector<Digest> digests); 301 AuthorizationSetBuilder& Padding(std::initializer_list<PaddingMode> paddings); 302 303 AuthorizationSetBuilder& SetDefaultValidity(); 304 AttestationChallenge(const std::string & challenge)305 AuthorizationSetBuilder& AttestationChallenge(const std::string& challenge) { 306 return Authorization(TAG_ATTESTATION_CHALLENGE, challenge); 307 } AttestationChallenge(std::vector<uint8_t> challenge)308 AuthorizationSetBuilder& AttestationChallenge(std::vector<uint8_t> challenge) { 309 return Authorization(TAG_ATTESTATION_CHALLENGE, challenge); 310 } 311 AttestationApplicationId(const std::string & id)312 AuthorizationSetBuilder& AttestationApplicationId(const std::string& id) { 313 return Authorization(TAG_ATTESTATION_APPLICATION_ID, id); 314 } AttestationApplicationId(std::vector<uint8_t> id)315 AuthorizationSetBuilder& AttestationApplicationId(std::vector<uint8_t> id) { 316 return Authorization(TAG_ATTESTATION_APPLICATION_ID, id); 317 } 318 319 template <typename... T> BlockMode(T &&...a)320 AuthorizationSetBuilder& BlockMode(T&&... a) { 321 return BlockMode({std::forward<T>(a)...}); 322 } 323 template <typename... T> Digest(T &&...a)324 AuthorizationSetBuilder& Digest(T&&... a) { 325 return Digest({std::forward<T>(a)...}); 326 } 327 template <typename... T> Padding(T &&...a)328 AuthorizationSetBuilder& Padding(T&&... a) { 329 return Padding({std::forward<T>(a)...}); 330 } 331 }; 332 333 } // namespace aidl::android::hardware::security::keymint 334