/* ** ** Copyright 2017, The Android Open Source Project ** ** Licensed under the Apache License, Version 2.0 (the "License"); ** you may not use this file except in compliance with the License. ** You may obtain a copy of the License at ** ** http://www.apache.org/licenses/LICENSE-2.0 ** ** Unless required by applicable law or agreed to in writing, software ** distributed under the License is distributed on an "AS IS" BASIS, ** WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. ** See the License for the specific language governing permissions and ** limitations under the License. */ #ifndef SYSTEM_KEYMASTER_KEYMASTER_PASSTHROUGH_OPERATION_H_ #define SYSTEM_KEYMASTER_KEYMASTER_PASSTHROUGH_OPERATION_H_ #include <hardware/keymaster1.h> #include <hardware/keymaster2.h> #include <keymaster/legacy_support/keymaster_passthrough_key.h> #include <keymaster/operation.h> namespace keymaster { class AuthorizationSet; class Key; class Operation; /** * Template implementation for KM1 and KM2 operations */ template <typename KeymasterDeviceType> class KeymasterPassthroughOperation : public Operation { public: explicit KeymasterPassthroughOperation(keymaster_purpose_t purpose, const KeymasterDeviceType* km_device, Key&& key) : Operation(purpose, key.hw_enforced_move(), key.sw_enforced_move()), key_blob_(key.key_material_move()), km_device_(km_device) { operation_handle_ = 0; } virtual ~KeymasterPassthroughOperation() {} keymaster_error_t Begin(const AuthorizationSet& input_params, AuthorizationSet* output_params) override { keymaster_key_param_set_t out_params = {}; keymaster_error_t rc; rc = km_device_->begin(km_device_, purpose(), &key_blob_, &input_params, &out_params, &operation_handle_); if (rc == KM_ERROR_OK && output_params) output_params->Reinitialize(out_params); keymaster_free_param_set(&out_params); return rc; } keymaster_error_t Update(const AuthorizationSet& input_params, const Buffer& input, AuthorizationSet* output_params, Buffer* output, size_t* input_consumed) override { keymaster_key_param_set_t out_params = {}; keymaster_blob_t in{input.peek_read(), input.available_read()}; keymaster_blob_t out = {}; keymaster_error_t rc; rc = km_device_->update(km_device_, operation_handle_, &input_params, &in, input_consumed, &out_params, &out); if (rc == KM_ERROR_OK) { if (output) output->Reinitialize(out.data, out.data_length); if (output_params) output_params->Reinitialize(out_params); } keymaster_free_param_set(&out_params); free(const_cast<uint8_t*>(out.data)); return rc; } keymaster_error_t Finish(const AuthorizationSet& input_params, const Buffer& input, const Buffer& signature, AuthorizationSet* output_params, Buffer* output) override; keymaster_error_t Abort() { return km_device_->abort(km_device_, operation_handle_); } private: KeymasterKeyBlob key_blob_; const KeymasterDeviceType* km_device_; }; template <> keymaster_error_t KeymasterPassthroughOperation<keymaster1_device_t>::Finish( const AuthorizationSet& input_params, const Buffer& input, const Buffer& signature, AuthorizationSet* output_params, Buffer* output); template <> keymaster_error_t KeymasterPassthroughOperation<keymaster2_device_t>::Finish( const AuthorizationSet& input_params, const Buffer& input, const Buffer& signature, AuthorizationSet* output_params, Buffer* output); template <typename KeymasterDeviceType> class KeymasterPassthroughOperationFactory : public OperationFactory { public: KeymasterPassthroughOperationFactory(keymaster_algorithm_t algorithm, keymaster_purpose_t purpose, const KeymasterDeviceType* km_device) : key_type_(algorithm, purpose), km_device_(km_device) {} virtual ~KeymasterPassthroughOperationFactory() {} KeyType registry_key() const override { return key_type_; } // Factory methods OperationPtr CreateOperation(Key&& key, const AuthorizationSet& /*begin_params*/, keymaster_error_t* error) const override { if (!error) return nullptr; *error = KM_ERROR_OK; OperationPtr op(new (std::nothrow) KeymasterPassthroughOperation<KeymasterDeviceType>( key_type_.purpose, km_device_, std::move(key))); if (!op) { *error = KM_ERROR_MEMORY_ALLOCATION_FAILED; } return op; } // Informational methods. The returned arrays reference static memory and must not be // deallocated or modified. const keymaster_padding_t* SupportedPaddingModes(size_t* padding_count) const override { *padding_count = 0; return nullptr; } const keymaster_block_mode_t* SupportedBlockModes(size_t* block_mode_count) const override { *block_mode_count = 0; return nullptr; } const keymaster_digest_t* SupportedDigests(size_t* digest_count) const override { *digest_count = 0; return nullptr; } private: KeyType key_type_; const KeymasterDeviceType* km_device_; }; } // namespace keymaster #endif // SYSTEM_KEYMASTER_KEYMASTER_PASSTHROUGH_OPERATION_H_