Coverage Report

Created: 2026-06-08 06:48

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