| // Copyright 2015 The Weave Authors. All rights reserved. |
| // Use of this source code is governed by a BSD-style license that can be |
| // found in the LICENSE file. |
| |
| #include "src/crypto_hmac.h" |
| |
| #include <stddef.h> |
| #include <stdint.h> |
| #include <string.h> |
| |
| #include <openssl/evp.h> |
| #include <openssl/hmac.h> |
| |
| size_t uw_crypto_hmac_required_buffer_size_() { |
| return sizeof(HMAC_CTX); |
| } |
| |
| bool uw_crypto_hmac_init_(uint8_t* state_buffer, |
| size_t state_buffer_len, |
| const uint8_t* key, |
| size_t key_len) { |
| if (sizeof(HMAC_CTX) > state_buffer_len) { |
| return false; |
| } |
| HMAC_CTX* context = (HMAC_CTX*)state_buffer; |
| HMAC_CTX_init(context); |
| return HMAC_Init(context, key, key_len, EVP_sha256()); |
| } |
| |
| bool uw_crypto_hmac_update_(uint8_t* state_buffer, |
| size_t state_buffer_len, |
| const uint8_t* data, |
| size_t data_len) { |
| if (sizeof(HMAC_CTX) > state_buffer_len) { |
| return false; |
| } |
| HMAC_CTX* context = (HMAC_CTX*)state_buffer; |
| return HMAC_Update(context, data, data_len); |
| } |
| |
| bool uw_crypto_hmac_final_(uint8_t* state_buffer, |
| size_t state_buffer_len, |
| uint8_t* truncated_digest, |
| size_t truncated_digest_len) { |
| if (sizeof(HMAC_CTX) > state_buffer_len) { |
| return false; |
| } |
| HMAC_CTX* context = (HMAC_CTX*)state_buffer; |
| |
| const size_t kFullDigestLen = (size_t)EVP_MD_size(EVP_sha256()); |
| if (truncated_digest_len > kFullDigestLen) { |
| return false; |
| } |
| |
| uint8_t digest[kFullDigestLen]; |
| uint32_t len = kFullDigestLen; |
| |
| bool result = HMAC_Final(context, digest, &len) && kFullDigestLen == len; |
| HMAC_CTX_cleanup(context); |
| if (result) { |
| memcpy(truncated_digest, digest, truncated_digest_len); |
| } |
| return result; |
| } |