Welcome to mirror list, hosted at ThFree Co, Russian Federation.

generate-ffdhe.py « scripts - github.com/mumble-voip/mumble.git - Unnamed repository; edit this file 'description' to name the repository.
summaryrefslogtreecommitdiff
blob: 80bee073fbaadcfaa6b6f38da6e98f489fa85790 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
#!/usr/bin/env python3
#
# Copyright 2017-2022 The Mumble Developers. All rights reserved.
# Use of this source code is governed by a BSD-style license
# that can be found in the LICENSE file at the root of the
# Mumble source tree or at <https://www.mumble.info/LICENSE>.
#
# generate-ffdhe.py
# Generate PEM strings for the FFDHE parameters in RFC 7919
# For reference, see https://tools.ietf.org/rfc/rfc7919.txt
#
# The idea of this file is to implement an easy way to generate
# PEM Diffie-Hellman parameters by using the data straight from
# the RFC. This is to make it easy to verify that the data is
# correct and not tampered with. For transparency.
#
# To verify the P and G values, you can copy/paste the listings
# from the RFC, and verify (via 'diff' or 'git diff') that the
# values used are same.
#
# This file also implements a minimal set of ASN.1 DER methods
# that allow us to marshal Diffie-Hellman parameters according
# to the PKCS#3 DH standard
# (ftp://ftp.rsasecurity.com/pub/pkcs/ascii/pkcs-3.asc)
#
# The ASN.1 definition of DHparameters is as follows:
#
#  DHParameter ::= SEQUENCE {
#     prime INTEGER, -- p
#     base INTEGER, -- g
#     privateValueLength INTEGER OPTIONAL
#  }
#
# We only use 'p' and 'g' for Diffie-Hellman parameters.
#
# To encode this, we need to be able to encode the ASN.1
# types SEQUENCE and INTEGER in DER form.
#
# A short ASN.1 DER-form crash course:
#
# All objects are marshalled as follows:
#   - Identifier/tag byte
#   - Length bytes
#   - Content bytes
#
# The identifier byte is a single byte tag that describes
# the object that follows. (0x30 for SEQUENCE, 0x02 for
# INTEGER).
#
# The length bytes contain the size of the content bytes
# that follow them. Length bytes have two forms, a short
# form (<= 127) and a long form (> 127).
# We use the derLength function to marshal a Python integer
# into a series of length bytes. See the doc string of the
# derLength function for a more thorough explanation.
#
# The content bytes contain the content itself.
#
# For the SEQUENCE type (identifier 0x30), the content
# bytes are a series of other marshalled ASN.1
# DER objects. In our case, for the DHparameters
# SEQUENCE, the content bytes are the result of
# marshalling the two ASN.1 DER INTEGERs and
# concatenating the resulting bytes.
#
# For the INTEGER type (identifier 0x02), the content
# bytes are the bytes of the integer itself in
# two's complement form, stored in big-endian order.
# Due to the use of two's complement form, the INTEGER
# type can represent both signed and unsigned integers.
# If the sign-bit is set, the integer is treated as
# signed. The integers we're going to marshal in this
# context are all stored as unsigned values. This means
# we have to be careful about marshalling them.
# For example, say we want to marshal the value 0xFF. If
# we output 0xFF as the integer's content bytes, it will
# have the value -1, and not 255. This is due to the sign
# bit being set. To allow unsigned integers to be
# marshalled, the trick is to prepend a zero byte. This
# ensures that the sign bit is not set, but does not change
# the value of the integer itself. So, the correct way to
# marshal the unsigned integer 0xff (255) is to output
# 0x00 0xff.
#
# This concludes the ASN.1 DER crash course.
#
# In this program, we use hex strings to represent
# 'bignums'. The main reason is that the input
# bignums from RFC 7919 are listed in this way.
# However, using hex strings also makes it easy to
# marshal them into ASN.1 DER form, because we can
# iterate over each byte in the number, and construct
# a bytearray from them. Functions that take these
# bignums use the parameter name 'hexString'. To convert
# from a hexString to a bytearray, we use the 'dehexify'
# function. The 'dehexify' function strips all whitespace
# from the string, so it is safe to use it on the 'bignum'
# value for P that we've copied from RFC 7919.

from __future__ import (unicode_literals, print_function, division)

# Python 3 doesn't have a long type.
# Map the long type to int here, so
# we can easily be compatible with both
# interpreters.
try:
    type(long)
except NameError:
    long = int

import doctest
import os
import base64
import codecs
import argparse

# The following P values are copy/pasted from RFC 7919 at
# https://tools.ietf.org/rfc/rfc7919.txt
#
# This is intentional: it allows others to do the same, and
# verify that the values match the data from the RFC.

ffdhe2048_p = '''
    FFFFFFFF FFFFFFFF ADF85458 A2BB4A9A AFDC5620 273D3CF1
    D8B9C583 CE2D3695 A9E13641 146433FB CC939DCE 249B3EF9
    7D2FE363 630C75D8 F681B202 AEC4617A D3DF1ED5 D5FD6561
    2433F51F 5F066ED0 85636555 3DED1AF3 B557135E 7F57C935
    984F0C70 E0E68B77 E2A689DA F3EFE872 1DF158A1 36ADE735
    30ACCA4F 483A797A BC0AB182 B324FB61 D108A94B B2C8E3FB
    B96ADAB7 60D7F468 1D4F42A3 DE394DF4 AE56EDE7 6372BB19
    0B07A7C8 EE0A6D70 9E02FCE1 CDF7E2EC C03404CD 28342F61
    9172FE9C E98583FF 8E4F1232 EEF28183 C3FE3B1B 4C6FAD73
    3BB5FCBC 2EC22005 C58EF183 7D1683B2 C6F34A26 C1B2EFFA
    886B4238 61285C97 FFFFFFFF FFFFFFFF
'''

ffdhe3072_p = '''
    FFFFFFFF FFFFFFFF ADF85458 A2BB4A9A AFDC5620 273D3CF1
    D8B9C583 CE2D3695 A9E13641 146433FB CC939DCE 249B3EF9
    7D2FE363 630C75D8 F681B202 AEC4617A D3DF1ED5 D5FD6561
    2433F51F 5F066ED0 85636555 3DED1AF3 B557135E 7F57C935
    984F0C70 E0E68B77 E2A689DA F3EFE872 1DF158A1 36ADE735
    30ACCA4F 483A797A BC0AB182 B324FB61 D108A94B B2C8E3FB
    B96ADAB7 60D7F468 1D4F42A3 DE394DF4 AE56EDE7 6372BB19
    0B07A7C8 EE0A6D70 9E02FCE1 CDF7E2EC C03404CD 28342F61
    9172FE9C E98583FF 8E4F1232 EEF28183 C3FE3B1B 4C6FAD73
    3BB5FCBC 2EC22005 C58EF183 7D1683B2 C6F34A26 C1B2EFFA
    886B4238 611FCFDC DE355B3B 6519035B BC34F4DE F99C0238
    61B46FC9 D6E6C907 7AD91D26 91F7F7EE 598CB0FA C186D91C
    AEFE1309 85139270 B4130C93 BC437944 F4FD4452 E2D74DD3
    64F2E21E 71F54BFF 5CAE82AB 9C9DF69E E86D2BC5 22363A0D
    ABC52197 9B0DEADA 1DBF9A42 D5C4484E 0ABCD06B FA53DDEF
    3C1B20EE 3FD59D7C 25E41D2B 66C62E37 FFFFFFFF FFFFFFFF
'''

ffdhe4096_p = '''
    FFFFFFFF FFFFFFFF ADF85458 A2BB4A9A AFDC5620 273D3CF1
    D8B9C583 CE2D3695 A9E13641 146433FB CC939DCE 249B3EF9
    7D2FE363 630C75D8 F681B202 AEC4617A D3DF1ED5 D5FD6561
    2433F51F 5F066ED0 85636555 3DED1AF3 B557135E 7F57C935
    984F0C70 E0E68B77 E2A689DA F3EFE872 1DF158A1 36ADE735
    30ACCA4F 483A797A BC0AB182 B324FB61 D108A94B B2C8E3FB
    B96ADAB7 60D7F468 1D4F42A3 DE394DF4 AE56EDE7 6372BB19
    0B07A7C8 EE0A6D70 9E02FCE1 CDF7E2EC C03404CD 28342F61
    9172FE9C E98583FF 8E4F1232 EEF28183 C3FE3B1B 4C6FAD73
    3BB5FCBC 2EC22005 C58EF183 7D1683B2 C6F34A26 C1B2EFFA
    886B4238 611FCFDC DE355B3B 6519035B BC34F4DE F99C0238
    61B46FC9 D6E6C907 7AD91D26 91F7F7EE 598CB0FA C186D91C
    AEFE1309 85139270 B4130C93 BC437944 F4FD4452 E2D74DD3
    64F2E21E 71F54BFF 5CAE82AB 9C9DF69E E86D2BC5 22363A0D
    ABC52197 9B0DEADA 1DBF9A42 D5C4484E 0ABCD06B FA53DDEF
    3C1B20EE 3FD59D7C 25E41D2B 669E1EF1 6E6F52C3 164DF4FB
    7930E9E4 E58857B6 AC7D5F42 D69F6D18 7763CF1D 55034004
    87F55BA5 7E31CC7A 7135C886 EFB4318A ED6A1E01 2D9E6832
    A907600A 918130C4 6DC778F9 71AD0038 092999A3 33CB8B7A
    1A1DB93D 7140003C 2A4ECEA9 F98D0ACC 0A8291CD CEC97DCF
    8EC9B55A 7F88A46B 4DB5A851 F44182E1 C68A007E 5E655F6A
    FFFFFFFF FFFFFFFF
'''

ffdhe6144_p = '''
    FFFFFFFF FFFFFFFF ADF85458 A2BB4A9A AFDC5620 273D3CF1
    D8B9C583 CE2D3695 A9E13641 146433FB CC939DCE 249B3EF9
    7D2FE363 630C75D8 F681B202 AEC4617A D3DF1ED5 D5FD6561
    2433F51F 5F066ED0 85636555 3DED1AF3 B557135E 7F57C935
    984F0C70 E0E68B77 E2A689DA F3EFE872 1DF158A1 36ADE735
    30ACCA4F 483A797A BC0AB182 B324FB61 D108A94B B2C8E3FB
    B96ADAB7 60D7F468 1D4F42A3 DE394DF4 AE56EDE7 6372BB19
    0B07A7C8 EE0A6D70 9E02FCE1 CDF7E2EC C03404CD 28342F61
    9172FE9C E98583FF 8E4F1232 EEF28183 C3FE3B1B 4C6FAD73
    3BB5FCBC 2EC22005 C58EF183 7D1683B2 C6F34A26 C1B2EFFA
    886B4238 611FCFDC DE355B3B 6519035B BC34F4DE F99C0238
    61B46FC9 D6E6C907 7AD91D26 91F7F7EE 598CB0FA C186D91C
    AEFE1309 85139270 B4130C93 BC437944 F4FD4452 E2D74DD3
    64F2E21E 71F54BFF 5CAE82AB 9C9DF69E E86D2BC5 22363A0D
    ABC52197 9B0DEADA 1DBF9A42 D5C4484E 0ABCD06B FA53DDEF
    3C1B20EE 3FD59D7C 25E41D2B 669E1EF1 6E6F52C3 164DF4FB
    7930E9E4 E58857B6 AC7D5F42 D69F6D18 7763CF1D 55034004
    87F55BA5 7E31CC7A 7135C886 EFB4318A ED6A1E01 2D9E6832
    A907600A 918130C4 6DC778F9 71AD0038 092999A3 33CB8B7A
    1A1DB93D 7140003C 2A4ECEA9 F98D0ACC 0A8291CD CEC97DCF
    8EC9B55A 7F88A46B 4DB5A851 F44182E1 C68A007E 5E0DD902
    0BFD64B6 45036C7A 4E677D2C 38532A3A 23BA4442 CAF53EA6
    3BB45432 9B7624C8 917BDD64 B1C0FD4C B38E8C33 4C701C3A
    CDAD0657 FCCFEC71 9B1F5C3E 4E46041F 388147FB 4CFDB477
    A52471F7 A9A96910 B855322E DB6340D8 A00EF092 350511E3
    0ABEC1FF F9E3A26E 7FB29F8C 183023C3 587E38DA 0077D9B4
    763E4E4B 94B2BBC1 94C6651E 77CAF992 EEAAC023 2A281BF6
    B3A739C1 22611682 0AE8DB58 47A67CBE F9C9091B 462D538C
    D72B0374 6AE77F5E 62292C31 1562A846 505DC82D B854338A
    E49F5235 C95B9117 8CCF2DD5 CACEF403 EC9D1810 C6272B04
    5B3B71F9 DC6B80D6 3FDD4A8E 9ADB1E69 62A69526 D43161C1
    A41D570D 7938DAD4 A40E329C D0E40E65 FFFFFFFF FFFFFFFF
'''

ffdhe8192_p = '''
    FFFFFFFF FFFFFFFF ADF85458 A2BB4A9A AFDC5620 273D3CF1
    D8B9C583 CE2D3695 A9E13641 146433FB CC939DCE 249B3EF9
    7D2FE363 630C75D8 F681B202 AEC4617A D3DF1ED5 D5FD6561
    2433F51F 5F066ED0 85636555 3DED1AF3 B557135E 7F57C935
    984F0C70 E0E68B77 E2A689DA F3EFE872 1DF158A1 36ADE735
    30ACCA4F 483A797A BC0AB182 B324FB61 D108A94B B2C8E3FB
    B96ADAB7 60D7F468 1D4F42A3 DE394DF4 AE56EDE7 6372BB19
    0B07A7C8 EE0A6D70 9E02FCE1 CDF7E2EC C03404CD 28342F61
    9172FE9C E98583FF 8E4F1232 EEF28183 C3FE3B1B 4C6FAD73
    3BB5FCBC 2EC22005 C58EF183 7D1683B2 C6F34A26 C1B2EFFA
    886B4238 611FCFDC DE355B3B 6519035B BC34F4DE F99C0238
    61B46FC9 D6E6C907 7AD91D26 91F7F7EE 598CB0FA C186D91C
    AEFE1309 85139270 B4130C93 BC437944 F4FD4452 E2D74DD3
    64F2E21E 71F54BFF 5CAE82AB 9C9DF69E E86D2BC5 22363A0D
    ABC52197 9B0DEADA 1DBF9A42 D5C4484E 0ABCD06B FA53DDEF
    3C1B20EE 3FD59D7C 25E41D2B 669E1EF1 6E6F52C3 164DF4FB
    7930E9E4 E58857B6 AC7D5F42 D69F6D18 7763CF1D 55034004
    87F55BA5 7E31CC7A 7135C886 EFB4318A ED6A1E01 2D9E6832
    A907600A 918130C4 6DC778F9 71AD0038 092999A3 33CB8B7A
    1A1DB93D 7140003C 2A4ECEA9 F98D0ACC 0A8291CD CEC97DCF
    8EC9B55A 7F88A46B 4DB5A851 F44182E1 C68A007E 5E0DD902
    0BFD64B6 45036C7A 4E677D2C 38532A3A 23BA4442 CAF53EA6
    3BB45432 9B7624C8 917BDD64 B1C0FD4C B38E8C33 4C701C3A
    CDAD0657 FCCFEC71 9B1F5C3E 4E46041F 388147FB 4CFDB477
    A52471F7 A9A96910 B855322E DB6340D8 A00EF092 350511E3
    0ABEC1FF F9E3A26E 7FB29F8C 183023C3 587E38DA 0077D9B4
    763E4E4B 94B2BBC1 94C6651E 77CAF992 EEAAC023 2A281BF6
    B3A739C1 22611682 0AE8DB58 47A67CBE F9C9091B 462D538C
    D72B0374 6AE77F5E 62292C31 1562A846 505DC82D B854338A
    E49F5235 C95B9117 8CCF2DD5 CACEF403 EC9D1810 C6272B04
    5B3B71F9 DC6B80D6 3FDD4A8E 9ADB1E69 62A69526 D43161C1
    A41D570D 7938DAD4 A40E329C CFF46AAA 36AD004C F600C838
    1E425A31 D951AE64 FDB23FCE C9509D43 687FEB69 EDD1CC5E
    0B8CC3BD F64B10EF 86B63142 A3AB8829 555B2F74 7C932665
    CB2C0F1C C01BD702 29388839 D2AF05E4 54504AC7 8B758282
    2846C0BA 35C35F5C 59160CC0 46FD8251 541FC68C 9C86B022
    BB709987 6A460E74 51A8A931 09703FEE 1C217E6C 3826E52C
    51AA691E 0E423CFC 99E9E316 50C1217B 624816CD AD9A95F9
    D5B80194 88D9C0A0 A1FE3075 A577E231 83F81D4A 3F2FA457
    1EFC8CE0 BA8A4FE8 B6855DFE 72B0A66E DED2FBAB FBE58A30
    FAFABE1C 5D71A87E 2F741EF8 C1FE86FE A6BBFDE5 30677F0D
    97D11D49 F7A8443D 0822E506 A9F4614E 011E2A94 838FF88C
    D68C8BB7 C5C6424C FFFFFFFF FFFFFFFF
'''

# The G value is the same for all parameters, so
# it is only here once. All the groups from RFC 7919
# use the generator 2.

g = '''02''' # Not just '2' because these hex strings must
             # only contain bytes.

def dehexify(hexString):
    '''
        Convert the "bignum" hexString
        representation to a bytearray
        containing the number.
    '''
    return bytearray.fromhex(' '.join(line.strip() for line in hexString.splitlines()))

def derLength(size):
    '''
        Returns the ASN.1 DER length bytes
        for the given size.

        Short form: For lengths <= 127,
        the length is just a byte containing
        the literal number. (Note that for
        numbers <= 127, the MSB is not
        set.)

        In doctest form, that is:

        >>> derLength(1)
        bytearray(b'\\x01')

        >>> derLength(127)
        bytearray(b'\\x7f')

        Long form: For lengths > 127,
        the encoding is different: The
        first byte has the MSB set (that
        is, by itself, 0x80). The rest of
        the bits in the first byte specify
        the number of bytes that follow.

        So, to encode the length 128, you
        would emit

            0x81 0x80

        The high bit of the first byte is set.
        This means the remaining 7 bits contain
        how many bytes follow it. If we mask the
        MSB away, we end up with 0x01. So, only
        a single byte follows.
        The value of the following byte is 0x80,
        which is 128 decimal.

        In doctest form, that is:

        >>> derLength(128)
        bytearray(b'\\x81\\x80')

        Error cases:

        The size passed to this function must be an integer:

        >>> derLength(0.1)
        Traceback (most recent call last):
            ...
        Exception: bad type

        >>> derLength('5')
        Traceback (most recent call last):
            ...
        Exception: bad type

        Negative numbers are not supported, and throw an exception:

        >>> derLength(-1)
        Traceback (most recent call last):
            ...
        Exception: bad size

        The long form does not support very large numbers. This is
        because the 'additional bytes' counter can only count up to
        127 (2**7-1) bytes.

        So, if we try to encode a length that will use exactly 127
        bytes, it will succeed:

        >>> derLength(int('ff'*127, 16)) # doctest:+ELLIPSIS
        bytearray(b'\\xff...\\xff')

        But if we try to go over that limit, we get an exception:

        >>> derLength(int('ff'*128, 16))
        Traceback (most recent call last):
            ...
        Exception: unencodable
    '''

    # Ensure that the passed-in size is an integer.
    if not isinstance(size, (int, long)):
        raise Exception('bad type')

    if size > 127:
        # Convert 'size' to a hexString-style
        # bignum that we can pass to dehexify.
        #
        # Strip '0x' prefix from the hex string.
        hexString = hex(size)[2:]
        # In Python 2.7, the output of hex()
        # can have an L suffix. Make sure we
        # strip that.
        if hexString.endswith('L'):
            hexString = hexString[:-1]
        # Ensure the final hex string only
        # contains full bytes. If not, prepend
        # a zero byte.
        if len(hexString) % 2 != 0:
            hexString = '0' + hexString

        # If the computed hexString bignum
        # contains more than 127 bytes, we
        # can't encode it using ASN.1 DER
        # encoding. Throw an exception.
        nbytes = len(hexString) // 2
        if nbytes > 127:
            raise Exception('unencodable')

        buf = dehexify(hexString)
        out = bytearray((0x80 | nbytes,))
        out += buf
        return out
    elif size > 0:
        # Short form is simply the number itself,
        # as a byte.
        return bytearray((size,))
    else:
        # We don't support negative sizes.
        raise Exception('bad size')

def derSequence(sequence):
    '''
        Encode an ASN.1 DER sequence

        Takes a sequence of DER data as its input.

        Returns a bytearray of the resulting data.

        The encoding of a DER sequence is simply
        an identifier (0x30), a length (the length
        of the content bytes -- see the derLength
        function), and the content itself (just a
        byte stream of other ASN.1 DER objects)
    '''
    out = bytearray((0x30,)) # SEQUENCE tag
    content = bytearray()
    for entry in sequence:
        content += entry
    out += derLength(len(content))
    out += content
    return out

def derUnsignedInteger(hexString):
    '''
        Encode a hex string to a ASN.1 DER INTEGER.
        This function only handles unsigned integers.

        Returns a bytearray of the resulting data.

        The encoding of an ASN.1 DER INTEGER is an
        identifier (0x02), a length (the length of the
        content bytes -- see the derLength function),
        and the content, which in this case is the
        bytes that make up the integer.

        An INTEGER can be signed, or unsigned. Our
        function here only deals with unsigned integers.

        Signed integers have the MSB/sign bit set.

        Our input data is guaranteed to be unsigned,
        so if we detect that the MSB is set in the
        data, we prepend a zero byte to it. This
        is the way to signal that the value is unsigned
        rather than signed.
    '''
    out = bytearray((0x02,)) # INTEGER tag
    buf = dehexify(hexString)

    # If the sign bit is set, prepend a zero byte.
    # Otherwise, the number will be treated as signed,
    # which we don't want.
    if (buf[0] & 0x80) != 0:
        buf = bytearray((0,)) + buf
    out += derLength(len(buf))
    out += buf
    return out

def pem(der, kind='DH PARAMETERS'):
    '''
        Convert the ASN.1 DER data to PEM form.

        Returns the resulting string (type 'unicode' for Python 2).

        The PEM format consists of a header, the content itself (base64,
        using the standard alphabet) and a trailer.

        The header is of the form:
            -----BEGIN $KIND-----

        and the trailer is of the form:
            -----END $KIND-----

        where $KIND defines the type of data
        contained in the PEM data.

        Between the header and the trailer is
        the content itself. The content is the
        DER form encoded using base64.

        Our implementation splits the base64
        data such that each line of base64 data
        can be no longer than 64 characters long.
    '''
    out = u'-----BEGIN {0}-----\n'.format(kind)
    enc = base64.b64encode(der)
    n = 0
    for ch in enc:
        if type(ch) == int: # Python 3
            out += chr(ch)
        else:
            out += ch
        n += 1
        if n == 64:
            out += '\n'
            n = 0
    if out[-1] != '\n':
        out += '\n'
    out += u'-----END {0}-----\n'.format(kind)
    return out

def writeHeader(f):
    '''
        Output a warning about the output file being auto-generated.

    '''
    f.write('\n')
    f.write('// Automatically generated by "{0}". DO NOT EDIT BY HAND.\n'.format(os.path.basename(__file__)))
    f.write('\n')

def writeCStr(f, identifier, str):
    '''
        Output the string 'str' as a static
        C-style string (in array form, so sizeof works).
    '''
    f.write('static const char {0}[] = \\\n'.format(identifier))
    splat = str.split('\n')
    splat = splat[:-1]
    lastLine = splat[-1]
    for line in splat:
        terminator = ''
        if line == lastLine:
            terminator = ';'
        if len(line) > 0:
            f.write('"{0}\\n"{1}\n'.format(line, terminator))
def main():
    parser = argparse.ArgumentParser()
    parser.add_argument("--output", help="Path to the file to which the generated output shall be written", required=True)

    args = parser.parse_args()

    doctest.testmod()

    f = codecs.open(args.output, 'w', 'utf-8')

    writeHeader(f)

    writeCStr(f, 'ffdhe2048_pem', pem(
        derSequence((
            derUnsignedInteger(ffdhe2048_p),
            derUnsignedInteger(g),
            ))
        ))

    f.write('\n')

    writeCStr(f, 'ffdhe3072_pem', pem(
        derSequence((
            derUnsignedInteger(ffdhe3072_p),
            derUnsignedInteger(g)
            ))
        ))

    f.write('\n')

    writeCStr(f, 'ffdhe4096_pem', pem(
        derSequence((
            derUnsignedInteger(ffdhe4096_p),
            derUnsignedInteger(g),
            ))
        ))

    f.write('\n')

    writeCStr(f, 'ffdhe6144_pem', pem(
        derSequence((
            derUnsignedInteger(ffdhe6144_p),
            derUnsignedInteger(g),
            ))
        ))

    f.write('\n')

    writeCStr(f, 'ffdhe8192_pem', pem(
        derSequence((
            derUnsignedInteger(ffdhe8192_p),
            derUnsignedInteger(g),
            ))
        ))

    f.close()

if __name__ == '__main__':
    main()