1 // TortoiseGit - a Windows shell extension for easy version control
3 // Copyright (C) 2013-2014 Sven Strickroth <email@cs-ware.de>
4 // Copyright (C) 2014 TortoiseGit
5 // Copyright (C) VLC project (http://videolan.org)
6 // - pgp parsing code was copied from src/misc/update(_crypto)?.c
7 // Copyright (C) The Internet Society (1998). All Rights Reserved.
8 // - crc_octets() was lamely copied from rfc 2440
10 // This program is free software; you can redistribute it and/or
11 // modify it under the terms of the GNU General Public License
12 // as published by the Free Software Foundation; either version 2
13 // of the License, or (at your option) any later version.
15 // This program is distributed in the hope that it will be useful,
16 // but WITHOUT ANY WARRANTY; without even the implied warranty of
17 // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
18 // GNU General Public License for more details.
20 // You should have received a copy of the GNU General Public License
21 // along with this program; if not, write to the Free Software Foundation,
22 // 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
25 #include "UpdateCrypto.h"
26 #include "FormatMessageWrapper.h"
28 #define NEED_SIGNING_KEY
29 #include "..\version.h"
32 #define packet_type(c) ((c & 0x3c) >> 2) /* 0x3C = 00111100 */
33 #define packet_header_len(c) ((c & 0x03) + 1) /* number of bytes in a packet header */
35 static inline int scalar_number(const uint8_t *p
, int header_len
)
37 ASSERT(header_len
== 1 || header_len
== 2 || header_len
== 4);
41 else if (header_len
== 2)
42 return (p
[0] << 8) + p
[1];
43 else if (header_len
== 4)
44 return (p
[0] << 24) + (p
[1] << 16) + (p
[2] << 8) + p
[3];
49 /* number of data bytes in a MPI */
50 static int mpi_len(const uint8_t* mpi
)
52 return (scalar_number(mpi
, 2) + 7) / 8;
55 static size_t read_mpi(uint8_t* dst
, const uint8_t* buf
, size_t buflen
, size_t bits
)
60 size_t n
= mpi_len(buf
);
74 #define READ_MPI(d, bits) do \
76 size_t n = read_mpi(d, p_buf, i_packet_len - i_read, bits); \
83 static size_t b64_decode_binary_to_buffer(uint8_t *p_dst
, size_t i_dst
, const char *p_src
, size_t srcLen
)
86 if (!Base64Decode(p_src
, (int)srcLen
, p_dst
, &len
))
92 * crc_octets() was lamely copied from rfc 2440
93 * Copyright (C) The Internet Society (1998). All Rights Reserved.
95 #define CRC24_INIT 0xB704CEL
96 #define CRC24_POLY 0x1864CFBL
98 static long crc_octets(uint8_t *octets
, size_t len
)
100 long crc
= CRC24_INIT
;
104 crc
^= (*octets
++) << 16;
105 for (i
= 0; i
< 8; i
++)
112 return crc
& 0xFFFFFFL
;
115 static ALG_ID
map_digestalgo(uint8_t digest_algo
)
119 case DIGEST_ALGO_SHA1
:
121 case DIGEST_ALGO_SHA256
:
123 case DIGEST_ALGO_SHA384
:
125 case DIGEST_ALGO_SHA512
:
132 static DWORD
map_algo(uint8_t digest_algo
)
136 case PUBLIC_KEY_ALGO_DSA
:
138 case PUBLIC_KEY_ALGO_RSA
:
139 return PROV_RSA_AES
; // needed for SHA2, see http://msdn.microsoft.com/en-us/library/windows/desktop/aa387447%28v=vs.85%29.aspx
145 static size_t parse_signature_v3_packet(signature_packet_t
*p_sig
, const uint8_t *p_buf
, size_t i_sig_len
)
147 size_t i_read
= 1; /* we already read the version byte */
149 if (i_sig_len
< 19) /* signature is at least 19 bytes + the 2 MPIs */
152 p_sig
->specific
.v3
.hashed_data_len
= *p_buf
++; i_read
++;
153 if (p_sig
->specific
.v3
.hashed_data_len
!= 5)
156 p_sig
->type
= *p_buf
++; i_read
++;
158 memcpy(p_sig
->specific
.v3
.timestamp
, p_buf
, 4);
159 p_buf
+= 4; i_read
+= 4;
161 memcpy(p_sig
->issuer_longid
, p_buf
, 8);
162 p_buf
+= 8; i_read
+= 8;
164 p_sig
->public_key_algo
= *p_buf
++; i_read
++;
166 p_sig
->digest_algo
= *p_buf
++; i_read
++;
168 p_sig
->hash_verification
[0] = *p_buf
++; i_read
++;
169 p_sig
->hash_verification
[1] = *p_buf
++; i_read
++;
171 ASSERT(i_read
== 19);
177 * fill a signature_packet_v4_t from signature packet data
178 * verify that it was used with a DSA or RSA public key
180 static size_t parse_signature_v4_packet(signature_packet_t
*p_sig
, const uint8_t *p_buf
, size_t i_sig_len
)
182 size_t i_read
= 1; /* we already read the version byte */
184 if (i_sig_len
< 10) /* signature is at least 10 bytes + the 2 MPIs */
187 p_sig
->type
= *p_buf
++; i_read
++;
189 p_sig
->public_key_algo
= *p_buf
++; i_read
++;
191 p_sig
->digest_algo
= *p_buf
++; i_read
++;
192 if (!map_algo(p_sig
->public_key_algo
))
195 memcpy(p_sig
->specific
.v4
.hashed_data_len
, p_buf
, 2);
196 p_buf
+= 2; i_read
+= 2;
198 size_t i_hashed_data_len
= scalar_number(p_sig
->specific
.v4
.hashed_data_len
, 2);
199 i_read
+= i_hashed_data_len
;
200 if (i_read
+ 4 > i_sig_len
)
203 p_sig
->specific
.v4
.hashed_data
= (uint8_t*) malloc(i_hashed_data_len
);
204 if (!p_sig
->specific
.v4
.hashed_data
)
206 memcpy(p_sig
->specific
.v4
.hashed_data
, p_buf
, i_hashed_data_len
);
207 p_buf
+= i_hashed_data_len
;
209 memcpy(p_sig
->specific
.v4
.unhashed_data_len
, p_buf
, 2);
210 p_buf
+= 2; i_read
+= 2;
212 size_t i_unhashed_data_len
= scalar_number(p_sig
->specific
.v4
.unhashed_data_len
, 2);
213 i_read
+= i_unhashed_data_len
;
214 if (i_read
+ 2 > i_sig_len
)
217 p_sig
->specific
.v4
.unhashed_data
= (uint8_t*) malloc(i_unhashed_data_len
);
218 if (!p_sig
->specific
.v4
.unhashed_data
)
221 memcpy(p_sig
->specific
.v4
.unhashed_data
, p_buf
, i_unhashed_data_len
);
222 p_buf
+= i_unhashed_data_len
;
224 memcpy(p_sig
->hash_verification
, p_buf
, 2);
225 p_buf
+= 2; i_read
+= 2;
227 uint8_t *p
, *max_pos
;
228 p
= p_sig
->specific
.v4
.unhashed_data
;
229 max_pos
= p
+ scalar_number(p_sig
->specific
.v4
.unhashed_data_len
, 2);
236 size_t i_subpacket_len
;
241 i_subpacket_len
= *p
++;
247 i_subpacket_len
= (*p
++ - 192) << 8;
248 i_subpacket_len
+= *p
++ + 192;
254 i_subpacket_len
= size_t(*++p
) << 24;
255 i_subpacket_len
+= *++p
<< 16;
256 i_subpacket_len
+= *++p
<< 8;
257 i_subpacket_len
+= *++p
;
260 if (*p
== ISSUER_SUBPACKET
)
265 memcpy(&p_sig
->issuer_longid
, p
+ 1, 8);
270 p
+= i_subpacket_len
;
274 static int parse_signature_packet(signature_packet_t
*p_sig
, const uint8_t *p_buf
, size_t i_packet_len
)
276 if (!i_packet_len
) /* 1st sanity check, we need at least the version */
279 p_sig
->version
= *p_buf
++;
282 switch (p_sig
->version
)
285 i_read
= parse_signature_v3_packet(p_sig
, p_buf
, i_packet_len
);
288 p_sig
->specific
.v4
.hashed_data
= nullptr;
289 p_sig
->specific
.v4
.unhashed_data
= nullptr;
290 i_read
= parse_signature_v4_packet(p_sig
, p_buf
, i_packet_len
);
296 if (i_read
== 0) /* signature packet parsing has failed */
299 if (!map_algo(p_sig
->public_key_algo
))
302 if (!map_digestalgo(p_sig
->digest_algo
))
307 case BINARY_SIGNATURE
:
309 case GENERIC_KEY_SIGNATURE
:
310 case PERSONA_KEY_SIGNATURE
:
311 case CASUAL_KEY_SIGNATURE
:
312 case POSITIVE_KEY_SIGNATURE
:
318 p_buf
--; /* rewind to the version byte */
321 if (p_sig
->public_key_algo
== PUBLIC_KEY_ALGO_DSA
)
323 READ_MPI(p_sig
->algo_specific
.dsa
.r
, 160);
324 READ_MPI(p_sig
->algo_specific
.dsa
.s
, 160);
326 else if (p_sig
->public_key_algo
== PUBLIC_KEY_ALGO_RSA
)
327 READ_MPI(p_sig
->algo_specific
.rsa
.s
, 4096);
331 ASSERT(i_read
== i_packet_len
);
332 if (i_read
< i_packet_len
) /* some extra data, hm ? */
338 if (p_sig
->version
== 4)
340 free(p_sig
->specific
.v4
.hashed_data
);
341 free(p_sig
->specific
.v4
.unhashed_data
);
348 * Transform an armored document in binary format
349 * Used on public keys and signatures
351 static int pgp_unarmor(const char *p_ibuf
, size_t i_ibuf_len
, uint8_t *p_obuf
, size_t i_obuf_len
)
353 const char *p_ipos
= p_ibuf
;
354 uint8_t *p_opos
= p_obuf
;
356 int i_header_skipped
= 0;
358 while (!i_end
&& p_ipos
< p_ibuf
+ i_ibuf_len
&& *p_ipos
!= '=')
360 if (*p_ipos
== '\r' || *p_ipos
== '\n')
366 size_t i_line_len
= strcspn(p_ipos
, "\r\n");
370 if (!i_header_skipped
)
372 if (!strncmp(p_ipos
, "-----BEGIN PGP", 14))
373 i_header_skipped
= 1;
375 p_ipos
+= i_line_len
+ 1;
379 if (!strncmp(p_ipos
, "Version:", 8))
381 p_ipos
+= i_line_len
+ 1;
385 if (p_ipos
[i_line_len
- 1] == '=')
390 p_opos
+= b64_decode_binary_to_buffer(p_opos
, p_obuf
- p_opos
+ i_obuf_len
, p_ipos
, (int)i_line_len
);
391 p_ipos
+= i_line_len
+ 1;
394 if (p_ipos
+ 1 < p_ibuf
+ i_ibuf_len
&& (*p_ipos
== '\r' || *p_ipos
== '\n'))
397 /* XXX: the CRC is OPTIONAL, really require it ? */
398 if (p_ipos
+ 5 > p_ibuf
+ i_ibuf_len
|| *p_ipos
++ != '=')
402 if (b64_decode_binary_to_buffer(p_crc
, sizeof(p_crc
), p_ipos
, 5) != 3)
405 long l_crc
= crc_octets(p_obuf
, p_opos
- p_obuf
);
406 long l_crc2
= (0 << 24) + (p_crc
[0] << 16) + (p_crc
[1] << 8) + p_crc
[2];
408 return (int)((l_crc2
== l_crc
) ? p_opos
- p_obuf
: 0);
412 * fill a public_key_packet_t structure from public key packet data
413 * verify that it is a version 4 public key packet, using DSA or RSA
415 static int parse_public_key_packet(public_key_packet_t
*p_key
, const uint8_t *p_buf
, size_t i_packet_len
)
417 if (i_packet_len
< 6)
422 p_key
->version
= *p_buf
++; i_read
++;
423 if (p_key
->version
!= 4)
426 /* XXX: warn when timestamp is > date ? */
427 memcpy(p_key
->timestamp
, p_buf
, 4); p_buf
+= 4; i_read
+= 4;
429 p_key
->algo
= *p_buf
++; i_read
++;
430 if (p_key
->algo
== PUBLIC_KEY_ALGO_DSA
)
432 if (i_packet_len
> 418) // we only support 1024-bit DSA keys and SHA1 signatures, see verify_signature_dsa
434 READ_MPI(p_key
->sig
.dsa
.p
, 1024);
435 READ_MPI(p_key
->sig
.dsa
.q
, 160);
436 READ_MPI(p_key
->sig
.dsa
.g
, 1024);
437 READ_MPI(p_key
->sig
.dsa
.y
, 1024);
439 else if (p_key
->algo
== PUBLIC_KEY_ALGO_RSA
)
441 READ_MPI(p_key
->sig
.rsa
.n
, 4096);
442 READ_MPI(p_key
->sig
.rsa
.e
, 4096);
447 if (i_read
== i_packet_len
)
455 * fill a public_key_t with public key data, including:
456 * * public key packet
457 * * signature packet issued by key which long id is p_sig_issuer
460 static int parse_public_key(const uint8_t *p_key_data
, size_t i_key_len
, public_key_t
*p_key
, const uint8_t *p_sig_issuer
)
462 const uint8_t *pos
= p_key_data
;
463 const uint8_t *max_pos
= pos
+ i_key_len
;
466 #define PUBLIC_KEY_FOUND 0x01
467 #define USER_ID_FOUND 0x02
468 #define SIGNATURE_FOUND 0X04
470 uint8_t *p_key_unarmored
= nullptr;
472 p_key
->psz_username
= nullptr;
473 p_key
->sig
.specific
.v4
.hashed_data
= nullptr;
474 p_key
->sig
.specific
.v4
.unhashed_data
= nullptr;
477 { /* first byte is ASCII, unarmoring */
478 p_key_unarmored
= (uint8_t*)malloc(i_key_len
);
479 if (!p_key_unarmored
)
481 int i_len
= pgp_unarmor((char*)p_key_data
, i_key_len
, p_key_unarmored
, i_key_len
);
486 pos
= p_key_unarmored
;
487 max_pos
= pos
+ i_len
;
490 while (pos
< max_pos
)
492 if (!(*pos
& 0x80) || *pos
& 0x40)
495 int i_type
= packet_type(*pos
);
497 int i_header_len
= packet_header_len(*pos
++);
498 if (pos
+ i_header_len
> max_pos
|| (i_header_len
!= 1 && i_header_len
!= 2 && i_header_len
!= 4))
501 int i_packet_len
= scalar_number(pos
, i_header_len
);
504 if (pos
+ i_packet_len
> max_pos
)
509 case PUBLIC_KEY_PACKET
:
510 i_status
|= PUBLIC_KEY_FOUND
;
511 if (parse_public_key_packet(&p_key
->key
, pos
, i_packet_len
) != 0)
515 case SIGNATURE_PACKET
: /* we accept only v4 signatures here */
516 if (i_status
& SIGNATURE_FOUND
|| !p_sig_issuer
)
518 if (parse_signature_packet(&p_key
->sig
, pos
, i_packet_len
) == 0)
520 if (p_key
->sig
.version
!= 4)
522 if (memcmp( p_key
->sig
.issuer_longid
, p_sig_issuer
, 8))
524 free(p_key
->sig
.specific
.v4
.hashed_data
);
525 free(p_key
->sig
.specific
.v4
.unhashed_data
);
526 p_key
->sig
.specific
.v4
.hashed_data
= nullptr;
527 p_key
->sig
.specific
.v4
.unhashed_data
= nullptr;
530 i_status
|= SIGNATURE_FOUND
;
535 if (p_key
->psz_username
) /* save only the first User ID */
537 i_status
|= USER_ID_FOUND
;
538 p_key
->psz_username
= (uint8_t*)malloc(i_packet_len
+ 1);
539 if (!p_key
->psz_username
)
542 memcpy(p_key
->psz_username
, pos
, i_packet_len
);
543 p_key
->psz_username
[i_packet_len
] = '\0';
551 free(p_key_unarmored
);
553 if (!(i_status
& (PUBLIC_KEY_FOUND
| USER_ID_FOUND
)))
556 if (p_sig_issuer
&& !(i_status
& SIGNATURE_FOUND
))
562 if (p_key
->sig
.version
== 4)
564 free(p_key
->sig
.specific
.v4
.hashed_data
);
565 free(p_key
->sig
.specific
.v4
.unhashed_data
);
567 free(p_key
->psz_username
);
568 free(p_key_unarmored
);
572 static int LoadSignature(const CString
&signatureFilename
, signature_packet_t
*p_sig
)
574 FILE * pFile
= _tfsopen(signatureFilename
, _T("rb"), SH_DENYWR
);
578 std::unique_ptr
<unsigned char[]> buffer(new unsigned char[size
]);
580 if ((length
= (int)fread(buffer
.get(), sizeof(char), size
, pFile
)) >= 8)
583 // is unpacking needed?
584 if ((uint8_t)buffer
[0] < 0x80)
586 std::unique_ptr
<unsigned char[]> unpacked(new unsigned char[size
]);
587 size
= pgp_unarmor((char *)buffer
.get(), length
, unpacked
.get(), length
);
592 buffer
.swap(unpacked
);
597 if (packet_type(buffer
[0]) != SIGNATURE_PACKET
)
600 DWORD i_header_len
= packet_header_len(buffer
[0]);
601 if ((i_header_len
!= 1 && i_header_len
!= 2 && i_header_len
!= 4) || i_header_len
+ 1 > (DWORD
)size
)
604 DWORD i_len
= scalar_number((uint8_t *)(buffer
.get() + 1), i_header_len
);
605 if (i_len
+ i_header_len
+ 1 != (DWORD
)size
)
608 if (parse_signature_packet(p_sig
, (uint8_t *)(buffer
.get() + 1 + i_header_len
), i_len
))
611 if (p_sig
->type
!= BINARY_SIGNATURE
&& p_sig
->type
!= TEXT_SIGNATURE
)
613 if (p_sig
->version
== 4)
615 free(p_sig
->specific
.v4
.hashed_data
);
616 free(p_sig
->specific
.v4
.unhashed_data
);
629 static void CryptHashChar(HCRYPTHASH hHash
, const int c
)
631 CryptHashData(hHash
, (BYTE
*)&c
, 1, 0);
634 /* final part of the hash */
635 static int hash_finish(HCRYPTHASH hHash
, signature_packet_t
*p_sig
)
637 if (p_sig
->version
== 3)
639 CryptHashChar(hHash
, p_sig
->type
);
640 CryptHashData(hHash
, (unsigned char*)&p_sig
->specific
.v3
.timestamp
, 4, 0);
642 else if (p_sig
->version
== 4)
644 CryptHashChar(hHash
, p_sig
->version
);
645 CryptHashChar(hHash
, p_sig
->type
);
646 CryptHashChar(hHash
, p_sig
->public_key_algo
);
647 CryptHashChar(hHash
, p_sig
->digest_algo
);
648 CryptHashData(hHash
, p_sig
->specific
.v4
.hashed_data_len
, 2, 0);
649 unsigned int i_len
= scalar_number(p_sig
->specific
.v4
.hashed_data_len
, 2);
650 CryptHashData(hHash
, p_sig
->specific
.v4
.hashed_data
, i_len
, 0);
652 CryptHashChar(hHash
, 0x04);
653 CryptHashChar(hHash
, 0xFF);
655 i_len
+= 6; /* hashed data + 6 bytes header */
657 CryptHashChar(hHash
, (i_len
>> 24) & 0xff);
658 CryptHashChar(hHash
, (i_len
>> 16) & 0xff);
659 CryptHashChar(hHash
, (i_len
>> 8) & 0xff);
660 CryptHashChar(hHash
, (i_len
) & 0xff);
663 { /* RFC 4880 only tells about versions 3 and 4 */
671 * Generate a hash on a public key, to verify a signature made on that hash
672 * Note that we need the signature (v4) to compute the hash
674 static int hash_from_public_key(HCRYPTHASH hHash
, public_key_t
* p_pkey
)
676 if (p_pkey
->sig
.version
!= 4)
679 if (p_pkey
->sig
.type
< GENERIC_KEY_SIGNATURE
|| p_pkey
->sig
.type
> POSITIVE_KEY_SIGNATURE
)
683 unsigned int i_p_len
= 0, i_g_len
= 0, i_q_len
= 0, i_y_len
= 0;
684 unsigned int i_n_len
= 0, i_e_len
= 0;
686 if (p_pkey
->key
.algo
== PUBLIC_KEY_ALGO_DSA
)
688 i_p_len
= mpi_len(p_pkey
->key
.sig
.dsa
.p
);
689 i_g_len
= mpi_len(p_pkey
->key
.sig
.dsa
.g
);
690 i_q_len
= mpi_len(p_pkey
->key
.sig
.dsa
.q
);
691 i_y_len
= mpi_len(p_pkey
->key
.sig
.dsa
.y
);
693 i_size
= 6 + 2 * 4 + i_p_len
+ i_g_len
+ i_q_len
+ i_y_len
;
695 else if (p_pkey
->key
.algo
== PUBLIC_KEY_ALGO_RSA
)
697 i_n_len
= mpi_len(p_pkey
->key
.sig
.rsa
.n
);
698 i_e_len
= mpi_len(p_pkey
->key
.sig
.rsa
.e
);
700 i_size
= 6 + 2 * 2 + i_n_len
+ i_e_len
;
705 CryptHashChar(hHash
, 0x99);
707 CryptHashChar(hHash
, (i_size
>> 8) & 0xff);
708 CryptHashChar(hHash
, i_size
& 0xff);
710 CryptHashChar(hHash
, p_pkey
->key
.version
);
711 CryptHashData(hHash
, p_pkey
->key
.timestamp
, 4, 0);
712 CryptHashChar(hHash
, p_pkey
->key
.algo
);
714 if (p_pkey
->key
.algo
== PUBLIC_KEY_ALGO_DSA
)
716 CryptHashData(hHash
, (uint8_t*)&p_pkey
->key
.sig
.dsa
.p
, 2 + i_p_len
, 0);
717 CryptHashData(hHash
, (uint8_t*)&p_pkey
->key
.sig
.dsa
.q
, 2 + i_q_len
, 0);
718 CryptHashData(hHash
, (uint8_t*)&p_pkey
->key
.sig
.dsa
.g
, 2 + i_g_len
, 0);
719 CryptHashData(hHash
, (uint8_t*)&p_pkey
->key
.sig
.dsa
.y
, 2 + i_y_len
, 0);
721 else if (p_pkey
->key
.algo
== PUBLIC_KEY_ALGO_RSA
)
723 CryptHashData(hHash
, (uint8_t*)&p_pkey
->key
.sig
.rsa
.n
, 2 + i_n_len
, 0);
724 CryptHashData(hHash
, (uint8_t*)&p_pkey
->key
.sig
.rsa
.e
, 2 + i_e_len
, 0);
727 CryptHashChar(hHash
, 0xb4);
729 size_t i_len
= strlen((char *)p_pkey
->psz_username
);
731 CryptHashChar(hHash
, (i_len
>> 24) & 0xff);
732 CryptHashChar(hHash
, (i_len
>> 16) & 0xff);
733 CryptHashChar(hHash
, (i_len
>> 8) & 0xff);
734 CryptHashChar(hHash
, (i_len
) & 0xff);
736 CryptHashData(hHash
, p_pkey
->psz_username
, (DWORD
)i_len
, 0);
738 return hash_finish(hHash
, &p_pkey
->sig
);
741 static int hash_from_file(HCRYPTHASH hHash
, CString filename
, signature_packet_t
* p_sig
)
743 FILE * pFile
= _tfsopen(filename
, _T("rb"), SH_DENYWR
);
747 char buf
[4097] = { 0 };
750 while ((read
= (int)fread(buf
, sizeof(char), sizeof(buf
) - 1, pFile
)) > 0)
752 if (p_sig
->type
== TEXT_SIGNATURE
)
755 char * psz_string
= buf
;
758 if (nlHandling
== 1 && (*psz_string
== '\r' || *psz_string
== '\n'))
760 CryptHashChar(hHash
, '\r');
761 CryptHashChar(hHash
, '\n');
764 if (nlHandling
== 2 && *psz_string
== '\r')
772 if ((nlHandling
== 2 || nlHandling
== 3) && *psz_string
== '\n')
779 size_t i_len
= strcspn(psz_string
, "\r\n");
783 CryptHashData(hHash
, (BYTE
*)psz_string
, (DWORD
)i_len
, 0);
788 if (*psz_string
== '\r' || *psz_string
== '\n')
790 CryptHashChar(hHash
, '\r');
791 CryptHashChar(hHash
, '\n');
794 if (*psz_string
== '\r')
800 if (*psz_string
== '\n')
809 CryptHashData(hHash
, (BYTE
*)buf
, read
, 0);
814 return hash_finish(hHash
, p_sig
);
817 static int check_hash(HCRYPTHASH hHash
, signature_packet_t
*p_sig
)
820 DWORD hashLenLen
= sizeof(DWORD
);
821 if (!CryptGetHashParam(hHash
, HP_HASHSIZE
, (BYTE
*)&hashLen
, &hashLenLen
, 0))
824 std::unique_ptr
<BYTE
[]> pHash(new BYTE
[hashLen
]);
825 CryptGetHashParam(hHash
, HP_HASHVAL
, pHash
.get(), &hashLen
, 0);
827 if (pHash
.get()[0] != p_sig
->hash_verification
[0] || pHash
.get()[1] != p_sig
->hash_verification
[1])
834 * Verify an OpenPGP signature made with some RSA public key
836 static int verify_signature_rsa(HCRYPTPROV hCryptProv
, HCRYPTHASH hHash
, public_key_t
& p_pkey
, signature_packet_t
& p_sig
)
838 int i_n_len
= min(mpi_len(p_pkey
.key
.sig
.rsa
.n
), sizeof(p_pkey
.key
.sig
.rsa
.n
) - 2);
839 int i_s_len
= min(mpi_len(p_sig
.algo_specific
.rsa
.s
), sizeof(p_sig
.algo_specific
.rsa
.s
) - 2);
842 rsakey
.blobheader
.bType
= PUBLICKEYBLOB
; // 0x06
843 rsakey
.blobheader
.bVersion
= CUR_BLOB_VERSION
; // 0x02
844 rsakey
.blobheader
.reserved
= 0;
845 rsakey
.blobheader
.aiKeyAlg
= CALG_RSA_KEYX
;
846 rsakey
.rsapubkey
.magic
= 0x31415352;// ASCII for RSA1
847 rsakey
.rsapubkey
.bitlen
= i_n_len
* 8;
848 rsakey
.rsapubkey
.pubexp
= 65537; // gnupg only uses this
850 memcpy(rsakey
.n
, p_pkey
.key
.sig
.rsa
.n
+ 2, i_n_len
); std::reverse(rsakey
.n
, rsakey
.n
+ i_n_len
);
853 if (CryptImportKey(hCryptProv
, (BYTE
*)&rsakey
, sizeof(BLOBHEADER
) + sizeof(RSAPUBKEY
) + i_n_len
, 0, 0, &hPubKey
) == 0)
856 std::unique_ptr
<BYTE
[]> pSig(new BYTE
[i_s_len
]);
857 memcpy(pSig
.get(), p_sig
.algo_specific
.rsa
.s
+ 2, i_s_len
);
858 std::reverse(pSig
.get(), pSig
.get() + i_s_len
);
859 if (!CryptVerifySignature(hHash
, pSig
.get(), i_s_len
, hPubKey
, nullptr, 0))
861 CryptDestroyKey(hPubKey
);
865 CryptDestroyKey(hPubKey
);
870 * Verify an OpenPGP signature made with some DSA public key
872 static int verify_signature_dsa(HCRYPTPROV hCryptProv
, HCRYPTHASH hHash
, public_key_t
& p_pkey
, signature_packet_t
& p_sig
)
874 if (p_sig
.digest_algo
!= DIGEST_ALGO_SHA1
) // PROV_DSS only supports SHA1 signatures, see http://msdn.microsoft.com/en-us/library/windows/desktop/aa387434%28v=vs.85%29.aspx
878 // based on http://www.derkeiler.com/Newsgroups/microsoft.public.platformsdk.security/2004-10/0040.html
880 dsakey
.blobheader
.bType
= PUBLICKEYBLOB
; // 0x06
881 dsakey
.blobheader
.bVersion
= CUR_BLOB_VERSION
+ 1; // 0x03
882 dsakey
.blobheader
.reserved
= 0;
883 dsakey
.blobheader
.aiKeyAlg
= CALG_DSS_SIGN
;
884 dsakey
.dsspubkeyver3
.magic
= 0x33535344; // ASCII of "DSS3";
885 dsakey
.dsspubkeyver3
.bitlenP
= 1024; // # of bits in prime modulus
886 dsakey
.dsspubkeyver3
.bitlenQ
= 160; // # of bits in prime q, 0 if not available
887 dsakey
.dsspubkeyver3
.bitlenJ
= 0; // # of bits in (p-1)/q, 0 if not available
888 dsakey
.dsspubkeyver3
.DSSSeed
.counter
= 0xFFFFFFFF; // not available
890 memcpy(dsakey
.p
, p_pkey
.key
.sig
.dsa
.p
+ 2, sizeof(p_pkey
.key
.sig
.dsa
.p
) - 2); std::reverse(dsakey
.p
, dsakey
.p
+ sizeof(dsakey
.p
));
891 memcpy(dsakey
.q
, p_pkey
.key
.sig
.dsa
.q
+ 2, sizeof(p_pkey
.key
.sig
.dsa
.q
) - 2); std::reverse(dsakey
.q
, dsakey
.q
+ sizeof(dsakey
.q
));
892 memcpy(dsakey
.g
, p_pkey
.key
.sig
.dsa
.g
+ 2, sizeof(p_pkey
.key
.sig
.dsa
.g
) - 2); std::reverse(dsakey
.g
, dsakey
.g
+ sizeof(dsakey
.g
));
893 memcpy(dsakey
.y
, p_pkey
.key
.sig
.dsa
.y
+ 2, sizeof(p_pkey
.key
.sig
.dsa
.y
) - 2); std::reverse(dsakey
.y
, dsakey
.y
+ sizeof(dsakey
.y
));
895 if (CryptImportKey(hCryptProv
, (BYTE
*)&dsakey
, sizeof(dsakey
), 0, 0, &hPubKey
) == 0)
898 unsigned char signature
[40] = { 0 };
899 memcpy(signature
, p_sig
.algo_specific
.dsa
.r
+ 2, 20);
900 memcpy(signature
+ 20, p_sig
.algo_specific
.dsa
.s
+ 2, 20);
901 std::reverse(signature
, signature
+ 20);
902 std::reverse(signature
+ 20, signature
+ 40);
903 if (!CryptVerifySignature(hHash
, signature
, sizeof(signature
), hPubKey
, nullptr, 0))
905 CryptDestroyKey(hPubKey
);
909 CryptDestroyKey(hPubKey
);
914 * Verify an OpenPGP signature made with some public key
916 int verify_signature(HCRYPTPROV hCryptProv
, HCRYPTHASH hHash
, public_key_t
& p_key
, signature_packet_t
& sign
)
918 if (sign
.public_key_algo
== PUBLIC_KEY_ALGO_DSA
)
919 return verify_signature_dsa(hCryptProv
, hHash
, p_key
, sign
);
920 else if (sign
.public_key_algo
== PUBLIC_KEY_ALGO_RSA
)
921 return verify_signature_rsa(hCryptProv
, hHash
, p_key
, sign
);
927 * download a public key (the last one) from TortoiseGit server, and parse it
929 static public_key_t
*download_key(const uint8_t *p_longid
, const uint8_t *p_signature_issuer
, CUpdateDownloader
*updateDownloader
)
931 ASSERT(updateDownloader
);
934 url
.Format(L
"http://download.tortoisegit.org/keys/%.2X%.2X%.2X%.2X%.2X%.2X%.2X%.2X.asc", p_longid
[0], p_longid
[1], p_longid
[2], p_longid
[3], p_longid
[4], p_longid
[5], p_longid
[6], p_longid
[7]);
936 CString tempfile
= CTempFiles::Instance().GetTempFilePath(true).GetWinPathString();
937 if (updateDownloader
->DownloadFile(url
, tempfile
, false))
941 std::unique_ptr
<char[]> buffer(new char[size
]);
942 FILE * pFile
= _tfsopen(tempfile
, _T("rb"), SH_DENYWR
);
946 if ((length
= (int)fread(buffer
.get(), sizeof(char), size
, pFile
)) >= 8)
960 public_key_t
*p_pkey
= (public_key_t
*) malloc(sizeof(public_key_t
));
963 DeleteUrlCacheEntry(url
);
967 memcpy(p_pkey
->longid
, p_longid
, 8);
969 if (parse_public_key((const uint8_t *)buffer
.get(), size
, p_pkey
, p_signature_issuer
))
978 int VerifyIntegrity(const CString
&filename
, const CString
&signatureFilename
, CUpdateDownloader
*updateDownloader
)
980 ASSERT(updateDownloader
);
982 signature_packet_t p_sig
;
983 memset(&p_sig
, 0, sizeof(signature_packet_t
));
984 if (LoadSignature(signatureFilename
, &p_sig
))
988 memset(&p_pkey
, 0, sizeof(public_key_t
));
989 if (parse_public_key(tortoisegit_public_key
, sizeof(tortoisegit_public_key
), &p_pkey
, nullptr))
991 if (p_sig
.version
== 4)
993 free(p_sig
.specific
.v4
.hashed_data
);
994 free(p_sig
.specific
.v4
.unhashed_data
);
998 memcpy(p_pkey
.longid
, tortoisegit_public_key_longid
, 8);
1000 HCRYPTPROV hCryptProv
;
1001 if (!CryptAcquireContext(&hCryptProv
, nullptr, nullptr, map_algo(p_pkey
.key
.algo
), CRYPT_VERIFYCONTEXT
))
1003 if (p_sig
.version
== 4)
1005 free(p_sig
.specific
.v4
.hashed_data
);
1006 free(p_sig
.specific
.v4
.unhashed_data
);
1008 free(p_pkey
.psz_username
);
1012 if (memcmp(p_sig
.issuer_longid
, p_pkey
.longid
, 8) != 0)
1014 public_key_t
*p_new_pkey
= download_key(p_sig
.issuer_longid
, tortoisegit_public_key_longid
, updateDownloader
);
1017 if (p_sig
.version
== 4)
1019 free(p_sig
.specific
.v4
.hashed_data
);
1020 free(p_sig
.specific
.v4
.unhashed_data
);
1022 free(p_pkey
.psz_username
);
1023 CryptReleaseContext(hCryptProv
, 0);
1028 if (!CryptCreateHash(hCryptProv
, map_digestalgo(p_sig
.digest_algo
), 0, 0, &hHash
))
1030 if (p_sig
.version
== 4)
1032 free(p_sig
.specific
.v4
.hashed_data
);
1033 free(p_sig
.specific
.v4
.unhashed_data
);
1035 free(p_pkey
.psz_username
);
1036 CryptReleaseContext(hCryptProv
, 0);
1037 free(p_new_pkey
->psz_username
);
1038 if (p_new_pkey
->sig
.version
== 4)
1040 free(p_new_pkey
->sig
.specific
.v4
.hashed_data
);
1041 free(p_new_pkey
->sig
.specific
.v4
.unhashed_data
);
1047 if (hash_from_public_key(hHash
, p_new_pkey
))
1049 if (p_sig
.version
== 4)
1051 free(p_sig
.specific
.v4
.hashed_data
);
1052 free(p_sig
.specific
.v4
.unhashed_data
);
1054 free(p_pkey
.psz_username
);
1055 CryptReleaseContext(hCryptProv
, 0);
1056 free(p_new_pkey
->psz_username
);
1057 if (p_new_pkey
->sig
.version
== 4)
1059 free(p_new_pkey
->sig
.specific
.v4
.hashed_data
);
1060 free(p_new_pkey
->sig
.specific
.v4
.unhashed_data
);
1063 CryptDestroyHash(hHash
);
1067 if (check_hash(hHash
, &p_new_pkey
->sig
))
1069 if (p_sig
.version
== 4)
1071 free(p_sig
.specific
.v4
.hashed_data
);
1072 free(p_sig
.specific
.v4
.unhashed_data
);
1074 free(p_pkey
.psz_username
);
1075 CryptReleaseContext(hCryptProv
, 0);
1076 free(p_new_pkey
->psz_username
);
1077 if (p_new_pkey
->sig
.version
== 4)
1079 free(p_new_pkey
->sig
.specific
.v4
.hashed_data
);
1080 free(p_new_pkey
->sig
.specific
.v4
.unhashed_data
);
1083 CryptDestroyHash(hHash
);
1087 if (verify_signature(hCryptProv
, hHash
, p_pkey
, p_new_pkey
->sig
))
1089 if (p_sig
.version
== 4)
1091 free(p_sig
.specific
.v4
.hashed_data
);
1092 free(p_sig
.specific
.v4
.unhashed_data
);
1094 free(p_pkey
.psz_username
);
1095 CryptReleaseContext(hCryptProv
, 0);
1096 free(p_new_pkey
->psz_username
);
1097 if (p_new_pkey
->sig
.version
== 4)
1099 free(p_new_pkey
->sig
.specific
.v4
.hashed_data
);
1100 free(p_new_pkey
->sig
.specific
.v4
.unhashed_data
);
1103 CryptDestroyHash(hHash
);
1108 CryptDestroyHash(hHash
);
1109 free(p_pkey
.psz_username
);
1110 p_pkey
= *p_new_pkey
;
1111 if (p_pkey
.sig
.version
== 4)
1113 p_pkey
.sig
.version
= 0;
1114 free(p_pkey
.sig
.specific
.v4
.hashed_data
);
1115 free(p_pkey
.sig
.specific
.v4
.unhashed_data
);
1124 if (!CryptCreateHash(hCryptProv
, map_digestalgo(p_sig
.digest_algo
), 0, 0, &hHash
))
1127 if (hash_from_file(hHash
, filename
, &p_sig
))
1130 if (check_hash(hHash
, &p_sig
))
1133 if (verify_signature(hCryptProv
, hHash
, p_pkey
, p_sig
))
1139 CryptDestroyHash(hHash
);
1140 CryptReleaseContext(hCryptProv
, 0);
1142 free(p_pkey
.psz_username
);
1143 if (p_sig
.version
== 4)
1145 free(p_sig
.specific
.v4
.hashed_data
);
1146 free(p_sig
.specific
.v4
.unhashed_data
);