1
0
mirror of git://git.gnupg.org/gnupg.git synced 2024-11-13 22:08:52 +01:00
gnupg/g10/pubkey-enc.c

350 lines
9.3 KiB
C
Raw Normal View History

1997-11-18 15:06:00 +01:00
/* pubkey-enc.c - public key encoded packet handling
2002-06-29 15:46:34 +02:00
* Copyright (C) 1998, 1999, 2000, 2001, 2002 Free Software Foundation, Inc.
1997-11-18 15:06:00 +01:00
*
* This file is part of GnuPG.
1997-11-18 15:06:00 +01:00
*
* GnuPG is free software; you can redistribute it and/or modify
1997-11-18 15:06:00 +01:00
* it under the terms of the GNU General Public License as published by
2007-10-23 12:48:09 +02:00
* the Free Software Foundation; either version 3 of the License, or
1997-11-18 15:06:00 +01:00
* (at your option) any later version.
*
* GnuPG is distributed in the hope that it will be useful,
1997-11-18 15:06:00 +01:00
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
2007-10-23 12:48:09 +02:00
* along with this program; if not, see <http://www.gnu.org/licenses/>.
1997-11-18 15:06:00 +01:00
*/
#include <config.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <assert.h>
#include "util.h"
2002-06-29 15:46:34 +02:00
#include "memory.h"
1997-11-18 15:06:00 +01:00
#include "packet.h"
2002-06-29 15:46:34 +02:00
#include "mpi.h"
1997-11-18 15:06:00 +01:00
#include "keydb.h"
1998-08-05 18:51:59 +02:00
#include "trustdb.h"
2002-06-29 15:46:34 +02:00
#include "cipher.h"
1998-05-29 13:53:54 +02:00
#include "status.h"
#include "options.h"
#include "main.h"
1998-08-08 21:27:00 +02:00
#include "i18n.h"
#include "cardglue.h"
1997-11-18 15:06:00 +01:00
1998-09-11 07:47:32 +02:00
static int get_it( PKT_pubkey_enc *k,
DEK *dek, PKT_secret_key *sk, u32 *keyid );
1997-11-18 15:06:00 +01:00
2002-06-29 15:46:34 +02:00
/* check that the given algo is mentioned in one of the valid user IDs */
static int
2002-06-29 15:46:34 +02:00
is_algo_in_prefs ( KBNODE keyblock, preftype_t type, int algo )
{
2002-06-29 15:46:34 +02:00
KBNODE k;
for (k=keyblock; k; k=k->next) {
if (k->pkt->pkttype == PKT_USER_ID) {
PKT_user_id *uid = k->pkt->pkt.user_id;
prefitem_t *prefs = uid->prefs;
2002-06-29 15:46:34 +02:00
if (uid->created && prefs &&
!uid->is_revoked && !uid->is_expired ) {
for (; prefs->type; prefs++ )
if (prefs->type == type && prefs->value == algo)
return 1;
}
}
}
return 0;
}
1997-11-18 15:06:00 +01:00
/****************
2002-06-29 15:46:34 +02:00
* Get the session key from a pubkey enc packet and return
1997-11-18 15:06:00 +01:00
* it in DEK, which should have been allocated in secure memory.
*/
int
get_session_key( PKT_pubkey_enc *k, DEK *dek )
{
1998-09-11 07:47:32 +02:00
PKT_secret_key *sk = NULL;
int rc;
1997-11-18 15:06:00 +01:00
2002-06-29 15:46:34 +02:00
rc = check_pubkey_algo2 (k->pubkey_algo, PUBKEY_USAGE_ENC);
1998-06-11 09:16:50 +02:00
if( rc )
goto leave;
1998-05-29 13:53:54 +02:00
if( (k->keyid[0] || k->keyid[1]) && !opt.try_all_secrets ) {
2005-07-27 20:10:56 +02:00
sk = xmalloc_clear( sizeof *sk );
1998-09-11 07:47:32 +02:00
sk->pubkey_algo = k->pubkey_algo; /* we want a pubkey with this algo*/
if( !(rc = get_seckey( sk, k->keyid )) )
rc = get_it( k, dek, sk, k->keyid );
}
else { /* anonymous receiver: Try all available secret keys */
void *enum_context = NULL;
u32 keyid[2];
char *p;
1998-09-11 07:47:32 +02:00
for(;;) {
if( sk )
free_secret_key( sk );
2005-07-27 20:10:56 +02:00
sk = xmalloc_clear( sizeof *sk );
rc=enum_secret_keys( &enum_context, sk, 1, 0);
1998-09-11 07:47:32 +02:00
if( rc ) {
2002-06-29 15:46:34 +02:00
rc = G10ERR_NO_SECKEY;
1998-09-11 07:47:32 +02:00
break;
}
if( sk->pubkey_algo != k->pubkey_algo )
continue;
keyid_from_sk( sk, keyid );
log_info(_("anonymous recipient; trying secret key %s ...\n"),
keystr(keyid));
if(!opt.try_all_secrets && !is_status_enabled())
{
p=get_last_passphrase();
set_next_passphrase(p);
2005-07-27 20:10:56 +02:00
xfree(p);
}
rc = check_secret_key( sk, opt.try_all_secrets?1:-1 ); /* ask
only
once */
1998-09-11 07:47:32 +02:00
if( !rc )
{
1998-09-11 07:47:32 +02:00
rc = get_it( k, dek, sk, keyid );
/* Successfully checked the secret key (either it was
a card, had no passphrase, or had the right
passphrase) but couldn't decrypt the session key,
so thus that key is not the anonymous recipient.
Move the next passphrase into last for the next
round. We only do this if the secret key was
successfully checked as in the normal case,
check_secret_key handles this for us via
passphrase_to_dek */
if(rc)
next_to_last_passphrase();
}
if( !rc )
{
log_info(_("okay, we are the anonymous recipient.\n") );
1998-09-11 07:47:32 +02:00
break;
}
1998-09-11 07:47:32 +02:00
}
enum_secret_keys( &enum_context, NULL, 0, 0 ); /* free context */
1998-09-11 07:47:32 +02:00
}
leave:
if( sk )
free_secret_key( sk );
return rc;
}
static int
2002-06-29 15:46:34 +02:00
get_it( PKT_pubkey_enc *enc, DEK *dek, PKT_secret_key *sk, u32 *keyid )
1998-09-11 07:47:32 +02:00
{
int rc;
MPI plain_dek = NULL;
byte *frame = NULL;
unsigned n, nframe;
u16 csum, csum2;
int card = 0;
if (sk->is_protected && sk->protect.s2k.mode == 1002)
{ /* Note, that we only support RSA for now. */
#ifdef ENABLE_CARD_SUPPORT
unsigned char *rbuf;
size_t rbuflen;
char *snbuf;
unsigned char *indata = NULL;
unsigned int indatalen;
snbuf = serialno_and_fpr_from_sk (sk->protect.iv, sk->protect.ivlen, sk);
indata = mpi_get_buffer (enc->data[0], &indatalen, NULL);
if (!indata)
BUG ();
rc = agent_scd_pkdecrypt (snbuf, indata, indatalen, &rbuf, &rbuflen);
xfree (snbuf);
xfree (indata);
if (rc)
goto leave;
frame = rbuf;
nframe = rbuflen;
card = 1;
#else
rc = G10ERR_UNSUPPORTED;
goto leave;
#endif /*!ENABLE_CARD_SUPPORT*/
}
else
{
rc = pubkey_decrypt(sk->pubkey_algo, &plain_dek, enc->data, sk->skey );
if( rc )
1997-11-18 15:06:00 +01:00
goto leave;
frame = mpi_get_buffer( plain_dek, &nframe, NULL );
mpi_free( plain_dek ); plain_dek = NULL;
}
1997-11-24 23:24:04 +01:00
1998-02-13 21:58:50 +01:00
/* Now get the DEK (data encryption key) from the frame
1997-11-18 15:06:00 +01:00
*
* Old versions encode the DEK in in this format (msb is left):
*
* 0 1 DEK(16 bytes) CSUM(2 bytes) 0 RND(n bytes) 2
*
* Later versions encode the DEK like this:
*
* 0 2 RND(n bytes) 0 A DEK(k bytes) CSUM(2 bytes)
*
2002-06-29 15:46:34 +02:00
* (mpi_get_buffer already removed the leading zero).
1998-04-07 20:16:10 +02:00
*
1997-11-18 15:06:00 +01:00
* RND are non-zero randow bytes.
1997-11-24 12:04:11 +01:00
* A is the cipher algorithm
1997-11-18 15:06:00 +01:00
* DEK is the encryption key (session key) with length k
* CSUM
*/
if( DBG_CIPHER )
1998-02-13 21:58:50 +01:00
log_hexdump("DEK frame:", frame, nframe );
1998-04-07 20:16:10 +02:00
n=0;
if (!card)
{
if( n + 7 > nframe )
{ rc = G10ERR_WRONG_SECKEY; goto leave; }
if( frame[n] == 1 && frame[nframe-1] == 2 ) {
log_info(_("old encoding of the DEK is not supported\n"));
rc = G10ERR_CIPHER_ALGO;
goto leave;
}
if( frame[n] != 2 ) /* somethink is wrong */
{ rc = G10ERR_WRONG_SECKEY; goto leave; }
for(n++; n < nframe && frame[n]; n++ ) /* skip the random bytes */
;
n++; /* and the zero byte */
}
1998-02-13 21:58:50 +01:00
if( n + 4 > nframe )
2002-06-29 15:46:34 +02:00
{ rc = G10ERR_WRONG_SECKEY; goto leave; }
1998-02-13 21:58:50 +01:00
dek->keylen = nframe - (n+1) - 2;
dek->algo = frame[n++];
2002-06-29 15:46:34 +02:00
rc = check_cipher_algo( dek->algo );
1998-06-13 08:59:14 +02:00
if( rc ) {
2002-06-29 15:46:34 +02:00
if( !opt.quiet && rc == G10ERR_CIPHER_ALGO ) {
log_info(_("cipher algorithm %d%s is unknown or disabled\n"),
dek->algo, dek->algo == CIPHER_ALGO_IDEA? " (IDEA)":"");
}
1998-02-13 21:58:50 +01:00
dek->algo = 0;
1997-11-18 15:06:00 +01:00
goto leave;
}
2002-06-29 15:46:34 +02:00
if( (dek->keylen*8) != cipher_get_keylen( dek->algo ) ) {
rc = G10ERR_WRONG_SECKEY;
1998-06-13 08:59:14 +02:00
goto leave;
}
1997-11-18 15:06:00 +01:00
/* copy the key to DEK and compare the checksum */
1998-02-13 21:58:50 +01:00
csum = frame[nframe-2] << 8;
csum |= frame[nframe-1];
memcpy( dek->key, frame+n, dek->keylen );
for( csum2=0, n=0; n < dek->keylen; n++ )
csum2 += dek->key[n];
1997-11-18 15:06:00 +01:00
if( csum != csum2 ) {
2002-06-29 15:46:34 +02:00
rc = G10ERR_WRONG_SECKEY;
1997-11-18 15:06:00 +01:00
goto leave;
}
if( DBG_CIPHER )
1997-11-18 15:06:00 +01:00
log_hexdump("DEK is:", dek->key, dek->keylen );
/* check that the algo is in the preferences and whether it has expired */
1998-08-05 18:51:59 +02:00
{
2002-06-29 15:46:34 +02:00
PKT_public_key *pk = NULL;
KBNODE pkb = get_pubkeyblock (keyid);
if( !pkb ) {
rc = -1;
log_error("oops: public key not found for preference check\n");
}
else if(pkb->pkt->pkt.public_key->selfsigversion > 3
&& dek->algo != CIPHER_ALGO_3DES
&& !opt.quiet
&& !is_algo_in_prefs( pkb, PREFTYPE_SYM, dek->algo ))
log_info(_("WARNING: cipher algorithm %s not found in recipient"
" preferences\n"),cipher_algo_to_string(dek->algo));
2002-06-29 15:46:34 +02:00
if (!rc) {
KBNODE k;
2002-06-29 15:46:34 +02:00
for (k=pkb; k; k = k->next) {
if (k->pkt->pkttype == PKT_PUBLIC_KEY
2002-06-29 15:46:34 +02:00
|| k->pkt->pkttype == PKT_PUBLIC_SUBKEY){
u32 aki[2];
keyid_from_pk(k->pkt->pkt.public_key, aki);
if (aki[0]==keyid[0] && aki[1]==keyid[1]) {
pk = k->pkt->pkt.public_key;
break;
}
}
}
if (!pk)
BUG ();
if ( pk->expiredate && pk->expiredate <= make_timestamp() ) {
log_info(_("NOTE: secret key %s expired at %s\n"),
keystr(keyid), asctimestamp( pk->expiredate) );
2002-06-29 15:46:34 +02:00
}
}
if ( pk && pk->is_revoked ) {
2002-06-29 15:46:34 +02:00
log_info( _("NOTE: key has been revoked") );
putc( '\n', log_stream() );
show_revocation_reason( pk, 1 );
}
release_kbnode (pkb);
1998-08-05 18:51:59 +02:00
rc = 0;
}
1997-11-18 15:06:00 +01:00
1997-11-18 15:06:00 +01:00
leave:
2002-06-29 15:46:34 +02:00
mpi_free(plain_dek);
2005-07-27 20:10:56 +02:00
xfree(frame);
1997-11-18 15:06:00 +01:00
return rc;
}
2002-06-29 15:46:34 +02:00
/****************
* Get the session key from the given string.
* String is supposed to be formatted as this:
* <algo-id>:<even-number-of-hex-digits>
*/
int
get_override_session_key( DEK *dek, const char *string )
{
const char *s;
int i;
if ( !string )
return G10ERR_BAD_KEY;
dek->algo = atoi(string);
if ( dek->algo < 1 )
return G10ERR_BAD_KEY;
if ( !(s = strchr ( string, ':' )) )
return G10ERR_BAD_KEY;
s++;
for(i=0; i < DIM(dek->key) && *s; i++, s +=2 ) {
int c = hextobyte ( s );
if (c == -1)
return G10ERR_BAD_KEY;
dek->key[i] = c;
}
if ( *s )
return G10ERR_BAD_KEY;
dek->keylen = i;
return 0;
}