Coverage Report

Created: 2026-05-16 06:52

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
18.3M
static ENCRYPT(DesSmallFipsEncrypt,TEMPSMALL, LOADFIPS,KEYMAPSMALL,SAVEFIPS)
63
43.8M
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
25.4k
{
124
  /* Hash function generated using gperf. */
125
25.4k
  static const unsigned char asso_values[0x81] =
126
25.4k
    {
127
25.4k
      16,  9, 26, 26, 26, 26, 26, 26, 26, 26,
128
25.4k
      26, 26, 26, 26, 26,  6,  2, 26, 26, 26,
129
25.4k
      26, 26, 26, 26, 26, 26, 26, 26, 26, 26,
130
25.4k
      26, 26, 26, 26, 26, 26, 26, 26, 26, 26,
131
25.4k
      26, 26, 26, 26, 26, 26, 26, 26, 26, 26,
132
25.4k
      26, 26, 26, 26, 26, 26, 26, 26, 26, 26,
133
25.4k
      26, 26, 26, 26, 26, 26, 26, 26, 26, 26,
134
25.4k
      26, 26, 26, 26, 26, 26, 26, 26, 26, 26,
135
25.4k
      26, 26, 26, 26, 26, 26, 26, 26, 26, 26,
136
25.4k
      26, 26, 26, 26, 26, 26, 26, 26, 26, 26,
137
25.4k
      26, 26, 26, 26, 26, 26, 26, 26, 26, 26,
138
25.4k
      26, 26,  3,  1, 26, 26, 26, 26, 26, 26,
139
25.4k
      26, 26, 26, 26, 26, 26, 26,  0,  0
140
25.4k
    };
141
142
25.4k
  static const int8_t weak_key_hash[26][4] =
143
25.4k
    {
144
25.4k
      /*  0 */ {0x7f,0x7f, 0x7f,0x7f},
145
25.4k
      /*  1 */ {0x7f,0x70, 0x7f,0x78},
146
25.4k
      /*  2 */ {0x7f,0x0f, 0x7f,0x07},
147
25.4k
      /*  3 */ {0x70,0x7f, 0x78,0x7f},
148
25.4k
      /*  4 */ {0x70,0x70, 0x78,0x78},
149
25.4k
      /*  5 */ {0x70,0x0f, 0x78,0x07},
150
25.4k
      /*  6 */ {0x0f,0x7f, 0x07,0x7f},
151
25.4k
      /*  7 */ {0x0f,0x70, 0x07,0x78},
152
25.4k
      /*  8 */ {0x0f,0x0f, 0x07,0x07},
153
25.4k
      /*  9 */ {0x7f,0x00, 0x7f,0x00},
154
25.4k
      /* 10 */ {-1,-1,-1,-1},
155
25.4k
      /* 11 */ {-1,-1,-1,-1},
156
25.4k
      /* 12 */ {0x70,0x00, 0x78,0x00},
157
25.4k
      /* 13 */ {-1,-1,-1,-1},
158
25.4k
      /* 14 */ {-1,-1,-1,-1},
159
25.4k
      /* 15 */ {0x0f,0x00, 0x07,0x00},
160
25.4k
      /* 16 */ {0x00,0x7f, 0x00,0x7f},
161
25.4k
      /* 17 */ {0x00,0x70, 0x00,0x78},
162
25.4k
      /* 18 */ {0x00,0x0f, 0x00,0x07},
163
25.4k
      /* 19 */ {-1,-1,-1,-1},
164
25.4k
      /* 20 */ {-1,-1,-1,-1},
165
25.4k
      /* 21 */ {-1,-1,-1,-1},
166
25.4k
      /* 22 */ {-1,-1,-1,-1},
167
25.4k
      /* 23 */ {-1,-1,-1,-1},
168
25.4k
      /* 24 */ {-1,-1,-1,-1},
169
25.4k
      /* 25 */ {0x00,0x00, 0x00,0x00}
170
25.4k
    };
171
172
25.4k
  int8_t k0 = key[0] >> 1;
173
25.4k
  int8_t k1 = key[1] >> 1;
174
175
25.4k
  unsigned hash = asso_values[k1 + 1] + asso_values[k0];
176
25.4k
  const int8_t *candidate;
177
178
25.4k
  if (hash > 25)
179
25.3k
    return 0;
180
181
88
  candidate = weak_key_hash[hash];
182
183
88
  if (k0 != candidate[0]
184
56
      || k1 != candidate[1])
185
40
    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
25.4k
{
206
25.4k
  register uint32_t n, w;
207
25.4k
  register char * b0, * b1;
208
25.4k
  char bits0[56], bits1[56];
209
25.4k
  uint32_t *method;
210
25.4k
  const uint8_t *k;
211
212
  /* explode the bits */
213
25.4k
  n = 56;
214
25.4k
  b0 = bits0;
215
25.4k
  b1 = bits1;
216
25.4k
  k = key;
217
203k
  do {
218
203k
    w = (256 | *k++) << 2;
219
1.42M
    do {
220
1.42M
      --n;
221
1.42M
      b1[n] = 8 & w;
222
1.42M
      w >>= 1;
223
1.42M
      b0[n] = 4 & w;
224
1.42M
    } while ( w >= 16 );
225
203k
  } while ( n );
226
227
  /* put the bits in the correct places */
228
25.4k
  n = 16;
229
25.4k
  k = rotors;
230
25.4k
  method = ctx->key;
231
  
232
407k
  do {
233
407k
    w   = (b1[k[ 0   ]] | b0[k[ 1   ]]) << 4;
234
407k
    w  |= (b1[k[ 2   ]] | b0[k[ 3   ]]) << 2;
235
407k
    w  |=  b1[k[ 4   ]] | b0[k[ 5   ]];
236
407k
    w <<= 8;
237
407k
    w  |= (b1[k[ 6   ]] | b0[k[ 7   ]]) << 4;
238
407k
    w  |= (b1[k[ 8   ]] | b0[k[ 9   ]]) << 2;
239
407k
    w  |=  b1[k[10   ]] | b0[k[11   ]];
240
407k
    w <<= 8;
241
407k
    w  |= (b1[k[12   ]] | b0[k[13   ]]) << 4;
242
407k
    w  |= (b1[k[14   ]] | b0[k[15   ]]) << 2;
243
407k
    w  |=  b1[k[16   ]] | b0[k[17   ]];
244
407k
    w <<= 8;
245
407k
    w  |= (b1[k[18   ]] | b0[k[19   ]]) << 4;
246
407k
    w  |= (b1[k[20   ]] | b0[k[21   ]]) << 2;
247
407k
    w  |=  b1[k[22   ]] | b0[k[23   ]];
248
249
407k
    method[0] = w;
250
251
407k
    w   = (b1[k[ 0+24]] | b0[k[ 1+24]]) << 4;
252
407k
    w  |= (b1[k[ 2+24]] | b0[k[ 3+24]]) << 2;
253
407k
    w  |=  b1[k[ 4+24]] | b0[k[ 5+24]];
254
407k
    w <<= 8;
255
407k
    w  |= (b1[k[ 6+24]] | b0[k[ 7+24]]) << 4;
256
407k
    w  |= (b1[k[ 8+24]] | b0[k[ 9+24]]) << 2;
257
407k
    w  |=  b1[k[10+24]] | b0[k[11+24]];
258
407k
    w <<= 8;
259
407k
    w  |= (b1[k[12+24]] | b0[k[13+24]]) << 4;
260
407k
    w  |= (b1[k[14+24]] | b0[k[15+24]]) << 2;
261
407k
    w  |=  b1[k[16+24]] | b0[k[17+24]];
262
407k
    w <<= 8;
263
407k
    w  |= (b1[k[18+24]] | b0[k[19+24]]) << 4;
264
407k
    w  |= (b1[k[20+24]] | b0[k[21+24]]) << 2;
265
407k
    w  |=  b1[k[22+24]] | b0[k[23+24]];
266
267
407k
    ROR(w, 4, 28);   /* could be eliminated */
268
407k
    method[1] = w;
269
270
407k
    k += 48;
271
407k
    method  += 2;
272
407k
  } while ( --n );
273
274
25.4k
  return !des_weak_p (key);
275
25.4k
}
276
277
void
278
des_encrypt(const struct des_ctx *ctx,
279
      size_t length, uint8_t *dst,
280
      const uint8_t *src)
281
11.3k
{
282
11.3k
  assert(!(length % DES_BLOCK_SIZE));
283
  
284
586k
  while (length)
285
574k
    {
286
574k
      DesSmallFipsEncrypt(dst, ctx->key, src);
287
574k
      length -= DES_BLOCK_SIZE;
288
574k
      src += DES_BLOCK_SIZE;
289
574k
      dst += DES_BLOCK_SIZE;
290
574k
    }
291
11.3k
}
292
293
void
294
des_decrypt(const struct des_ctx *ctx,
295
      size_t length, uint8_t *dst,
296
      const uint8_t *src)
297
26.3k
{
298
26.3k
  assert(!(length % DES_BLOCK_SIZE));
299
300
1.39M
  while (length)
301
1.36M
    {
302
1.36M
      DesSmallFipsDecrypt(dst, ctx->key, src);
303
1.36M
      length -= DES_BLOCK_SIZE;
304
1.36M
      src += DES_BLOCK_SIZE;
305
1.36M
      dst += DES_BLOCK_SIZE;
306
1.36M
    }
307
26.3k
}