Coverage Report

Created: 2023-09-25 06:33

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