aboutsummaryrefslogtreecommitdiffstats
path: root/src/lib/prov/openssl/openssl_hash.cpp
blob: fa31d36c8a69d0d80aaa03f5bf5ba1c12f669a88 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
/*
* OpenSSL Hash Functions
* (C) 1999-2007,2015 Jack Lloyd
*
* Botan is released under the Simplified BSD License (see license.txt)
*/

#include <botan/hash.h>
#include <botan/internal/openssl.h>
#include <botan/internal/algo_registry.h>
#include <openssl/evp.h>

namespace Botan {

namespace {

class OpenSSL_HashFunction : public HashFunction
   {
   public:
      void clear() override
         {
         const EVP_MD* algo = EVP_MD_CTX_md(&m_md);
         EVP_DigestInit_ex(&m_md, algo, nullptr);
         }

      const char* provider() const override { return "openssl"; }
      std::string name() const override { return m_name; }

      HashFunction* clone() const override
         {
         const EVP_MD* algo = EVP_MD_CTX_md(&m_md);
         return new OpenSSL_HashFunction(algo, name());
         }

      size_t output_length() const override
         {
         return EVP_MD_size(EVP_MD_CTX_md(&m_md));
         }

      size_t hash_block_size() const override
         {
         return EVP_MD_block_size(EVP_MD_CTX_md(&m_md));
         }

      OpenSSL_HashFunction(const EVP_MD* md, const std::string& name) : m_name(name)
         {
         EVP_MD_CTX_init(&m_md);
         EVP_DigestInit_ex(&m_md, md, nullptr);
         }

      ~OpenSSL_HashFunction()
         {
         EVP_MD_CTX_cleanup(&m_md);
         }

   private:
      void add_data(const byte input[], size_t length) override
         {
         EVP_DigestUpdate(&m_md, input, length);
         }

      void final_result(byte output[]) override
         {
         EVP_DigestFinal_ex(&m_md, output, nullptr);
         const EVP_MD* algo = EVP_MD_CTX_md(&m_md);
         EVP_DigestInit_ex(&m_md, algo, nullptr);
         }

      std::string m_name;
      EVP_MD_CTX m_md;
   };

std::function<HashFunction* (const HashFunction::Spec&)>
make_evp_hash_maker(const EVP_MD* md, const char* algo)
   {
   return [md,algo](const HashFunction::Spec&)
      {
      return new OpenSSL_HashFunction(md, algo);
      };
   }

#define BOTAN_REGISTER_OPENSSL_EVP_HASH(NAME, EVP)                      \
   BOTAN_REGISTER_TYPE(HashFunction, OpenSSL_HashFunction ## EVP, NAME, \
                       make_evp_hash_maker(EVP(), NAME), "openssl", BOTAN_OPENSSL_HASH_PRIO)

#if !defined(OPENSSL_NO_SHA)
   BOTAN_REGISTER_OPENSSL_EVP_HASH("SHA-160", EVP_sha1);
#endif

#if !defined(OPENSSL_NO_SHA256)
   BOTAN_REGISTER_OPENSSL_EVP_HASH("SHA-224", EVP_sha224);
   BOTAN_REGISTER_OPENSSL_EVP_HASH("SHA-256", EVP_sha256);
#endif

#if !defined(OPENSSL_NO_SHA512)
   BOTAN_REGISTER_OPENSSL_EVP_HASH("SHA-384", EVP_sha384);
   BOTAN_REGISTER_OPENSSL_EVP_HASH("SHA-512", EVP_sha512);
#endif

#if !defined(OPENSSL_NO_MD4)
   BOTAN_REGISTER_OPENSSL_EVP_HASH("MD4", EVP_md4);
#endif

#if !defined(OPENSSL_NO_MD5)
   BOTAN_REGISTER_OPENSSL_EVP_HASH("MD5", EVP_md5);
#endif

#if !defined(OPENSSL_NO_RIPEMD)
   BOTAN_REGISTER_OPENSSL_EVP_HASH("RIPEMD-160", EVP_ripemd160);
#endif

}

}