LCOV - code coverage report
Current view: top level - src/crypto - aes.cpp (source / functions) Coverage Total Hit
Test: test_bitcoin_coverage.info Lines: 100.0 % 79 79
Test Date: 2026-03-16 04:49:17 Functions: 100.0 % 14 14
Branches: 79.4 % 34 27

             Branch data     Line data    Source code
       1                 :             : // Copyright (c) 2016-present The Bitcoin Core developers
       2                 :             : // Distributed under the MIT software license, see the accompanying
       3                 :             : // file COPYING or http://www.opensource.org/licenses/mit-license.php.
       4                 :             : 
       5                 :             : #include <crypto/aes.h>
       6                 :             : #include <support/allocators/secure.h>
       7                 :             : 
       8                 :             : #include <cstring>
       9                 :             : 
      10                 :             : extern "C" {
      11                 :             : #include <crypto/ctaes/ctaes.c>
      12                 :             : }
      13                 :             : 
      14                 :        3346 : AES256Encrypt::AES256Encrypt(const unsigned char key[32])
      15                 :             : {
      16                 :        3346 :     ctx = allocator.allocate(1);
      17                 :        3346 :     AES256_init(ctx, key);
      18                 :        3346 : }
      19                 :             : 
      20                 :        3346 : AES256Encrypt::~AES256Encrypt()
      21                 :             : {
      22                 :        3346 :     allocator.deallocate(ctx, 1);
      23                 :        3346 : }
      24                 :             : 
      25                 :        5442 : void AES256Encrypt::Encrypt(unsigned char ciphertext[16], const unsigned char plaintext[16]) const
      26                 :             : {
      27                 :        5442 :     AES256_encrypt(ctx, 1, ciphertext, plaintext);
      28                 :        5442 : }
      29                 :             : 
      30                 :        3452 : AES256Decrypt::AES256Decrypt(const unsigned char key[32])
      31                 :             : {
      32                 :        3452 :     ctx = allocator.allocate(1);
      33                 :        3452 :     AES256_init(ctx, key);
      34                 :        3452 : }
      35                 :             : 
      36                 :        3452 : AES256Decrypt::~AES256Decrypt()
      37                 :             : {
      38                 :        3452 :     allocator.deallocate(ctx, 1);
      39                 :        3452 : }
      40                 :             : 
      41                 :        5654 : void AES256Decrypt::Decrypt(unsigned char plaintext[16], const unsigned char ciphertext[16]) const
      42                 :             : {
      43                 :        5654 :     AES256_decrypt(ctx, 1, plaintext, ciphertext);
      44                 :        5654 : }
      45                 :             : 
      46                 :             : 
      47                 :             : template <typename T>
      48                 :        3469 : static int CBCEncrypt(const T& enc, const unsigned char iv[AES_BLOCKSIZE], const unsigned char* data, int size, bool pad, unsigned char* out)
      49                 :             : {
      50                 :        3469 :     int written = 0;
      51                 :        3469 :     int padsize = size % AES_BLOCKSIZE;
      52                 :             :     unsigned char mixed[AES_BLOCKSIZE];
      53                 :             : 
      54   [ +  -  +  - ]:        3469 :     if (!data || !size || !out)
      55                 :             :         return 0;
      56                 :             : 
      57         [ +  + ]:        3469 :     if (!pad && padsize != 0)
      58                 :             :         return 0;
      59                 :             : 
      60                 :        3409 :     memcpy(mixed, iv, AES_BLOCKSIZE);
      61                 :             : 
      62                 :             :     // Write all but the last block
      63         [ +  + ]:        5445 :     while (written + AES_BLOCKSIZE <= size) {
      64         [ +  + ]:       34612 :         for (int i = 0; i != AES_BLOCKSIZE; i++)
      65                 :       32576 :             mixed[i] ^= *data++;
      66                 :        2036 :         enc.Encrypt(out + written, mixed);
      67                 :        2036 :         memcpy(mixed, out + written, AES_BLOCKSIZE);
      68                 :        2036 :         written += AES_BLOCKSIZE;
      69                 :             :     }
      70         [ +  + ]:        3409 :     if (pad) {
      71                 :             :         // For all that remains, pad each byte with the value of the remaining
      72                 :             :         // space. If there is none, pad by a full block.
      73         [ +  + ]:       28121 :         for (int i = 0; i != padsize; i++)
      74                 :       24720 :             mixed[i] ^= *data++;
      75         [ +  + ]:       33097 :         for (int i = padsize; i != AES_BLOCKSIZE; i++)
      76                 :       29696 :             mixed[i] ^= AES_BLOCKSIZE - padsize;
      77                 :        3401 :         enc.Encrypt(out + written, mixed);
      78                 :        3401 :         written += AES_BLOCKSIZE;
      79                 :             :     }
      80                 :             :     return written;
      81                 :             : }
      82                 :             : 
      83                 :             : template <typename T>
      84                 :        3515 : static int CBCDecrypt(const T& dec, const unsigned char iv[AES_BLOCKSIZE], const unsigned char* data, int size, bool pad, unsigned char* out)
      85                 :             : {
      86                 :        3515 :     int written = 0;
      87                 :        3515 :     bool fail = false;
      88                 :        3515 :     const unsigned char* prev = iv;
      89                 :             : 
      90   [ +  -  +  - ]:        3515 :     if (!data || !size || !out)
      91                 :             :         return 0;
      92                 :             : 
      93         [ +  - ]:        3515 :     if (size % AES_BLOCKSIZE != 0)
      94                 :             :         return 0;
      95                 :             : 
      96                 :             :     // Decrypt all data. Padding will be checked in the output.
      97         [ +  + ]:        9164 :     while (written != size) {
      98                 :        5649 :         dec.Decrypt(out, data + written);
      99         [ +  + ]:       96033 :         for (int i = 0; i != AES_BLOCKSIZE; i++)
     100                 :       90384 :             *out++ ^= prev[i];
     101                 :        5649 :         prev = data + written;
     102                 :        5649 :         written += AES_BLOCKSIZE;
     103                 :             :     }
     104                 :             : 
     105                 :             :     // When decrypting padding, attempt to run in constant-time
     106         [ +  + ]:        3515 :     if (pad) {
     107                 :             :         // If used, padding size is the value of the last decrypted byte. For
     108                 :             :         // it to be valid, It must be between 1 and AES_BLOCKSIZE.
     109                 :        3507 :         unsigned char padsize = *--out;
     110                 :        3507 :         fail = !padsize | (padsize > AES_BLOCKSIZE);
     111                 :             : 
     112                 :             :         // If not well-formed, treat it as though there's no padding.
     113                 :        3507 :         padsize *= !fail;
     114                 :             : 
     115                 :             :         // All padding must equal the last byte otherwise it's not well-formed
     116         [ +  + ]:       59619 :         for (int i = AES_BLOCKSIZE; i != 0; i--)
     117                 :       56112 :             fail |= ((i > AES_BLOCKSIZE - padsize) & (*out-- != padsize));
     118                 :             : 
     119                 :        3507 :         written -= padsize;
     120                 :             :     }
     121                 :        3515 :     return written * !fail;
     122                 :             : }
     123                 :             : 
     124                 :        3341 : AES256CBCEncrypt::AES256CBCEncrypt(const unsigned char key[AES256_KEYSIZE], const unsigned char ivIn[AES_BLOCKSIZE], bool padIn)
     125                 :        3341 :     : enc(key), pad(padIn)
     126                 :             : {
     127         [ +  - ]:        3341 :     iv = allocator.allocate(AES_BLOCKSIZE);
     128                 :        3341 :     memcpy(iv, ivIn, AES_BLOCKSIZE);
     129                 :        3341 : }
     130                 :             : 
     131                 :        3469 : int AES256CBCEncrypt::Encrypt(const unsigned char* data, int size, unsigned char* out) const
     132                 :             : {
     133                 :        3469 :     return CBCEncrypt(enc, iv, data, size, pad, out);
     134                 :             : }
     135                 :             : 
     136                 :        6682 : AES256CBCEncrypt::~AES256CBCEncrypt()
     137                 :             : {
     138                 :        3341 :     allocator.deallocate(iv, AES_BLOCKSIZE);
     139                 :        3341 : }
     140                 :             : 
     141                 :        3447 : AES256CBCDecrypt::AES256CBCDecrypt(const unsigned char key[AES256_KEYSIZE], const unsigned char ivIn[AES_BLOCKSIZE], bool padIn)
     142                 :        3447 :     : dec(key), pad(padIn)
     143                 :             : {
     144         [ +  - ]:        3447 :     iv = allocator.allocate(AES_BLOCKSIZE);
     145                 :        3447 :     memcpy(iv, ivIn, AES_BLOCKSIZE);
     146                 :        3447 : }
     147                 :             : 
     148                 :             : 
     149                 :        3515 : int AES256CBCDecrypt::Decrypt(const unsigned char* data, int size, unsigned char* out) const
     150                 :             : {
     151                 :        3515 :     return CBCDecrypt(dec, iv, data, size, pad, out);
     152                 :             : }
     153                 :             : 
     154                 :        6894 : AES256CBCDecrypt::~AES256CBCDecrypt()
     155                 :             : {
     156                 :        3447 :     allocator.deallocate(iv, AES_BLOCKSIZE);
     157                 :        3447 : }
        

Generated by: LCOV version 2.0-1