1*d9f75844SAndroid Build Coastguard Worker /*
2*d9f75844SAndroid Build Coastguard Worker * Copyright 2004 The WebRTC Project Authors. All rights reserved.
3*d9f75844SAndroid Build Coastguard Worker *
4*d9f75844SAndroid Build Coastguard Worker * Use of this source code is governed by a BSD-style license
5*d9f75844SAndroid Build Coastguard Worker * that can be found in the LICENSE file in the root of the source
6*d9f75844SAndroid Build Coastguard Worker * tree. An additional intellectual property rights grant can be found
7*d9f75844SAndroid Build Coastguard Worker * in the file PATENTS. All contributing project authors may
8*d9f75844SAndroid Build Coastguard Worker * be found in the AUTHORS file in the root of the source tree.
9*d9f75844SAndroid Build Coastguard Worker */
10*d9f75844SAndroid Build Coastguard Worker
11*d9f75844SAndroid Build Coastguard Worker #include "rtc_base/openssl_digest.h"
12*d9f75844SAndroid Build Coastguard Worker
13*d9f75844SAndroid Build Coastguard Worker #include "absl/strings/string_view.h"
14*d9f75844SAndroid Build Coastguard Worker #include "rtc_base/checks.h" // RTC_DCHECK, RTC_CHECK
15*d9f75844SAndroid Build Coastguard Worker #include "rtc_base/openssl.h"
16*d9f75844SAndroid Build Coastguard Worker
17*d9f75844SAndroid Build Coastguard Worker namespace rtc {
18*d9f75844SAndroid Build Coastguard Worker
OpenSSLDigest(absl::string_view algorithm)19*d9f75844SAndroid Build Coastguard Worker OpenSSLDigest::OpenSSLDigest(absl::string_view algorithm) {
20*d9f75844SAndroid Build Coastguard Worker ctx_ = EVP_MD_CTX_new();
21*d9f75844SAndroid Build Coastguard Worker RTC_CHECK(ctx_ != nullptr);
22*d9f75844SAndroid Build Coastguard Worker EVP_MD_CTX_init(ctx_);
23*d9f75844SAndroid Build Coastguard Worker if (GetDigestEVP(algorithm, &md_)) {
24*d9f75844SAndroid Build Coastguard Worker EVP_DigestInit_ex(ctx_, md_, nullptr);
25*d9f75844SAndroid Build Coastguard Worker } else {
26*d9f75844SAndroid Build Coastguard Worker md_ = nullptr;
27*d9f75844SAndroid Build Coastguard Worker }
28*d9f75844SAndroid Build Coastguard Worker }
29*d9f75844SAndroid Build Coastguard Worker
~OpenSSLDigest()30*d9f75844SAndroid Build Coastguard Worker OpenSSLDigest::~OpenSSLDigest() {
31*d9f75844SAndroid Build Coastguard Worker EVP_MD_CTX_destroy(ctx_);
32*d9f75844SAndroid Build Coastguard Worker }
33*d9f75844SAndroid Build Coastguard Worker
Size() const34*d9f75844SAndroid Build Coastguard Worker size_t OpenSSLDigest::Size() const {
35*d9f75844SAndroid Build Coastguard Worker if (!md_) {
36*d9f75844SAndroid Build Coastguard Worker return 0;
37*d9f75844SAndroid Build Coastguard Worker }
38*d9f75844SAndroid Build Coastguard Worker return EVP_MD_size(md_);
39*d9f75844SAndroid Build Coastguard Worker }
40*d9f75844SAndroid Build Coastguard Worker
Update(const void * buf,size_t len)41*d9f75844SAndroid Build Coastguard Worker void OpenSSLDigest::Update(const void* buf, size_t len) {
42*d9f75844SAndroid Build Coastguard Worker if (!md_) {
43*d9f75844SAndroid Build Coastguard Worker return;
44*d9f75844SAndroid Build Coastguard Worker }
45*d9f75844SAndroid Build Coastguard Worker EVP_DigestUpdate(ctx_, buf, len);
46*d9f75844SAndroid Build Coastguard Worker }
47*d9f75844SAndroid Build Coastguard Worker
Finish(void * buf,size_t len)48*d9f75844SAndroid Build Coastguard Worker size_t OpenSSLDigest::Finish(void* buf, size_t len) {
49*d9f75844SAndroid Build Coastguard Worker if (!md_ || len < Size()) {
50*d9f75844SAndroid Build Coastguard Worker return 0;
51*d9f75844SAndroid Build Coastguard Worker }
52*d9f75844SAndroid Build Coastguard Worker unsigned int md_len;
53*d9f75844SAndroid Build Coastguard Worker EVP_DigestFinal_ex(ctx_, static_cast<unsigned char*>(buf), &md_len);
54*d9f75844SAndroid Build Coastguard Worker EVP_DigestInit_ex(ctx_, md_, nullptr); // prepare for future Update()s
55*d9f75844SAndroid Build Coastguard Worker RTC_DCHECK(md_len == Size());
56*d9f75844SAndroid Build Coastguard Worker return md_len;
57*d9f75844SAndroid Build Coastguard Worker }
58*d9f75844SAndroid Build Coastguard Worker
GetDigestEVP(absl::string_view algorithm,const EVP_MD ** mdp)59*d9f75844SAndroid Build Coastguard Worker bool OpenSSLDigest::GetDigestEVP(absl::string_view algorithm,
60*d9f75844SAndroid Build Coastguard Worker const EVP_MD** mdp) {
61*d9f75844SAndroid Build Coastguard Worker const EVP_MD* md;
62*d9f75844SAndroid Build Coastguard Worker if (algorithm == DIGEST_MD5) {
63*d9f75844SAndroid Build Coastguard Worker md = EVP_md5();
64*d9f75844SAndroid Build Coastguard Worker } else if (algorithm == DIGEST_SHA_1) {
65*d9f75844SAndroid Build Coastguard Worker md = EVP_sha1();
66*d9f75844SAndroid Build Coastguard Worker } else if (algorithm == DIGEST_SHA_224) {
67*d9f75844SAndroid Build Coastguard Worker md = EVP_sha224();
68*d9f75844SAndroid Build Coastguard Worker } else if (algorithm == DIGEST_SHA_256) {
69*d9f75844SAndroid Build Coastguard Worker md = EVP_sha256();
70*d9f75844SAndroid Build Coastguard Worker } else if (algorithm == DIGEST_SHA_384) {
71*d9f75844SAndroid Build Coastguard Worker md = EVP_sha384();
72*d9f75844SAndroid Build Coastguard Worker } else if (algorithm == DIGEST_SHA_512) {
73*d9f75844SAndroid Build Coastguard Worker md = EVP_sha512();
74*d9f75844SAndroid Build Coastguard Worker } else {
75*d9f75844SAndroid Build Coastguard Worker return false;
76*d9f75844SAndroid Build Coastguard Worker }
77*d9f75844SAndroid Build Coastguard Worker
78*d9f75844SAndroid Build Coastguard Worker // Can't happen
79*d9f75844SAndroid Build Coastguard Worker RTC_DCHECK(EVP_MD_size(md) >= 16);
80*d9f75844SAndroid Build Coastguard Worker *mdp = md;
81*d9f75844SAndroid Build Coastguard Worker return true;
82*d9f75844SAndroid Build Coastguard Worker }
83*d9f75844SAndroid Build Coastguard Worker
GetDigestName(const EVP_MD * md,std::string * algorithm)84*d9f75844SAndroid Build Coastguard Worker bool OpenSSLDigest::GetDigestName(const EVP_MD* md, std::string* algorithm) {
85*d9f75844SAndroid Build Coastguard Worker RTC_DCHECK(md != nullptr);
86*d9f75844SAndroid Build Coastguard Worker RTC_DCHECK(algorithm != nullptr);
87*d9f75844SAndroid Build Coastguard Worker
88*d9f75844SAndroid Build Coastguard Worker int md_type = EVP_MD_type(md);
89*d9f75844SAndroid Build Coastguard Worker if (md_type == NID_md5) {
90*d9f75844SAndroid Build Coastguard Worker *algorithm = DIGEST_MD5;
91*d9f75844SAndroid Build Coastguard Worker } else if (md_type == NID_sha1) {
92*d9f75844SAndroid Build Coastguard Worker *algorithm = DIGEST_SHA_1;
93*d9f75844SAndroid Build Coastguard Worker } else if (md_type == NID_sha224) {
94*d9f75844SAndroid Build Coastguard Worker *algorithm = DIGEST_SHA_224;
95*d9f75844SAndroid Build Coastguard Worker } else if (md_type == NID_sha256) {
96*d9f75844SAndroid Build Coastguard Worker *algorithm = DIGEST_SHA_256;
97*d9f75844SAndroid Build Coastguard Worker } else if (md_type == NID_sha384) {
98*d9f75844SAndroid Build Coastguard Worker *algorithm = DIGEST_SHA_384;
99*d9f75844SAndroid Build Coastguard Worker } else if (md_type == NID_sha512) {
100*d9f75844SAndroid Build Coastguard Worker *algorithm = DIGEST_SHA_512;
101*d9f75844SAndroid Build Coastguard Worker } else {
102*d9f75844SAndroid Build Coastguard Worker algorithm->clear();
103*d9f75844SAndroid Build Coastguard Worker return false;
104*d9f75844SAndroid Build Coastguard Worker }
105*d9f75844SAndroid Build Coastguard Worker
106*d9f75844SAndroid Build Coastguard Worker return true;
107*d9f75844SAndroid Build Coastguard Worker }
108*d9f75844SAndroid Build Coastguard Worker
GetDigestSize(absl::string_view algorithm,size_t * length)109*d9f75844SAndroid Build Coastguard Worker bool OpenSSLDigest::GetDigestSize(absl::string_view algorithm, size_t* length) {
110*d9f75844SAndroid Build Coastguard Worker const EVP_MD* md;
111*d9f75844SAndroid Build Coastguard Worker if (!GetDigestEVP(algorithm, &md))
112*d9f75844SAndroid Build Coastguard Worker return false;
113*d9f75844SAndroid Build Coastguard Worker
114*d9f75844SAndroid Build Coastguard Worker *length = EVP_MD_size(md);
115*d9f75844SAndroid Build Coastguard Worker return true;
116*d9f75844SAndroid Build Coastguard Worker }
117*d9f75844SAndroid Build Coastguard Worker
118*d9f75844SAndroid Build Coastguard Worker } // namespace rtc
119