xref: /aosp_15_r20/external/webrtc/rtc_base/openssl_digest.cc (revision d9f758449e529ab9291ac668be2861e7a55c2422)
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