blob: d3dca65442caa0586ca8b6981fa45166c3a33910 [file] [log] [blame]
Vitaly Buka47fe6f82015-12-01 14:37:24 -08001// Copyright 2015 The Weave Authors. All rights reserved.
2// Use of this source code is governed by a BSD-style license that can be
3// found in the LICENSE file.
4
5#include "src/crypto_hmac.h"
6
7#include <stddef.h>
8#include <stdint.h>
9#include <string.h>
10
11#include <openssl/evp.h>
12#include <openssl/hmac.h>
13
Vitaly Bukaa821f2e2016-01-27 11:44:02 -080014bool uw_crypto_hmac_(const uint8_t* key,
15 size_t key_len,
16 const UwCryptoHmacMsg messages[],
17 size_t num_messages,
18 uint8_t* truncated_digest,
19 size_t truncated_digest_len) {
20 HMAC_CTX context = {0};
21 HMAC_CTX_init(&context);
22 if (!HMAC_Init(&context, key, key_len, EVP_sha256()))
Vitaly Buka47fe6f82015-12-01 14:37:24 -080023 return false;
Vitaly Buka47fe6f82015-12-01 14:37:24 -080024
Vitaly Bukaa821f2e2016-01-27 11:44:02 -080025 for (size_t i = 0; i < num_messages; ++i) {
26 if (messages[i].num_bytes &&
27 (!messages[i].bytes ||
28 !HMAC_Update(&context, messages[i].bytes, messages[i].num_bytes))) {
29 return false;
30 }
Vitaly Buka47fe6f82015-12-01 14:37:24 -080031 }
Vitaly Buka47fe6f82015-12-01 14:37:24 -080032
33 const size_t kFullDigestLen = (size_t)EVP_MD_size(EVP_sha256());
34 if (truncated_digest_len > kFullDigestLen) {
35 return false;
36 }
37
38 uint8_t digest[kFullDigestLen];
39 uint32_t len = kFullDigestLen;
40
Vitaly Bukaa821f2e2016-01-27 11:44:02 -080041 bool result = HMAC_Final(&context, digest, &len) && kFullDigestLen == len;
42 HMAC_CTX_cleanup(&context);
Vitaly Buka47fe6f82015-12-01 14:37:24 -080043 if (result) {
44 memcpy(truncated_digest, digest, truncated_digest_len);
45 }
46 return result;
47}