| // 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> | 
 |  | 
 | bool uw_crypto_hmac_(const uint8_t* key, | 
 |                      size_t key_len, | 
 |                      const UwCryptoHmacMsg messages[], | 
 |                      size_t num_messages, | 
 |                      uint8_t* truncated_digest, | 
 |                      size_t truncated_digest_len) { | 
 |   HMAC_CTX context = {0}; | 
 |   HMAC_CTX_init(&context); | 
 |   if (!HMAC_Init(&context, key, key_len, EVP_sha256())) | 
 |     return false; | 
 |  | 
 |   for (size_t i = 0; i < num_messages; ++i) { | 
 |     if (messages[i].num_bytes && | 
 |         (!messages[i].bytes || | 
 |          !HMAC_Update(&context, messages[i].bytes, messages[i].num_bytes))) { | 
 |       return false; | 
 |     } | 
 |   } | 
 |  | 
 |   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; | 
 | } |