LCOV - code coverage report
Current view: top level - src/common - hmac_openssl.c (source / functions) Hit Total Coverage
Test: PostgreSQL 18devel Lines: 52 134 38.8 %
Date: 2025-01-18 04:15:08 Functions: 7 10 70.0 %
Legend: Lines: hit not hit

          Line data    Source code
       1             : /*-------------------------------------------------------------------------
       2             :  *
       3             :  * hmac_openssl.c
       4             :  *    Implementation of HMAC with OpenSSL.
       5             :  *
       6             :  * This should only be used if code is compiled with OpenSSL support.
       7             :  *
       8             :  * Portions Copyright (c) 1996-2025, PostgreSQL Global Development Group
       9             :  * Portions Copyright (c) 1994, Regents of the University of California
      10             :  *
      11             :  * IDENTIFICATION
      12             :  *    src/common/hmac_openssl.c
      13             :  *
      14             :  *-------------------------------------------------------------------------
      15             :  */
      16             : 
      17             : #ifndef FRONTEND
      18             : #include "postgres.h"
      19             : #else
      20             : #include "postgres_fe.h"
      21             : #endif
      22             : 
      23             : 
      24             : #include <openssl/err.h>
      25             : #include <openssl/hmac.h>
      26             : 
      27             : #include "common/hmac.h"
      28             : #include "common/md5.h"
      29             : #include "common/sha1.h"
      30             : #include "common/sha2.h"
      31             : #ifndef FRONTEND
      32             : #include "utils/memutils.h"
      33             : #include "utils/resowner.h"
      34             : #endif
      35             : 
      36             : /*
      37             :  * In backend, use an allocation in TopMemoryContext to count for resowner
      38             :  * cleanup handling if necessary.  In frontend, use malloc to be able to return
      39             :  * a failure status back to the caller.
      40             :  */
      41             : #ifndef FRONTEND
      42             : #define USE_RESOWNER_FOR_HMAC
      43             : #define ALLOC(size) MemoryContextAlloc(TopMemoryContext, size)
      44             : #define FREE(ptr) pfree(ptr)
      45             : #else                           /* FRONTEND */
      46             : #define ALLOC(size) malloc(size)
      47             : #define FREE(ptr) free(ptr)
      48             : #endif                          /* FRONTEND */
      49             : 
      50             : /* Set of error states */
      51             : typedef enum pg_hmac_errno
      52             : {
      53             :     PG_HMAC_ERROR_NONE = 0,
      54             :     PG_HMAC_ERROR_DEST_LEN,
      55             :     PG_HMAC_ERROR_OPENSSL,
      56             : } pg_hmac_errno;
      57             : 
      58             : /* Internal pg_hmac_ctx structure */
      59             : struct pg_hmac_ctx
      60             : {
      61             :     HMAC_CTX   *hmacctx;
      62             :     pg_cryptohash_type type;
      63             :     pg_hmac_errno error;
      64             :     const char *errreason;
      65             : 
      66             : #ifdef USE_RESOWNER_FOR_HMAC
      67             :     ResourceOwner resowner;
      68             : #endif
      69             : };
      70             : 
      71             : /* ResourceOwner callbacks to hold HMAC contexts */
      72             : #ifdef USE_RESOWNER_FOR_HMAC
      73             : static void ResOwnerReleaseHMAC(Datum res);
      74             : 
      75             : static const ResourceOwnerDesc hmac_resowner_desc =
      76             : {
      77             :     .name = "OpenSSL HMAC context",
      78             :     .release_phase = RESOURCE_RELEASE_BEFORE_LOCKS,
      79             :     .release_priority = RELEASE_PRIO_HMAC_CONTEXTS,
      80             :     .ReleaseResource = ResOwnerReleaseHMAC,
      81             :     .DebugPrint = NULL          /* the default message is fine */
      82             : };
      83             : 
      84             : /* Convenience wrappers over ResourceOwnerRemember/Forget */
      85             : static inline void
      86         594 : ResourceOwnerRememberHMAC(ResourceOwner owner, pg_hmac_ctx *ctx)
      87             : {
      88         594 :     ResourceOwnerRemember(owner, PointerGetDatum(ctx), &hmac_resowner_desc);
      89         594 : }
      90             : static inline void
      91         594 : ResourceOwnerForgetHMAC(ResourceOwner owner, pg_hmac_ctx *ctx)
      92             : {
      93         594 :     ResourceOwnerForget(owner, PointerGetDatum(ctx), &hmac_resowner_desc);
      94         594 : }
      95             : #endif
      96             : 
      97             : static const char *
      98           0 : SSLerrmessage(unsigned long ecode)
      99             : {
     100           0 :     if (ecode == 0)
     101           0 :         return NULL;
     102             : 
     103             :     /*
     104             :      * This may return NULL, but we would fall back to a default error path if
     105             :      * that were the case.
     106             :      */
     107           0 :     return ERR_reason_error_string(ecode);
     108             : }
     109             : 
     110             : /*
     111             :  * pg_hmac_create
     112             :  *
     113             :  * Allocate a hash context.  Returns NULL on failure for an OOM.  The
     114             :  * backend issues an error, without returning.
     115             :  */
     116             : pg_hmac_ctx *
     117        1042 : pg_hmac_create(pg_cryptohash_type type)
     118             : {
     119             :     pg_hmac_ctx *ctx;
     120             : 
     121        1042 :     ctx = ALLOC(sizeof(pg_hmac_ctx));
     122        1042 :     if (ctx == NULL)
     123           0 :         return NULL;
     124        1042 :     memset(ctx, 0, sizeof(pg_hmac_ctx));
     125             : 
     126        1042 :     ctx->type = type;
     127        1042 :     ctx->error = PG_HMAC_ERROR_NONE;
     128        1042 :     ctx->errreason = NULL;
     129             : 
     130             : 
     131             :     /*
     132             :      * Initialization takes care of assigning the correct type for OpenSSL.
     133             :      * Also ensure that there aren't any unconsumed errors in the queue from
     134             :      * previous runs.
     135             :      */
     136        1042 :     ERR_clear_error();
     137             : 
     138             : #ifdef USE_RESOWNER_FOR_HMAC
     139         594 :     ResourceOwnerEnlarge(CurrentResourceOwner);
     140             : #endif
     141             : 
     142        1042 :     ctx->hmacctx = HMAC_CTX_new();
     143             : 
     144        1042 :     if (ctx->hmacctx == NULL)
     145             :     {
     146           0 :         explicit_bzero(ctx, sizeof(pg_hmac_ctx));
     147           0 :         FREE(ctx);
     148             : #ifndef FRONTEND
     149           0 :         ereport(ERROR,
     150             :                 (errcode(ERRCODE_OUT_OF_MEMORY),
     151             :                  errmsg("out of memory")));
     152             : #endif
     153           0 :         return NULL;
     154             :     }
     155             : 
     156             : 
     157             : #ifdef USE_RESOWNER_FOR_HMAC
     158         594 :     ctx->resowner = CurrentResourceOwner;
     159         594 :     ResourceOwnerRememberHMAC(CurrentResourceOwner, ctx);
     160             : #endif
     161             : 
     162        1042 :     return ctx;
     163             : }
     164             : 
     165             : /*
     166             :  * pg_hmac_init
     167             :  *
     168             :  * Initialize a HMAC context.  Returns 0 on success, -1 on failure.
     169             :  */
     170             : int
     171      922648 : pg_hmac_init(pg_hmac_ctx *ctx, const uint8 *key, size_t len)
     172             : {
     173      922648 :     int         status = 0;
     174             : 
     175      922648 :     if (ctx == NULL)
     176           0 :         return -1;
     177             : 
     178      922648 :     switch (ctx->type)
     179             :     {
     180           0 :         case PG_MD5:
     181           0 :             status = HMAC_Init_ex(ctx->hmacctx, key, len, EVP_md5(), NULL);
     182           0 :             break;
     183           0 :         case PG_SHA1:
     184           0 :             status = HMAC_Init_ex(ctx->hmacctx, key, len, EVP_sha1(), NULL);
     185           0 :             break;
     186           0 :         case PG_SHA224:
     187           0 :             status = HMAC_Init_ex(ctx->hmacctx, key, len, EVP_sha224(), NULL);
     188           0 :             break;
     189      922648 :         case PG_SHA256:
     190      922648 :             status = HMAC_Init_ex(ctx->hmacctx, key, len, EVP_sha256(), NULL);
     191      922648 :             break;
     192           0 :         case PG_SHA384:
     193           0 :             status = HMAC_Init_ex(ctx->hmacctx, key, len, EVP_sha384(), NULL);
     194           0 :             break;
     195           0 :         case PG_SHA512:
     196           0 :             status = HMAC_Init_ex(ctx->hmacctx, key, len, EVP_sha512(), NULL);
     197           0 :             break;
     198             :     }
     199             : 
     200             :     /* OpenSSL internals return 1 on success, 0 on failure */
     201      922648 :     if (status <= 0)
     202             :     {
     203           0 :         ctx->errreason = SSLerrmessage(ERR_get_error());
     204           0 :         ctx->error = PG_HMAC_ERROR_OPENSSL;
     205           0 :         return -1;
     206             :     }
     207             : 
     208      922648 :     return 0;
     209             : }
     210             : 
     211             : /*
     212             :  * pg_hmac_update
     213             :  *
     214             :  * Update a HMAC context.  Returns 0 on success, -1 on failure.
     215             :  */
     216             : int
     217      924366 : pg_hmac_update(pg_hmac_ctx *ctx, const uint8 *data, size_t len)
     218             : {
     219      924366 :     int         status = 0;
     220             : 
     221      924366 :     if (ctx == NULL)
     222           0 :         return -1;
     223             : 
     224      924366 :     status = HMAC_Update(ctx->hmacctx, data, len);
     225             : 
     226             :     /* OpenSSL internals return 1 on success, 0 on failure */
     227      924366 :     if (status <= 0)
     228             :     {
     229           0 :         ctx->errreason = SSLerrmessage(ERR_get_error());
     230           0 :         ctx->error = PG_HMAC_ERROR_OPENSSL;
     231           0 :         return -1;
     232             :     }
     233      924366 :     return 0;
     234             : }
     235             : 
     236             : /*
     237             :  * pg_hmac_final
     238             :  *
     239             :  * Finalize a HMAC context.  Returns 0 on success, -1 on failure.
     240             :  */
     241             : int
     242      922648 : pg_hmac_final(pg_hmac_ctx *ctx, uint8 *dest, size_t len)
     243             : {
     244      922648 :     int         status = 0;
     245             :     uint32      outlen;
     246             : 
     247      922648 :     if (ctx == NULL)
     248           0 :         return -1;
     249             : 
     250      922648 :     switch (ctx->type)
     251             :     {
     252           0 :         case PG_MD5:
     253           0 :             if (len < MD5_DIGEST_LENGTH)
     254             :             {
     255           0 :                 ctx->error = PG_HMAC_ERROR_DEST_LEN;
     256           0 :                 return -1;
     257             :             }
     258           0 :             break;
     259           0 :         case PG_SHA1:
     260           0 :             if (len < SHA1_DIGEST_LENGTH)
     261             :             {
     262           0 :                 ctx->error = PG_HMAC_ERROR_DEST_LEN;
     263           0 :                 return -1;
     264             :             }
     265           0 :             break;
     266           0 :         case PG_SHA224:
     267           0 :             if (len < PG_SHA224_DIGEST_LENGTH)
     268             :             {
     269           0 :                 ctx->error = PG_HMAC_ERROR_DEST_LEN;
     270           0 :                 return -1;
     271             :             }
     272           0 :             break;
     273      922648 :         case PG_SHA256:
     274      922648 :             if (len < PG_SHA256_DIGEST_LENGTH)
     275             :             {
     276           0 :                 ctx->error = PG_HMAC_ERROR_DEST_LEN;
     277           0 :                 return -1;
     278             :             }
     279      922648 :             break;
     280           0 :         case PG_SHA384:
     281           0 :             if (len < PG_SHA384_DIGEST_LENGTH)
     282             :             {
     283           0 :                 ctx->error = PG_HMAC_ERROR_DEST_LEN;
     284           0 :                 return -1;
     285             :             }
     286           0 :             break;
     287           0 :         case PG_SHA512:
     288           0 :             if (len < PG_SHA512_DIGEST_LENGTH)
     289             :             {
     290           0 :                 ctx->error = PG_HMAC_ERROR_DEST_LEN;
     291           0 :                 return -1;
     292             :             }
     293           0 :             break;
     294             :     }
     295             : 
     296      922648 :     status = HMAC_Final(ctx->hmacctx, dest, &outlen);
     297             : 
     298             :     /* OpenSSL internals return 1 on success, 0 on failure */
     299      922648 :     if (status <= 0)
     300             :     {
     301           0 :         ctx->errreason = SSLerrmessage(ERR_get_error());
     302           0 :         ctx->error = PG_HMAC_ERROR_OPENSSL;
     303           0 :         return -1;
     304             :     }
     305      922648 :     return 0;
     306             : }
     307             : 
     308             : /*
     309             :  * pg_hmac_free
     310             :  *
     311             :  * Free a HMAC context.
     312             :  */
     313             : void
     314        1042 : pg_hmac_free(pg_hmac_ctx *ctx)
     315             : {
     316        1042 :     if (ctx == NULL)
     317           0 :         return;
     318             : 
     319        1042 :     HMAC_CTX_free(ctx->hmacctx);
     320             : #ifdef USE_RESOWNER_FOR_HMAC
     321         594 :     if (ctx->resowner)
     322         594 :         ResourceOwnerForgetHMAC(ctx->resowner, ctx);
     323             : #endif
     324             : 
     325        1042 :     explicit_bzero(ctx, sizeof(pg_hmac_ctx));
     326        1042 :     FREE(ctx);
     327             : }
     328             : 
     329             : /*
     330             :  * pg_hmac_error
     331             :  *
     332             :  * Returns a static string providing details about an error that happened
     333             :  * during a HMAC computation.
     334             :  */
     335             : const char *
     336           0 : pg_hmac_error(pg_hmac_ctx *ctx)
     337             : {
     338           0 :     if (ctx == NULL)
     339           0 :         return _("out of memory");
     340             : 
     341             :     /*
     342             :      * If a reason is provided, rely on it, else fallback to any error code
     343             :      * set.
     344             :      */
     345           0 :     if (ctx->errreason)
     346           0 :         return ctx->errreason;
     347             : 
     348           0 :     switch (ctx->error)
     349             :     {
     350           0 :         case PG_HMAC_ERROR_NONE:
     351           0 :             return _("success");
     352           0 :         case PG_HMAC_ERROR_DEST_LEN:
     353           0 :             return _("destination buffer too small");
     354           0 :         case PG_HMAC_ERROR_OPENSSL:
     355           0 :             return _("OpenSSL failure");
     356             :     }
     357             : 
     358             :     Assert(false);              /* cannot be reached */
     359           0 :     return _("success");
     360             : }
     361             : 
     362             : /* ResourceOwner callbacks */
     363             : 
     364             : #ifdef USE_RESOWNER_FOR_HMAC
     365             : static void
     366           0 : ResOwnerReleaseHMAC(Datum res)
     367             : {
     368           0 :     pg_hmac_ctx *ctx = (pg_hmac_ctx *) DatumGetPointer(res);
     369             : 
     370           0 :     ctx->resowner = NULL;
     371           0 :     pg_hmac_free(ctx);
     372           0 : }
     373             : #endif

Generated by: LCOV version 1.14