/* sys-crypto-md.h - message digest (MD) wrapper * * message digest (MD) algorithms are not necessarily cryptographically secure * (often provided by crypto libraries, hence this file named sys-crypto-md.h) * * Copyright(c) 2020 Glenn Strauss gstrauss()gluelogic.com All rights reserved * License: BSD 3-clause (same as lighttpd) */ #ifndef LI_SYS_CRYPTO_MD_H #define LI_SYS_CRYPTO_MD_H #include "first.h" #include "sys-crypto.h" /* USE_LIB_CRYPTO */ #ifdef USE_LIB_CRYPTO #if defined(USE_NETTLE_CRYPTO) #include #include #include #define USE_LIB_CRYPTO_MD4 typedef struct md4_ctx MD4_CTX; static inline int MD4_Init(MD4_CTX *ctx) { nettle_md4_init(ctx); return 1; } static inline int MD4_Final(unsigned char *digest, MD4_CTX *ctx) { nettle_md4_digest(ctx, MD4_DIGEST_SIZE, digest); return 1; } static inline int MD4_Update(MD4_CTX *ctx, const void *data, size_t length) { nettle_md4_update(ctx, length, data); return 1; } #define USE_LIB_CRYPTO_MD5 typedef struct md5_ctx MD5_CTX; static inline int MD5_Init(MD5_CTX *ctx) { nettle_md5_init(ctx); return 1; } static inline int MD5_Final(unsigned char *digest, MD5_CTX *ctx) { nettle_md5_digest(ctx, MD5_DIGEST_SIZE, digest); return 1; } static inline int MD5_Update(MD5_CTX *ctx, const void *data, size_t length) { nettle_md5_update(ctx, length, data); return 1; } #define USE_LIB_CRYPTO_SHA1 typedef struct sha1_ctx SHA_CTX; static inline int SHA1_Init(SHA_CTX *ctx) { nettle_sha1_init(ctx); return 1; } static inline int SHA1_Final(unsigned char *digest, SHA_CTX *ctx) { nettle_sha1_digest(ctx, SHA1_DIGEST_SIZE, digest); return 1; } static inline int SHA1_Update(SHA_CTX *ctx, const void *data, size_t length) { nettle_sha1_update(ctx, length, data); return 1; } #define USE_LIB_CRYPTO_SHA256 typedef struct sha256_ctx SHA256_CTX; static inline int SHA256_Init(SHA256_CTX *ctx) { nettle_sha256_init(ctx); return 1; } static inline int SHA256_Final(unsigned char *digest, SHA256_CTX *ctx) { nettle_sha256_digest(ctx, SHA256_DIGEST_SIZE, digest); return 1; } static inline int SHA256_Update(SHA256_CTX *ctx, const void *data, size_t length) { nettle_sha256_update(ctx, length, data); return 1; } #define USE_LIB_CRYPTO_SHA512_256 #define SHA512_256_CTX SHA512_CTX /*(nettle/sha2.h: #define sha512_256_ctx sha512_ctx)*/ /*typedef struct sha512_256_ctx SHA512_CTX;*/ /*(yes, SHA512_CTX)*/ typedef struct sha512_ctx SHA512_CTX; static inline int SHA512_256_Init(SHA512_CTX *ctx) { nettle_sha512_256_init(ctx); return 1; } static inline int SHA512_256_Final(unsigned char *digest, SHA512_CTX *ctx) { nettle_sha512_256_digest(ctx, SHA256_DIGEST_SIZE, digest); return 1; } static inline int SHA512_256_Update(SHA512_CTX *ctx, const void *data, size_t length) { nettle_sha512_update(ctx, length, data); /*(yes, nettle_sha512_update())*/ return 1; } #define USE_LIB_CRYPTO_SHA512 /*typedef struct sha512_ctx SHA512_CTX;*//*(defined above)*/ static inline int SHA512_Init(SHA512_CTX *ctx) { nettle_sha512_init(ctx); return 1; } static inline int SHA512_Final(unsigned char *digest, SHA512_CTX *ctx) { nettle_sha512_digest(ctx, SHA512_DIGEST_SIZE, digest); return 1; } static inline int SHA512_Update(SHA512_CTX *ctx, const void *data, size_t length) { nettle_sha512_update(ctx, length, data); return 1; } #elif defined(USE_MBEDTLS_CRYPTO) #include #ifdef MBEDTLS_MD4_C #define USE_LIB_CRYPTO_MD4 #include typedef struct mbedtls_md4_context MD4_CTX; static inline int MD4_Init(MD4_CTX *ctx) { mbedtls_md4_init(ctx); return (0 == mbedtls_md4_starts_ret(ctx)); } static inline int MD4_Final(unsigned char *digest, MD4_CTX *ctx) { int rc = mbedtls_md4_finish_ret(ctx, digest); mbedtls_md4_free(ctx); return (0 == rc); } static inline int MD4_Update(MD4_CTX *ctx, const void *data, size_t length) { return (0 == mbedtls_md4_update_ret(ctx, data, length)); } #endif #ifdef MBEDTLS_MD5_C #define USE_LIB_CRYPTO_MD5 #include typedef struct mbedtls_md5_context MD5_CTX; static inline int MD5_Init(MD5_CTX *ctx) { mbedtls_md5_init(ctx); return (0 == mbedtls_md5_starts_ret(ctx)); } static inline int MD5_Final(unsigned char *digest, MD5_CTX *ctx) { int rc = mbedtls_md5_finish_ret(ctx, digest); mbedtls_md5_free(ctx); return (0 == rc); } static inline int MD5_Update(MD5_CTX *ctx, const void *data, size_t length) { return (0 == mbedtls_md5_update_ret(ctx, data, length)); } #endif #ifdef MBEDTLS_SHA1_C #define USE_LIB_CRYPTO_SHA1 #include typedef struct mbedtls_sha1_context SHA_CTX; static inline int SHA1_Init(SHA_CTX *ctx) { mbedtls_sha1_init(ctx); return (0 == mbedtls_sha1_starts_ret(ctx)); } static inline int SHA1_Final(unsigned char *digest, SHA_CTX *ctx) { int rc = mbedtls_sha1_finish_ret(ctx, digest); mbedtls_sha1_free(ctx); return (0 == rc); } static inline int SHA1_Update(SHA_CTX *ctx, const void *data, size_t length) { return (0 == mbedtls_sha1_update_ret(ctx, data, length)); } #endif #ifdef MBEDTLS_SHA256_C #define USE_LIB_CRYPTO_SHA256 #include typedef struct mbedtls_sha256_context SHA256_CTX; static inline int SHA256_Init(SHA256_CTX *ctx) { mbedtls_sha256_init(ctx); return (0 == mbedtls_sha256_starts_ret(ctx, 0)); } static inline int SHA256_Final(unsigned char *digest, SHA256_CTX *ctx) { int rc = mbedtls_sha256_finish_ret(ctx, digest); mbedtls_sha256_free(ctx); return (0 == rc); } static inline int SHA256_Update(SHA256_CTX *ctx, const void *data, size_t length) { return (0 == mbedtls_sha256_update_ret(ctx, data, length)); } #endif #ifdef MBEDTLS_SHA512_C #define USE_LIB_CRYPTO_SHA512 #include typedef struct mbedtls_sha512_context SHA512_CTX; static inline int SHA512_Init(SHA512_CTX *ctx) { mbedtls_sha512_init(ctx); return (0 == mbedtls_sha512_starts_ret(ctx, 0)); } static inline int SHA512_Final(unsigned char *digest, SHA512_CTX *ctx) { int rc = mbedtls_sha512_finish_ret(ctx, digest); mbedtls_sha512_free(ctx); return (0 == rc); } static inline int SHA512_Update(SHA512_CTX *ctx, const void *data, size_t length) { return (0 == mbedtls_sha512_update_ret(ctx, data, length)); } #endif #elif defined(USE_WOLFSSL_CRYPTO) /* WolfSSL compatibility API for OpenSSL unnecessarily bounces through an extra * layer of indirection. However, to avoid conflicting typedefs when includers * also include headers from the WolfSSL compatibility API for OpenSSL, we * include those headers here, as well, and use the compatibility API typedefs. * (undef of OPENSSL_EXTRA and NO_OLD_WC_NAMES not sufficient, and not friendly * to do in a header when others might rely on them) */ /* workaround fragile code in wolfssl/wolfcrypto/types.h */ #if !defined(SIZEOF_LONG) || !defined(SIZEOF_LONG_LONG) #undef SIZEOF_LONG #undef SIZEOF_LONG_LONG #endif #include /* wolfssl NO_* macros */ #ifndef NO_MD4 #include #include #undef MD4_Init #undef MD4_Final #undef MD4_Update #define USE_LIB_CRYPTO_MD4 /*typedef Md4 MD4_CTX;*/ static inline int MD4_Init(MD4_CTX *ctx) { wc_InitMd4((Md4 *)ctx); return 1; } static inline int MD4_Final(unsigned char *digest, MD4_CTX *ctx) { wc_Md4Final((Md4 *)ctx, digest); return 1; } static inline int MD4_Update(MD4_CTX *ctx, const void *data, size_t length) { wc_Md4Update((Md4 *)ctx, data, length); return 1; } #endif #ifndef NO_MD5 #include #include #undef MD5_Init #undef MD5_Final #undef MD5_Update #define USE_LIB_CRYPTO_MD5 /*typedef wc_Md5 MD5_CTX;*/ static inline int MD5_Init(MD5_CTX *ctx) { return (0 == wc_InitMd5((wc_Md5 *)ctx)); } static inline int MD5_Final(unsigned char *digest, MD5_CTX *ctx) { return (0 == wc_Md5Final((wc_Md5 *)ctx, digest)); } static inline int MD5_Update(MD5_CTX *ctx, const void *data, size_t length) { wc_Md5Update((wc_Md5 *)ctx, data, length); return 1; } #endif #ifndef NO_SHA #include #include #undef SHA1_Init #undef SHA1_Final #undef SHA1_Update #define USE_LIB_CRYPTO_SHA1 /*typedef wc_Sha SHA_CTX;*/ static inline int SHA1_Init(SHA_CTX *ctx) { return (0 == wc_InitSha((wc_Sha *)ctx)); } static inline int SHA1_Final(unsigned char *digest, SHA_CTX *ctx) { return (0 == wc_ShaFinal((wc_Sha *)ctx, digest)); } static inline int SHA1_Update(SHA_CTX *ctx, const void *data, size_t length) { wc_ShaUpdate((wc_Sha *)ctx, data, length); return 1; } #endif #ifndef NO_SHA256 #include #include #undef SHA256_Init #undef SHA256_Final #undef SHA256_Update #define USE_LIB_CRYPTO_SHA256 /*typedef wc_Sha256 SHA256_CTX;*/ static inline int SHA256_Init(SHA256_CTX *ctx) { return (0 == wc_InitSha256((wc_Sha256 *)ctx)); } static inline int SHA256_Final(unsigned char *digest, SHA256_CTX *ctx) { return (0 == wc_Sha256Final((wc_Sha256 *)ctx, digest)); } static inline int SHA256_Update(SHA256_CTX *ctx, const void *data, size_t length) { wc_Sha256Update((wc_Sha256 *)ctx, data, length); return 1; } #endif #ifndef NO_SHA512 #ifdef WOLFSSL_SHA512 #include #include #undef SHA512_Init #undef SHA512_Final #undef SHA512_Update #define USE_LIB_CRYPTO_SHA512 /*typedef wc_Sha512 SHA512_CTX;*/ static inline int SHA512_Init(SHA512_CTX *ctx) { return (0 == wc_InitSha512((wc_Sha512 *)ctx)); } static inline int SHA512_Final(unsigned char *digest, SHA512_CTX *ctx) { return (0 == wc_Sha512Final((wc_Sha512 *)ctx, digest)); } static inline int SHA512_Update(SHA512_CTX *ctx, const void *data, size_t length) { wc_Sha512Update((wc_Sha512 *)ctx, data, length); return 1; } #endif #endif #elif defined(USE_OPENSSL_CRYPTO) #include #include #include #ifndef OPENSSL_NO_MD4 #define USE_LIB_CRYPTO_MD4 #endif #ifndef OPENSSL_NO_MD5 #define USE_LIB_CRYPTO_MD5 #endif #define USE_LIB_CRYPTO_SHA1 #define USE_LIB_CRYPTO_SHA256 #ifdef SHA512_256_DIGEST_LENGTH #define USE_LIB_CRYPTO_SHA512_256 #endif #ifdef SHA512_DIGEST_LENGTH #define USE_LIB_CRYPTO_SHA512 #endif #include #if OPENSSL_VERSION_NUMBER >= 0x30000000L #include #ifdef USE_LIB_CRYPTO_MD4 #define MD4_CTX EVP_MD4_CTX #define MD4_Init EVP_MD4_Init #define MD4_Final EVP_MD4_Final #define MD4_Update EVP_MD4_Update typedef EVP_MD_CTX * MD4_CTX; static inline int EVP_MD4_Init(EVP_MD4_CTX *ctx) { return ((*ctx = EVP_MD_CTX_new()) != NULL && 1 == EVP_DigestInit_ex(*ctx, EVP_md4(), NULL)); } static inline int EVP_MD4_Final(unsigned char *digest, EVP_MD4_CTX *ctx) { /* MD4_DIGEST_LENGTH; EVP_MD_size(EVP_md4()) */ int rc = EVP_DigestFinal_ex(*ctx, digest, NULL); EVP_MD_CTX_free(*ctx); return (1 == rc); } static inline int EVP_MD4_Update(EVP_MD4_CTX *ctx, const void *data, size_t length) { return (1 == EVP_DigestUpdate(*ctx, data, length)); } #endif #ifdef USE_LIB_CRYPTO_MD5 #define MD5_CTX EVP_MD5_CTX #define MD5_Init EVP_MD5_Init #define MD5_Final EVP_MD5_Final #define MD5_Update EVP_MD5_Update typedef EVP_MD_CTX * EVP_MD5_CTX; static inline int EVP_MD5_Init(EVP_MD5_CTX *ctx) { return ((*ctx = EVP_MD_CTX_new()) != NULL && 1 == EVP_DigestInit_ex(*ctx, EVP_md5(), NULL)); } static inline int EVP_MD5_Final(unsigned char *digest, EVP_MD5_CTX *ctx) { /* MD5_DIGEST_LENGTH; EVP_MD_size(EVP_md5()) */ int rc = EVP_DigestFinal_ex(*ctx, digest, NULL); EVP_MD_CTX_free(*ctx); return (1 == rc); } static inline int EVP_MD5_Update(EVP_MD5_CTX *ctx, const void *data, size_t length) { return (1 == EVP_DigestUpdate(*ctx, data, length)); } #endif #ifdef USE_LIB_CRYPTO_SHA1 #define SHA_CTX EVP_SHA1_CTX #define SHA1_Init EVP_SHA1_Init #define SHA1_Final EVP_SHA1_Final #define SHA1_Update EVP_SHA1_Update typedef EVP_MD_CTX * EVP_SHA1_CTX; static inline int EVP_SHA1_Init(EVP_SHA1_CTX *ctx) { return ((*ctx = EVP_MD_CTX_new()) != NULL && 1 == EVP_DigestInit_ex(*ctx, EVP_sha1(), NULL)); } static inline int EVP_SHA1_Final(unsigned char *digest, EVP_SHA1_CTX *ctx) { /* SHA_DIGEST_LENGTH; EVP_MD_size(EVP_sha1()) */ int rc = EVP_DigestFinal_ex(*ctx, digest, NULL); EVP_MD_CTX_free(*ctx); return (1 == rc); } static inline int EVP_SHA1_Update(EVP_SHA1_CTX *ctx, const void *data, size_t length) { return (1 == EVP_DigestUpdate(*ctx, data, length)); } #endif #ifdef USE_LIB_CRYPTO_SHA256 #define SHA256_CTX EVP_SHA256_CTX #define SHA256_Init EVP_SHA256_Init #define SHA256_Final EVP_SHA256_Final #define SHA256_Update EVP_SHA256_Update typedef EVP_MD_CTX * EVP_SHA256_CTX; static inline int EVP_SHA256_Init(EVP_SHA256_CTX *ctx) { return ((*ctx = EVP_MD_CTX_new()) != NULL && 1 == EVP_DigestInit_ex(*ctx, EVP_sha256(), NULL)); } static inline int EVP_SHA256_Final(unsigned char *digest, EVP_SHA256_CTX *ctx) { /* SHA256_DIGEST_LENGTH; EVP_MD_size(EVP_sha256()) */ int rc = EVP_DigestFinal_ex(*ctx, digest, NULL); EVP_MD_CTX_free(*ctx); return (1 == rc); } static inline int EVP_SHA256_Update(EVP_SHA256_CTX *ctx, const void *data, size_t length) { return (1 == EVP_DigestUpdate(*ctx, data, length)); } #endif #ifdef USE_LIB_CRYPTO_SHA512_256 #define SHA512_256_CTX EVP_SHA512_256_CTX #define SHA512_256_Init EVP_SHA512_256_Init #define SHA512_256_Final EVP_SHA512_256_Final #define SHA512_256_Update EVP_SHA512_256_Update typedef EVP_MD_CTX * EVP_SHA512_256_CTX; static inline int EVP_SHA512_256_Init(EVP_SHA512_256_CTX *ctx) { return ((*ctx = EVP_MD_CTX_new()) != NULL && 1 == EVP_DigestInit_ex(*ctx, EVP_sha512_256(), NULL)); } static inline int EVP_SHA512_256_Final(unsigned char *digest, EVP_SHA512_256_CTX *ctx) { /* SHA256_DIGEST_LENGTH; EVP_MD_size(EVP_sha512_256()) */ int rc = EVP_DigestFinal_ex(*ctx, digest, NULL); EVP_MD_CTX_free(*ctx); return (1 == rc); } static inline int EVP_SHA512_256_Update(EVP_SHA512_256_CTX *ctx, const void *data, size_t length) { return (1 == EVP_DigestUpdate(*ctx, data, length)); } #endif #ifdef USE_LIB_CRYPTO_SHA512 #define SHA512_CTX EVP_SHA512_CTX #define SHA512_Init EVP_SHA512_Init #define SHA512_Final EVP_SHA512_Final #define SHA512_Update EVP_SHA512_Update typedef EVP_MD_CTX * EVP_SHA512_CTX; static inline int EVP_SHA512_Init(EVP_SHA512_CTX *ctx) { return ((*ctx = EVP_MD_CTX_new()) != NULL && 1 == EVP_DigestInit_ex(*ctx, EVP_sha512(), NULL)); } static inline int EVP_SHA512_Final(unsigned char *digest, EVP_SHA512_CTX *ctx) { /* SHA512_DIGEST_LENGTH; EVP_MD_size(EVP_sha512()) */ int rc = EVP_DigestFinal_ex(*ctx, digest, NULL); EVP_MD_CTX_free(*ctx); return (1 == rc); } static inline int EVP_SHA512_Update(EVP_SHA512_CTX *ctx, const void *data, size_t length) { return (1 == EVP_DigestUpdate(*ctx, data, length)); } #endif #endif /* OPENSSL_VERSION_NUMBER >= 0x30000000L */ #elif defined(USE_GNUTLS_CRYPTO) #include #include "ck.h" #define USE_LIB_CRYPTO_MD5 typedef gnutls_hash_hd_t MD5_CTX; static inline int MD5_Init(MD5_CTX *ctx) { if (gnutls_hash_init(ctx, GNUTLS_DIG_MD5) < 0) ck_bt_abort(__FILE__, __LINE__, "aborted"); return 1; } static inline int MD5_Final(unsigned char *digest, MD5_CTX *ctx) { gnutls_hash_deinit(*ctx, digest); return 1; } static inline int MD5_Update(MD5_CTX *ctx, const void *data, size_t length) { gnutls_hash(*ctx, data, length); return 1; } #define USE_LIB_CRYPTO_SHA1 typedef gnutls_hash_hd_t SHA_CTX; static inline int SHA1_Init(SHA_CTX *ctx) { if (gnutls_hash_init(ctx, GNUTLS_DIG_SHA1) < 0) ck_bt_abort(__FILE__, __LINE__, "aborted"); return 1; } static inline int SHA1_Final(unsigned char *digest, SHA_CTX *ctx) { gnutls_hash_deinit(*ctx, digest); return 1; } static inline int SHA1_Update(SHA_CTX *ctx, const void *data, size_t length) { gnutls_hash(*ctx, data, length); return 1; } #define USE_LIB_CRYPTO_SHA256 typedef gnutls_hash_hd_t SHA256_CTX; static inline int SHA256_Init(SHA256_CTX *ctx) { if (gnutls_hash_init(ctx, GNUTLS_DIG_SHA256) < 0) ck_bt_abort(__FILE__, __LINE__, "aborted"); return 1; } static inline int SHA256_Final(unsigned char *digest, SHA256_CTX *ctx) { gnutls_hash_deinit(*ctx, digest); return 1; } static inline int SHA256_Update(SHA256_CTX *ctx, const void *data, size_t length) { gnutls_hash(*ctx, data, length); return 1; } #define USE_LIB_CRYPTO_SHA512 typedef gnutls_hash_hd_t SHA512_CTX; static inline int SHA512_Init(SHA512_CTX *ctx) { if (gnutls_hash_init(ctx, GNUTLS_DIG_SHA512) < 0) ck_bt_abort(__FILE__, __LINE__, "aborted"); return 1; } static inline int SHA512_Final(unsigned char *digest, SHA512_CTX *ctx) { gnutls_hash_deinit(*ctx, digest); return 1; } static inline int SHA512_Update(SHA512_CTX *ctx, const void *data, size_t length) { gnutls_hash(*ctx, data, length); return 1; } #elif defined(USE_NSS_CRYPTO) #ifdef __has_include #if __has_include() #define NSS_VER_INCLUDE #endif #endif /* basic algorithms fail if NSS library has not been init'd (WTH). * lighttpd defers initialization of rand and crypto until first use * to attempt to avoid long, blocking init at startup while waiting * for sufficient system entropy to become available */ #ifdef NSS_VER_INCLUDE #include /* NSS_IsInitialized() NSS_NoDB_Init() */ #else #include /* NSS_IsInitialized() NSS_NoDB_Init() */ #endif #include /* abort() */ __attribute_cold__ static inline void nss_requires_explicit_init_for_basic_crypto_wth(void) { if (NSS_NoDB_Init(NULL) < 0) abort(); } #ifdef NSS_VER_INCLUDE #include #else #include #endif #define NSS_gen_hashfuncs(name, typ) \ static inline int \ name##_Init(void **ctx) \ { \ if (!NSS_IsInitialized()) \ nss_requires_explicit_init_for_basic_crypto_wth(); \ const SECHashObject * const hashObj = HASH_GetHashObject(typ); \ return ((*ctx=hashObj->create()) != NULL) ? (hashObj->begin(*ctx),1) : 0; \ } \ static inline int \ name##_Final(unsigned char *dest, void **ctx) \ { \ const SECHashObject * const hashObj = HASH_GetHashObject(typ); \ unsigned int retLen; \ hashObj->end(*ctx, dest, &retLen, hashObj->length); \ hashObj->destroy(*ctx, PR_TRUE); \ return 1; \ } \ static inline int \ name##_Update(void **ctx, const void *src, size_t len) \ { \ const SECHashObject * const hashObj = HASH_GetHashObject(typ); \ hashObj->update(*ctx, src, (int)len); \ return 1; \ } \ typedef void * name##_CTX typedef void * SHA_CTX; #define USE_LIB_CRYPTO_MD5 /* MD5_Init() * MD5_Update() * MD5_Final() */ NSS_gen_hashfuncs(MD5, HASH_AlgMD5); #define USE_LIB_CRYPTO_SHA1 /* SHA1_Init() * SHA1_Update() * SHA1_Final() */ NSS_gen_hashfuncs(SHA1, HASH_AlgSHA1); #define USE_LIB_CRYPTO_SHA256 /* SHA256_Init() * SHA256_Update() * SHA256_Final() */ NSS_gen_hashfuncs(SHA256, HASH_AlgSHA256); #define USE_LIB_CRYPTO_SHA512 /* SHA512_Init() * SHA512_Update() * SHA512_Final() */ NSS_gen_hashfuncs(SHA512, HASH_AlgSHA512); #endif #endif /* USE_LIB_CRYPTO */ #ifdef USE_LIB_CRYPTO_MD4 #ifndef MD4_DIGEST_LENGTH #define MD4_DIGEST_LENGTH 16 #endif #undef MD_DIGEST_LENGTH_MAX #define MD_DIGEST_LENGTH_MAX MD4_DIGEST_LENGTH #endif #ifdef USE_LIB_CRYPTO_MD5 #ifndef MD5_DIGEST_LENGTH #define MD5_DIGEST_LENGTH 16 #endif #include "algo_md5.h" /*(for legacy li_MD5_*() name mangling)*/ #else #include "algo_md5.h" /* MD5 implementation included with lighttpd */ #endif #undef MD_DIGEST_LENGTH_MAX #define MD_DIGEST_LENGTH_MAX MD5_DIGEST_LENGTH #ifdef USE_LIB_CRYPTO_SHA1 typedef SHA_CTX SHA1_CTX; /*(naming consistency with other algos)*/ #ifndef SHA_DIGEST_LENGTH #define SHA_DIGEST_LENGTH 20 #endif #ifndef SHA1_DIGEST_LENGTH /*(naming consistency with other algos)*/ #define SHA1_DIGEST_LENGTH SHA_DIGEST_LENGTH #endif #else #include "algo_sha1.h" /* SHA1 implementation included with lighttpd */ typedef SHA_CTX SHA1_CTX; /*(naming consistency with other algos)*/ #endif #undef MD_DIGEST_LENGTH_MAX #define MD_DIGEST_LENGTH_MAX SHA_DIGEST_LENGTH #ifdef USE_LIB_CRYPTO_SHA256 #ifndef SHA256_DIGEST_LENGTH #define SHA256_DIGEST_LENGTH 32 #endif #undef MD_DIGEST_LENGTH_MAX #define MD_DIGEST_LENGTH_MAX SHA256_DIGEST_LENGTH #endif #ifdef USE_LIB_CRYPTO_SHA512_256 #ifndef SHA512_256_DIGEST_LENGTH #define SHA512_256_DIGEST_LENGTH 32 #endif #undef MD_DIGEST_LENGTH_MAX #define MD_DIGEST_LENGTH_MAX SHA512_256_DIGEST_LENGTH #endif #ifdef USE_LIB_CRYPTO_SHA512 #ifndef SHA512_DIGEST_LENGTH #define SHA512_DIGEST_LENGTH 64 #endif #undef MD_DIGEST_LENGTH_MAX #define MD_DIGEST_LENGTH_MAX SHA512_DIGEST_LENGTH #endif /* message digest wrappers operating on single ptr, and on const_iovec */ typedef void(*li_md_once_fn)(unsigned char *digest, const void *data, size_t n); #define li_md_once(algo) \ static inline void \ algo##_once (unsigned char * const digest, \ const void * const data, const size_t n) \ { \ algo##_CTX ctx; \ algo##_Init(&ctx); \ algo##_Update(&ctx, data, n); \ algo##_Final(digest, &ctx); \ } #ifndef LI_CONST_IOVEC #define LI_CONST_IOVEC struct const_iovec { const void *iov_base; size_t iov_len; }; #endif typedef void(*li_md_iov_fn)(unsigned char *digest, const struct const_iovec *iov, size_t n); #define li_md_iov(algo) \ static inline void \ algo##_iov (unsigned char * const digest, \ const struct const_iovec * const iov, const size_t n) \ { \ algo##_CTX ctx; \ algo##_Init(&ctx); \ for (size_t i = 0; i < n; ++i) { \ if (iov[i].iov_len) \ algo##_Update(&ctx, iov[i].iov_base, iov[i].iov_len); \ } \ algo##_Final(digest, &ctx); \ } #ifdef USE_LIB_CRYPTO_MD4 li_md_once(MD4) li_md_iov(MD4) #endif /* MD4_once() MD4_iov() */ /*#ifdef USE_LIB_CRYPTO_MD5*/ li_md_once(MD5) li_md_iov(MD5) /*#endif*/ /* MD5_once() MD5_iov() */ /*#ifdef USE_LIB_CRYPTO_SHA1*/ li_md_once(SHA1) li_md_iov(SHA1) /*#endif*/ /* SHA1_once() SHA1_iov() */ #ifdef USE_LIB_CRYPTO_SHA256 li_md_once(SHA256) li_md_iov(SHA256) #endif /* SHA256_once() SHA256_iov() */ #ifdef USE_LIB_CRYPTO_SHA512_256 li_md_once(SHA512_256) li_md_iov(SHA512_256) #endif /* SHA512_256_once() SHA512_256_iov() */ #ifdef USE_LIB_CRYPTO_SHA512 li_md_once(SHA512) li_md_iov(SHA512) #endif /* SHA512_once() SHA512_iov() */ #endif /* LI_SYS_CRYPTO_MD_H */