Coverage Report

Created: 2026-03-31 06:37

next uncovered line (L), next uncovered region (R), next uncovered branch (B)
/src/nettle/des.c
Line
Count
Source
1
/* des.c
2
3
   The des block cipher.
4
5
   Copyright (C) 2001, 2010 Niels Möller
6
   Copyright (C) 1992  Dana L. How
7
8
   This file is part of GNU Nettle.
9
10
   GNU Nettle is free software: you can redistribute it and/or
11
   modify it under the terms of either:
12
13
     * the GNU Lesser General Public License as published by the Free
14
       Software Foundation; either version 3 of the License, or (at your
15
       option) any later version.
16
17
   or
18
19
     * the GNU General Public License as published by the Free
20
       Software Foundation; either version 2 of the License, or (at your
21
       option) any later version.
22
23
   or both in parallel, as here.
24
25
   GNU Nettle is distributed in the hope that it will be useful,
26
   but WITHOUT ANY WARRANTY; without even the implied warranty of
27
   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
28
   General Public License for more details.
29
30
   You should have received copies of the GNU General Public License and
31
   the GNU Lesser General Public License along with this program.  If
32
   not, see http://www.gnu.org/licenses/.
33
*/
34
35
/*  des - fast & portable DES encryption & decryption.
36
 *  Copyright (C) 1992  Dana L. How
37
 *  Please see the file `descore.README' for the complete copyright notice.
38
 */
39
40
#if HAVE_CONFIG_H
41
# include "config.h"
42
#endif
43
44
#include <assert.h>
45
46
#include "des.h"
47
48
#include "desCode.h"
49
50
/* various tables */
51
52
static const uint32_t
53
des_keymap[] = {
54
#include  "keymap.h"
55
};
56
57
static const uint8_t
58
rotors[] = {
59
#include  "rotors.h"
60
};
61
62
19.9M
static ENCRYPT(DesSmallFipsEncrypt,TEMPSMALL, LOADFIPS,KEYMAPSMALL,SAVEFIPS)
63
47.3M
static DECRYPT(DesSmallFipsDecrypt,TEMPSMALL, LOADFIPS,KEYMAPSMALL,SAVEFIPS)
64
65
/* If parity bits are used, keys should have odd parity. We use a
66
   small table, to not waste any memory on this fairly obscure DES
67
   feature. */
68
69
static const unsigned
70
parity_16[16] =
71
{ 0, 1, 1, 0, 1, 0, 0, 1, 1, 0, 0, 1, 0, 1, 1, 0 };
72
73
0
#define PARITY(x) (parity_16[(x)&0xf] ^ parity_16[((x)>>4) & 0xf])
74
75
int
76
des_check_parity(size_t length, const uint8_t *key)
77
0
{
78
0
  size_t i;
79
0
  for (i = 0; i<length; i++)
80
0
    if (!PARITY(key[i]))
81
0
      return 0;
82
83
0
  return 1;
84
0
}
85
86
void
87
des_fix_parity(size_t length, uint8_t *dst,
88
         const uint8_t *src)
89
0
{
90
0
  size_t i;
91
0
  for (i = 0; i<length; i++)
92
0
    dst[i] = src[i] ^ PARITY(src[i]) ^ 1;
93
0
}
94
95
/* Weak and semiweak keys, excluding parity:
96
 *
97
 * 00 00 00 00  00 00 00 00
98
 * 7f 7f 7f 7f  7f 7f 7f 7f 
99
 * 0f 0f 0f 0f  07 07 07 07
100
 * 70 70 70 70  78 78 78 78
101
 *
102
 * 00 7f 00 7f  00 7f 00 7f
103
 * 7f 00 7f 00  7f 00 7f 00
104
 *
105
 * 0f 70 0f 70  07 78 07 78
106
 * 70 0f 70 0f  78 07 78 07
107
 *
108
 * 00 70 00 70  00 78 00 78
109
 * 70 00 70 00  78 00 78 00
110
 *
111
 * 0f 7f 0f 7f  07 7f 07 7f
112
 * 7f 0f 7f 0f  7f 07 7f 07
113
 *
114
 * 00 0f 00 0f  00 07 00 07
115
 * 0f 00 0f 00  07 00 07 00
116
 *
117
 * 70 7f 70 7f  78 7f 78 7f
118
 * 7f 70 7f 70  7f 78 7f 78
119
 */
120
121
static int
122
des_weak_p(const uint8_t *key)
123
27.3k
{
124
  /* Hash function generated using gperf. */
125
27.3k
  static const unsigned char asso_values[0x81] =
126
27.3k
    {
127
27.3k
      16,  9, 26, 26, 26, 26, 26, 26, 26, 26,
128
27.3k
      26, 26, 26, 26, 26,  6,  2, 26, 26, 26,
129
27.3k
      26, 26, 26, 26, 26, 26, 26, 26, 26, 26,
130
27.3k
      26, 26, 26, 26, 26, 26, 26, 26, 26, 26,
131
27.3k
      26, 26, 26, 26, 26, 26, 26, 26, 26, 26,
132
27.3k
      26, 26, 26, 26, 26, 26, 26, 26, 26, 26,
133
27.3k
      26, 26, 26, 26, 26, 26, 26, 26, 26, 26,
134
27.3k
      26, 26, 26, 26, 26, 26, 26, 26, 26, 26,
135
27.3k
      26, 26, 26, 26, 26, 26, 26, 26, 26, 26,
136
27.3k
      26, 26, 26, 26, 26, 26, 26, 26, 26, 26,
137
27.3k
      26, 26, 26, 26, 26, 26, 26, 26, 26, 26,
138
27.3k
      26, 26,  3,  1, 26, 26, 26, 26, 26, 26,
139
27.3k
      26, 26, 26, 26, 26, 26, 26,  0,  0
140
27.3k
    };
141
142
27.3k
  static const int8_t weak_key_hash[26][4] =
143
27.3k
    {
144
27.3k
      /*  0 */ {0x7f,0x7f, 0x7f,0x7f},
145
27.3k
      /*  1 */ {0x7f,0x70, 0x7f,0x78},
146
27.3k
      /*  2 */ {0x7f,0x0f, 0x7f,0x07},
147
27.3k
      /*  3 */ {0x70,0x7f, 0x78,0x7f},
148
27.3k
      /*  4 */ {0x70,0x70, 0x78,0x78},
149
27.3k
      /*  5 */ {0x70,0x0f, 0x78,0x07},
150
27.3k
      /*  6 */ {0x0f,0x7f, 0x07,0x7f},
151
27.3k
      /*  7 */ {0x0f,0x70, 0x07,0x78},
152
27.3k
      /*  8 */ {0x0f,0x0f, 0x07,0x07},
153
27.3k
      /*  9 */ {0x7f,0x00, 0x7f,0x00},
154
27.3k
      /* 10 */ {-1,-1,-1,-1},
155
27.3k
      /* 11 */ {-1,-1,-1,-1},
156
27.3k
      /* 12 */ {0x70,0x00, 0x78,0x00},
157
27.3k
      /* 13 */ {-1,-1,-1,-1},
158
27.3k
      /* 14 */ {-1,-1,-1,-1},
159
27.3k
      /* 15 */ {0x0f,0x00, 0x07,0x00},
160
27.3k
      /* 16 */ {0x00,0x7f, 0x00,0x7f},
161
27.3k
      /* 17 */ {0x00,0x70, 0x00,0x78},
162
27.3k
      /* 18 */ {0x00,0x0f, 0x00,0x07},
163
27.3k
      /* 19 */ {-1,-1,-1,-1},
164
27.3k
      /* 20 */ {-1,-1,-1,-1},
165
27.3k
      /* 21 */ {-1,-1,-1,-1},
166
27.3k
      /* 22 */ {-1,-1,-1,-1},
167
27.3k
      /* 23 */ {-1,-1,-1,-1},
168
27.3k
      /* 24 */ {-1,-1,-1,-1},
169
27.3k
      /* 25 */ {0x00,0x00, 0x00,0x00}
170
27.3k
    };
171
172
27.3k
  int8_t k0 = key[0] >> 1;
173
27.3k
  int8_t k1 = key[1] >> 1;
174
175
27.3k
  unsigned hash = asso_values[k1 + 1] + asso_values[k0];
176
27.3k
  const int8_t *candidate;
177
178
27.3k
  if (hash > 25)
179
27.3k
    return 0;
180
181
98
  candidate = weak_key_hash[hash];
182
183
98
  if (k0 != candidate[0]
184
62
      || k1 != candidate[1])
185
50
    return 0;
186
  
187
48
  if ( (key[2] >> 1) != k0
188
16
       || (key[3] >> 1) != k1)
189
46
    return 0;
190
191
2
  k0 = key[4] >> 1;
192
2
  k1 = key[5] >> 1;
193
2
  if (k0 != candidate[2]
194
0
      || k1 != candidate[3])
195
2
    return 0;
196
0
  if ( (key[6] >> 1) != k0
197
0
       || (key[7] >> 1) != k1)
198
0
    return 0;
199
200
0
  return 1;
201
0
}
202
203
int
204
des_set_key(struct des_ctx *ctx, const uint8_t *key)
205
27.3k
{
206
27.3k
  register uint32_t n, w;
207
27.3k
  register char * b0, * b1;
208
27.3k
  char bits0[56], bits1[56];
209
27.3k
  uint32_t *method;
210
27.3k
  const uint8_t *k;
211
212
  /* explode the bits */
213
27.3k
  n = 56;
214
27.3k
  b0 = bits0;
215
27.3k
  b1 = bits1;
216
27.3k
  k = key;
217
219k
  do {
218
219k
    w = (256 | *k++) << 2;
219
1.53M
    do {
220
1.53M
      --n;
221
1.53M
      b1[n] = 8 & w;
222
1.53M
      w >>= 1;
223
1.53M
      b0[n] = 4 & w;
224
1.53M
    } while ( w >= 16 );
225
219k
  } while ( n );
226
227
  /* put the bits in the correct places */
228
27.3k
  n = 16;
229
27.3k
  k = rotors;
230
27.3k
  method = ctx->key;
231
  
232
438k
  do {
233
438k
    w   = (b1[k[ 0   ]] | b0[k[ 1   ]]) << 4;
234
438k
    w  |= (b1[k[ 2   ]] | b0[k[ 3   ]]) << 2;
235
438k
    w  |=  b1[k[ 4   ]] | b0[k[ 5   ]];
236
438k
    w <<= 8;
237
438k
    w  |= (b1[k[ 6   ]] | b0[k[ 7   ]]) << 4;
238
438k
    w  |= (b1[k[ 8   ]] | b0[k[ 9   ]]) << 2;
239
438k
    w  |=  b1[k[10   ]] | b0[k[11   ]];
240
438k
    w <<= 8;
241
438k
    w  |= (b1[k[12   ]] | b0[k[13   ]]) << 4;
242
438k
    w  |= (b1[k[14   ]] | b0[k[15   ]]) << 2;
243
438k
    w  |=  b1[k[16   ]] | b0[k[17   ]];
244
438k
    w <<= 8;
245
438k
    w  |= (b1[k[18   ]] | b0[k[19   ]]) << 4;
246
438k
    w  |= (b1[k[20   ]] | b0[k[21   ]]) << 2;
247
438k
    w  |=  b1[k[22   ]] | b0[k[23   ]];
248
249
438k
    method[0] = w;
250
251
438k
    w   = (b1[k[ 0+24]] | b0[k[ 1+24]]) << 4;
252
438k
    w  |= (b1[k[ 2+24]] | b0[k[ 3+24]]) << 2;
253
438k
    w  |=  b1[k[ 4+24]] | b0[k[ 5+24]];
254
438k
    w <<= 8;
255
438k
    w  |= (b1[k[ 6+24]] | b0[k[ 7+24]]) << 4;
256
438k
    w  |= (b1[k[ 8+24]] | b0[k[ 9+24]]) << 2;
257
438k
    w  |=  b1[k[10+24]] | b0[k[11+24]];
258
438k
    w <<= 8;
259
438k
    w  |= (b1[k[12+24]] | b0[k[13+24]]) << 4;
260
438k
    w  |= (b1[k[14+24]] | b0[k[15+24]]) << 2;
261
438k
    w  |=  b1[k[16+24]] | b0[k[17+24]];
262
438k
    w <<= 8;
263
438k
    w  |= (b1[k[18+24]] | b0[k[19+24]]) << 4;
264
438k
    w  |= (b1[k[20+24]] | b0[k[21+24]]) << 2;
265
438k
    w  |=  b1[k[22+24]] | b0[k[23+24]];
266
267
438k
    ROR(w, 4, 28);   /* could be eliminated */
268
438k
    method[1] = w;
269
270
438k
    k += 48;
271
438k
    method  += 2;
272
438k
  } while ( --n );
273
274
27.3k
  return !des_weak_p (key);
275
27.3k
}
276
277
void
278
des_encrypt(const struct des_ctx *ctx,
279
      size_t length, uint8_t *dst,
280
      const uint8_t *src)
281
12.2k
{
282
12.2k
  assert(!(length % DES_BLOCK_SIZE));
283
  
284
636k
  while (length)
285
623k
    {
286
623k
      DesSmallFipsEncrypt(dst, ctx->key, src);
287
623k
      length -= DES_BLOCK_SIZE;
288
623k
      src += DES_BLOCK_SIZE;
289
623k
      dst += DES_BLOCK_SIZE;
290
623k
    }
291
12.2k
}
292
293
void
294
des_decrypt(const struct des_ctx *ctx,
295
      size_t length, uint8_t *dst,
296
      const uint8_t *src)
297
28.4k
{
298
28.4k
  assert(!(length % DES_BLOCK_SIZE));
299
300
1.50M
  while (length)
301
1.48M
    {
302
1.48M
      DesSmallFipsDecrypt(dst, ctx->key, src);
303
1.48M
      length -= DES_BLOCK_SIZE;
304
1.48M
      src += DES_BLOCK_SIZE;
305
1.48M
      dst += DES_BLOCK_SIZE;
306
1.48M
    }
307
28.4k
}