2003-08-05 17:11:04 +00:00
|
|
|
|
/* keybox-blob.c - KBX Blob handling
|
2008-03-13 08:46:08 +00:00
|
|
|
|
* Copyright (C) 2000, 2001, 2002, 2003, 2008 Free Software Foundation, Inc.
|
2003-08-05 17:11:04 +00:00
|
|
|
|
*
|
|
|
|
|
* This file is part of GnuPG.
|
|
|
|
|
*
|
|
|
|
|
* GnuPG is free software; you can redistribute it and/or modify
|
|
|
|
|
* it under the terms of the GNU General Public License as published by
|
2007-07-04 19:49:40 +00:00
|
|
|
|
* the Free Software Foundation; either version 3 of the License, or
|
2003-08-05 17:11:04 +00:00
|
|
|
|
* (at your option) any later version.
|
|
|
|
|
*
|
|
|
|
|
* GnuPG is distributed in the hope that it will be useful,
|
|
|
|
|
* 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
|
2016-11-05 12:02:19 +01:00
|
|
|
|
* along with this program; if not, see <https://www.gnu.org/licenses/>.
|
2003-08-05 17:11:04 +00:00
|
|
|
|
*/
|
|
|
|
|
|
2013-01-08 17:40:56 +01:00
|
|
|
|
/*
|
|
|
|
|
* The keybox data format
|
|
|
|
|
|
|
|
|
|
The KeyBox uses an augmented OpenPGP/X.509 key format. This makes
|
|
|
|
|
random access to a keyblock/certificate easier and also gives the
|
|
|
|
|
opportunity to store additional information (e.g. the fingerprint)
|
|
|
|
|
along with the key. All integers are stored in network byte order,
|
|
|
|
|
offsets are counted from the beginning of the Blob.
|
|
|
|
|
|
|
|
|
|
** Overview of blob types
|
|
|
|
|
|
|
|
|
|
| Byte 4 | Blob type |
|
|
|
|
|
|--------+--------------|
|
|
|
|
|
| 0 | Empty blob |
|
|
|
|
|
| 1 | First blob |
|
|
|
|
|
| 2 | OpenPGP blob |
|
|
|
|
|
| 3 | X.509 blob |
|
|
|
|
|
|
|
|
|
|
** The First blob
|
|
|
|
|
|
|
|
|
|
The first blob of a plain KBX file has a special format:
|
|
|
|
|
|
|
|
|
|
- u32 Length of this blob
|
|
|
|
|
- byte Blob type (1)
|
|
|
|
|
- byte Version number (1)
|
2014-10-09 19:10:32 +02:00
|
|
|
|
- u16 Header flags
|
|
|
|
|
bit 0 - RFU
|
|
|
|
|
bit 1 - Is being or has been used for OpenPGP blobs
|
2013-01-08 17:40:56 +01:00
|
|
|
|
- b4 Magic 'KBXf'
|
|
|
|
|
- u32 RFU
|
|
|
|
|
- u32 file_created_at
|
|
|
|
|
- u32 last_maintenance_run
|
|
|
|
|
- u32 RFU
|
|
|
|
|
- u32 RFU
|
|
|
|
|
|
|
|
|
|
** The OpenPGP and X.509 blobs
|
|
|
|
|
|
2015-11-16 12:41:46 +01:00
|
|
|
|
The OpenPGP and X.509 blobs are very similar, things which are
|
2013-01-08 17:40:56 +01:00
|
|
|
|
X.509 specific are noted like [X.509: xxx]
|
|
|
|
|
|
|
|
|
|
- u32 Length of this blob (including these 4 bytes)
|
|
|
|
|
- byte Blob type
|
|
|
|
|
2 = OpenPGP
|
|
|
|
|
3 = X509
|
|
|
|
|
- byte Version number of this blob type
|
2019-03-14 08:54:59 +01:00
|
|
|
|
1 = Blob with 20 byte fingerprints
|
|
|
|
|
2 = Blob with 32 byte fingerprints and no keyids.
|
2013-01-08 17:40:56 +01:00
|
|
|
|
- u16 Blob flags
|
|
|
|
|
bit 0 = contains secret key material (not used)
|
2017-04-28 10:06:33 +09:00
|
|
|
|
bit 1 = ephemeral blob (e.g. used while querying external resources)
|
2013-01-08 17:40:56 +01:00
|
|
|
|
- u32 Offset to the OpenPGP keyblock or the X.509 DER encoded
|
|
|
|
|
certificate
|
|
|
|
|
- u32 The length of the keyblock or certificate
|
|
|
|
|
- u16 [NKEYS] Number of keys (at least 1!) [X509: always 1]
|
2019-03-14 08:54:59 +01:00
|
|
|
|
- u16 Size of the key information structure (at least 28 or 56).
|
2013-01-08 17:40:56 +01:00
|
|
|
|
- NKEYS times:
|
2019-03-14 08:54:59 +01:00
|
|
|
|
Version 1 blob:
|
2013-01-08 17:40:56 +01:00
|
|
|
|
- b20 The fingerprint of the key.
|
|
|
|
|
Fingerprints are always 20 bytes, MD5 left padded with zeroes.
|
|
|
|
|
- u32 Offset to the n-th key's keyID (a keyID is always 8 byte)
|
|
|
|
|
or 0 if not known which is the case only for X.509.
|
2019-03-14 08:54:59 +01:00
|
|
|
|
Note that this separate keyid is not anymore used by
|
|
|
|
|
gnupg since the support for v3 keys has been removed.
|
|
|
|
|
We create this field anyway for backward compatibility with
|
|
|
|
|
old EOL-ed versions. Eventually we will completely move
|
|
|
|
|
to the version 2 blob format.
|
2013-01-08 17:40:56 +01:00
|
|
|
|
- u16 Key flags
|
|
|
|
|
bit 0 = qualified signature (not yet implemented}
|
|
|
|
|
- u16 RFU
|
|
|
|
|
- bN Optional filler up to the specified length of this
|
|
|
|
|
structure.
|
2019-03-14 08:54:59 +01:00
|
|
|
|
Version 2 blob:
|
|
|
|
|
- b32 The fingerprint of the key. This fingerprint is
|
|
|
|
|
either 20 or 32 bytes. A 20 byte fingerprint is
|
|
|
|
|
right filled with zeroes.
|
|
|
|
|
- u16 Key flags
|
|
|
|
|
bit 0 = qualified signature (not yet implemented}
|
|
|
|
|
bit 7 = 32 byte fingerprint in use.
|
|
|
|
|
- u16 RFU
|
|
|
|
|
- b20 keygrip
|
|
|
|
|
- bN Optional filler up to the specified length of this
|
|
|
|
|
structure.
|
2013-01-08 17:40:56 +01:00
|
|
|
|
- u16 Size of the serial number (may be zero)
|
2019-03-14 08:54:59 +01:00
|
|
|
|
- bN The serial number. N as given above.
|
2013-01-08 17:40:56 +01:00
|
|
|
|
- u16 Number of user IDs
|
|
|
|
|
- u16 [NUIDS] Size of user ID information structure
|
|
|
|
|
- NUIDS times:
|
|
|
|
|
|
|
|
|
|
For X509, the first user ID is the Issuer, the second the
|
|
|
|
|
Subject and the others are subjectAltNames. For OpenPGP we only
|
|
|
|
|
store the information from UserID packets here.
|
|
|
|
|
|
|
|
|
|
- u32 Blob offset to the n-th user ID
|
|
|
|
|
- u32 Length of this user ID.
|
|
|
|
|
- u16 User ID flags.
|
|
|
|
|
(not yet used)
|
|
|
|
|
- byte Validity
|
|
|
|
|
- byte RFU
|
|
|
|
|
|
|
|
|
|
- u16 [NSIGS] Number of signatures
|
|
|
|
|
- u16 Size of signature information (4)
|
|
|
|
|
- NSIGS times:
|
2017-03-30 15:18:45 +02:00
|
|
|
|
- u32 Expiration time of signature with some special values.
|
|
|
|
|
Since version 2.1.20 these special valuesare not anymore
|
|
|
|
|
used for OpenPGP:
|
2013-01-08 17:40:56 +01:00
|
|
|
|
- 0x00000000 = not checked
|
|
|
|
|
- 0x00000001 = missing key
|
|
|
|
|
- 0x00000002 = bad signature
|
|
|
|
|
- 0x10000000 = valid and expires at some date in 1978.
|
|
|
|
|
- 0xffffffff = valid and does not expire
|
|
|
|
|
- u8 Assigned ownertrust [X509: not used]
|
|
|
|
|
- u8 All_Validity
|
|
|
|
|
OpenPGP: See ../g10/trustdb/TRUST_* [not yet used]
|
|
|
|
|
X509: Bit 4 set := key has been revoked.
|
|
|
|
|
Note that this value matches TRUST_FLAG_REVOKED
|
|
|
|
|
- u16 RFU
|
|
|
|
|
- u32 Recheck_after
|
2019-06-23 20:21:02 -04:00
|
|
|
|
- u32 Latest timestamp in the keyblock (useful for KS synchronization?)
|
2013-01-08 17:40:56 +01:00
|
|
|
|
- u32 Blob created at
|
|
|
|
|
- u32 [NRES] Size of reserved space (not including this field)
|
|
|
|
|
- bN Reserved space of size NRES for future use.
|
|
|
|
|
- bN Arbitrary space for example used to store data which is not
|
|
|
|
|
part of the keyblock or certificate. For example the v3 key
|
|
|
|
|
IDs go here.
|
2013-11-15 15:54:31 +01:00
|
|
|
|
- bN Space for the keyblock or certificate.
|
|
|
|
|
- bN RFU. This is the remaining space after keyblock and before
|
2017-02-20 16:19:50 -05:00
|
|
|
|
the checksum. It is not covered by the checksum.
|
2019-06-23 20:21:02 -04:00
|
|
|
|
- b20 SHA-1 checksum (useful for KS synchronization?)
|
2013-01-08 17:40:56 +01:00
|
|
|
|
Note, that KBX versions before GnuPG 2.1 used an MD5
|
|
|
|
|
checksum. However it was only created but never checked.
|
|
|
|
|
Thus we do not expect problems if we switch to SHA-1. If
|
|
|
|
|
the checksum fails and the first 4 bytes are zero, we can
|
|
|
|
|
try again with MD5. SHA-1 has the advantage that it is
|
|
|
|
|
faster on CPUs with dedicated SHA-1 support.
|
2003-08-05 17:11:04 +00:00
|
|
|
|
|
|
|
|
|
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
#include <config.h>
|
|
|
|
|
#include <stdio.h>
|
|
|
|
|
#include <stdlib.h>
|
|
|
|
|
#include <string.h>
|
|
|
|
|
#include <errno.h>
|
|
|
|
|
#include <assert.h>
|
2003-11-12 15:17:44 +00:00
|
|
|
|
#include <time.h>
|
2003-08-05 17:11:04 +00:00
|
|
|
|
|
|
|
|
|
#include "keybox-defs.h"
|
|
|
|
|
#include <gcrypt.h>
|
|
|
|
|
|
|
|
|
|
#ifdef KEYBOX_WITH_X509
|
|
|
|
|
#include <ksba.h>
|
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
|
2010-04-21 16:26:17 +00:00
|
|
|
|
#include "../common/gettime.h"
|
|
|
|
|
|
2003-08-05 17:11:04 +00:00
|
|
|
|
|
|
|
|
|
/* special values of the signature status */
|
|
|
|
|
#define SF_NONE(a) ( !(a) )
|
|
|
|
|
#define SF_NOKEY(a) ((a) & (1<<0))
|
|
|
|
|
#define SF_BAD(a) ((a) & (1<<1))
|
|
|
|
|
#define SF_VALID(a) ((a) & (1<<29))
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
struct membuf {
|
|
|
|
|
size_t len;
|
|
|
|
|
size_t size;
|
|
|
|
|
char *buf;
|
|
|
|
|
int out_of_core;
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
struct keyboxblob_key {
|
2019-03-14 08:54:59 +01:00
|
|
|
|
char fpr[32];
|
2003-08-05 17:11:04 +00:00
|
|
|
|
u32 off_kid;
|
|
|
|
|
ulong off_kid_addr;
|
|
|
|
|
u16 flags;
|
2019-03-14 08:54:59 +01:00
|
|
|
|
u16 fprlen; /* Either 20 or 32 */
|
2003-08-05 17:11:04 +00:00
|
|
|
|
};
|
|
|
|
|
struct keyboxblob_uid {
|
2012-12-28 14:03:16 +01:00
|
|
|
|
u32 off;
|
2003-08-05 17:11:04 +00:00
|
|
|
|
ulong off_addr;
|
|
|
|
|
char *name; /* used only with x509 */
|
|
|
|
|
u32 len;
|
|
|
|
|
u16 flags;
|
|
|
|
|
byte validity;
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
struct keyid_list {
|
|
|
|
|
struct keyid_list *next;
|
|
|
|
|
int seqno;
|
|
|
|
|
byte kid[8];
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
struct fixup_list {
|
|
|
|
|
struct fixup_list *next;
|
|
|
|
|
u32 off;
|
|
|
|
|
u32 val;
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
struct keyboxblob {
|
|
|
|
|
byte *blob;
|
|
|
|
|
size_t bloblen;
|
|
|
|
|
off_t fileoffset;
|
2011-02-04 12:57:53 +01:00
|
|
|
|
|
2003-08-05 17:11:04 +00:00
|
|
|
|
/* stuff used only by keybox_create_blob */
|
|
|
|
|
unsigned char *serialbuf;
|
|
|
|
|
const unsigned char *serial;
|
|
|
|
|
size_t seriallen;
|
|
|
|
|
int nkeys;
|
|
|
|
|
struct keyboxblob_key *keys;
|
|
|
|
|
int nuids;
|
|
|
|
|
struct keyboxblob_uid *uids;
|
|
|
|
|
int nsigs;
|
|
|
|
|
u32 *sigs;
|
|
|
|
|
struct fixup_list *fixups;
|
|
|
|
|
int fixup_out_of_core;
|
2011-02-04 12:57:53 +01:00
|
|
|
|
|
2003-08-05 17:11:04 +00:00
|
|
|
|
struct keyid_list *temp_kids;
|
|
|
|
|
struct membuf bufbuf; /* temporary store for the blob */
|
2011-02-04 12:57:53 +01:00
|
|
|
|
struct membuf *buf;
|
2003-08-05 17:11:04 +00:00
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
2017-04-28 10:06:33 +09:00
|
|
|
|
/* A simple implementation of a dynamic buffer. Use init_membuf() to
|
2003-08-05 17:11:04 +00:00
|
|
|
|
create a buffer, put_membuf to append bytes and get_membuf to
|
|
|
|
|
release and return the buffer. Allocation errors are detected but
|
|
|
|
|
only returned at the final get_membuf(), this helps not to clutter
|
|
|
|
|
the code with out of core checks. */
|
|
|
|
|
|
|
|
|
|
static void
|
|
|
|
|
init_membuf (struct membuf *mb, int initiallen)
|
|
|
|
|
{
|
|
|
|
|
mb->len = 0;
|
|
|
|
|
mb->size = initiallen;
|
|
|
|
|
mb->out_of_core = 0;
|
|
|
|
|
mb->buf = xtrymalloc (initiallen);
|
|
|
|
|
if (!mb->buf)
|
|
|
|
|
mb->out_of_core = 1;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static void
|
|
|
|
|
put_membuf (struct membuf *mb, const void *buf, size_t len)
|
|
|
|
|
{
|
|
|
|
|
if (mb->out_of_core)
|
|
|
|
|
return;
|
|
|
|
|
|
|
|
|
|
if (mb->len + len >= mb->size)
|
|
|
|
|
{
|
|
|
|
|
char *p;
|
2011-02-04 12:57:53 +01:00
|
|
|
|
|
2003-08-05 17:11:04 +00:00
|
|
|
|
mb->size += len + 1024;
|
|
|
|
|
p = xtryrealloc (mb->buf, mb->size);
|
|
|
|
|
if (!p)
|
|
|
|
|
{
|
|
|
|
|
mb->out_of_core = 1;
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
mb->buf = p;
|
|
|
|
|
}
|
2013-01-08 18:15:49 +01:00
|
|
|
|
if (buf)
|
|
|
|
|
memcpy (mb->buf + mb->len, buf, len);
|
|
|
|
|
else
|
|
|
|
|
memset (mb->buf + mb->len, 0, len);
|
2003-08-05 17:11:04 +00:00
|
|
|
|
mb->len += len;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static void *
|
|
|
|
|
get_membuf (struct membuf *mb, size_t *len)
|
|
|
|
|
{
|
|
|
|
|
char *p;
|
|
|
|
|
|
|
|
|
|
if (mb->out_of_core)
|
|
|
|
|
{
|
|
|
|
|
xfree (mb->buf);
|
|
|
|
|
mb->buf = NULL;
|
|
|
|
|
return NULL;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
p = mb->buf;
|
|
|
|
|
*len = mb->len;
|
|
|
|
|
mb->buf = NULL;
|
|
|
|
|
mb->out_of_core = 1; /* don't allow a reuse */
|
|
|
|
|
return p;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
static void
|
|
|
|
|
put8 (struct membuf *mb, byte a )
|
|
|
|
|
{
|
|
|
|
|
put_membuf (mb, &a, 1);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static void
|
|
|
|
|
put16 (struct membuf *mb, u16 a )
|
|
|
|
|
{
|
|
|
|
|
unsigned char tmp[2];
|
|
|
|
|
tmp[0] = a>>8;
|
|
|
|
|
tmp[1] = a;
|
|
|
|
|
put_membuf (mb, tmp, 2);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static void
|
|
|
|
|
put32 (struct membuf *mb, u32 a )
|
|
|
|
|
{
|
|
|
|
|
unsigned char tmp[4];
|
|
|
|
|
tmp[0] = a>>24;
|
|
|
|
|
tmp[1] = a>>16;
|
|
|
|
|
tmp[2] = a>>8;
|
|
|
|
|
tmp[3] = a;
|
|
|
|
|
put_membuf (mb, tmp, 4);
|
|
|
|
|
}
|
|
|
|
|
|
2013-01-08 18:15:49 +01:00
|
|
|
|
|
2003-08-05 17:11:04 +00:00
|
|
|
|
|
|
|
|
|
/* Store a value in the fixup list */
|
|
|
|
|
static void
|
|
|
|
|
add_fixup (KEYBOXBLOB blob, u32 off, u32 val)
|
|
|
|
|
{
|
|
|
|
|
struct fixup_list *fl;
|
2011-02-04 12:57:53 +01:00
|
|
|
|
|
2003-08-05 17:11:04 +00:00
|
|
|
|
if (blob->fixup_out_of_core)
|
|
|
|
|
return;
|
|
|
|
|
|
|
|
|
|
fl = xtrycalloc(1, sizeof *fl);
|
|
|
|
|
if (!fl)
|
|
|
|
|
blob->fixup_out_of_core = 1;
|
2011-02-04 12:57:53 +01:00
|
|
|
|
else
|
2003-08-05 17:11:04 +00:00
|
|
|
|
{
|
|
|
|
|
fl->off = off;
|
|
|
|
|
fl->val = val;
|
|
|
|
|
fl->next = blob->fixups;
|
|
|
|
|
blob->fixups = fl;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/*
|
2011-02-04 12:57:53 +01:00
|
|
|
|
OpenPGP specific stuff
|
2003-08-05 17:11:04 +00:00
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
|
2012-12-28 14:03:16 +01:00
|
|
|
|
/* We must store the keyid at some place because we can't calculate
|
|
|
|
|
the offset yet. This is only used for v3 keyIDs. Function returns
|
|
|
|
|
an index value for later fixup or -1 for out of core. The value
|
|
|
|
|
must be a non-zero value. */
|
2003-08-05 17:11:04 +00:00
|
|
|
|
static int
|
2012-12-28 14:03:16 +01:00
|
|
|
|
pgp_temp_store_kid (KEYBOXBLOB blob, struct _keybox_openpgp_key_info *kinfo)
|
2003-08-05 17:11:04 +00:00
|
|
|
|
{
|
|
|
|
|
struct keyid_list *k, *r;
|
2011-02-04 12:57:53 +01:00
|
|
|
|
|
|
|
|
|
k = xtrymalloc (sizeof *k);
|
2003-08-05 17:11:04 +00:00
|
|
|
|
if (!k)
|
|
|
|
|
return -1;
|
2012-12-28 14:03:16 +01:00
|
|
|
|
memcpy (k->kid, kinfo->keyid, 8);
|
2003-08-05 17:11:04 +00:00
|
|
|
|
k->seqno = 0;
|
|
|
|
|
k->next = blob->temp_kids;
|
|
|
|
|
blob->temp_kids = k;
|
2011-02-04 12:57:53 +01:00
|
|
|
|
for (r=k; r; r = r->next)
|
2003-08-05 17:11:04 +00:00
|
|
|
|
k->seqno++;
|
2011-02-04 12:57:53 +01:00
|
|
|
|
|
2003-08-05 17:11:04 +00:00
|
|
|
|
return k->seqno;
|
|
|
|
|
}
|
|
|
|
|
|
2012-12-28 14:03:16 +01:00
|
|
|
|
|
|
|
|
|
/* Helper for pgp_create_key_part. */
|
|
|
|
|
static gpg_error_t
|
|
|
|
|
pgp_create_key_part_single (KEYBOXBLOB blob, int n,
|
|
|
|
|
struct _keybox_openpgp_key_info *kinfo)
|
2003-08-05 17:11:04 +00:00
|
|
|
|
{
|
|
|
|
|
size_t fprlen;
|
2012-12-28 14:03:16 +01:00
|
|
|
|
int off;
|
2003-08-05 17:11:04 +00:00
|
|
|
|
|
2012-12-28 14:03:16 +01:00
|
|
|
|
fprlen = kinfo->fprlen;
|
|
|
|
|
memcpy (blob->keys[n].fpr, kinfo->fpr, fprlen);
|
2019-03-14 08:54:59 +01:00
|
|
|
|
blob->keys[n].fprlen = fprlen;
|
|
|
|
|
if (fprlen < 20) /* v3 fpr - shift right and fill with zeroes. */
|
2003-08-05 17:11:04 +00:00
|
|
|
|
{
|
2012-12-28 14:03:16 +01:00
|
|
|
|
memmove (blob->keys[n].fpr + 20 - fprlen, blob->keys[n].fpr, fprlen);
|
|
|
|
|
memset (blob->keys[n].fpr, 0, 20 - fprlen);
|
|
|
|
|
off = pgp_temp_store_kid (blob, kinfo);
|
|
|
|
|
if (off == -1)
|
|
|
|
|
return gpg_error_from_syserror ();
|
|
|
|
|
blob->keys[n].off_kid = off;
|
2003-08-05 17:11:04 +00:00
|
|
|
|
}
|
2012-12-28 14:03:16 +01:00
|
|
|
|
else
|
|
|
|
|
blob->keys[n].off_kid = 0; /* Will be fixed up later */
|
|
|
|
|
blob->keys[n].flags = 0;
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
static gpg_error_t
|
|
|
|
|
pgp_create_key_part (KEYBOXBLOB blob, keybox_openpgp_info_t info)
|
|
|
|
|
{
|
|
|
|
|
gpg_error_t err;
|
|
|
|
|
int n = 0;
|
|
|
|
|
struct _keybox_openpgp_key_info *kinfo;
|
|
|
|
|
|
|
|
|
|
err = pgp_create_key_part_single (blob, n++, &info->primary);
|
|
|
|
|
if (err)
|
|
|
|
|
return err;
|
|
|
|
|
if (info->nsubkeys)
|
|
|
|
|
for (kinfo = &info->subkeys; kinfo; kinfo = kinfo->next)
|
|
|
|
|
if ((err=pgp_create_key_part_single (blob, n++, kinfo)))
|
|
|
|
|
return err;
|
|
|
|
|
|
2003-08-05 17:11:04 +00:00
|
|
|
|
assert (n == blob->nkeys);
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
2012-12-28 14:03:16 +01:00
|
|
|
|
|
|
|
|
|
static void
|
|
|
|
|
pgp_create_uid_part (KEYBOXBLOB blob, keybox_openpgp_info_t info)
|
2003-08-05 17:11:04 +00:00
|
|
|
|
{
|
2012-12-28 14:03:16 +01:00
|
|
|
|
int n = 0;
|
|
|
|
|
struct _keybox_openpgp_uid_info *u;
|
2003-08-05 17:11:04 +00:00
|
|
|
|
|
2012-12-28 14:03:16 +01:00
|
|
|
|
if (info->nuids)
|
2003-08-05 17:11:04 +00:00
|
|
|
|
{
|
2012-12-28 14:03:16 +01:00
|
|
|
|
for (u = &info->uids; u; u = u->next)
|
2003-08-05 17:11:04 +00:00
|
|
|
|
{
|
2012-12-28 14:03:16 +01:00
|
|
|
|
blob->uids[n].off = u->off;
|
2003-08-05 17:11:04 +00:00
|
|
|
|
blob->uids[n].len = u->len;
|
|
|
|
|
blob->uids[n].flags = 0;
|
|
|
|
|
blob->uids[n].validity = 0;
|
|
|
|
|
n++;
|
2012-12-28 14:03:16 +01:00
|
|
|
|
}
|
2003-08-05 17:11:04 +00:00
|
|
|
|
}
|
2012-12-28 14:03:16 +01:00
|
|
|
|
|
2003-08-05 17:11:04 +00:00
|
|
|
|
assert (n == blob->nuids);
|
|
|
|
|
}
|
|
|
|
|
|
2012-12-28 14:03:16 +01:00
|
|
|
|
|
|
|
|
|
static void
|
2012-12-28 17:17:56 +01:00
|
|
|
|
pgp_create_sig_part (KEYBOXBLOB blob, u32 *sigstatus)
|
2003-08-05 17:11:04 +00:00
|
|
|
|
{
|
|
|
|
|
int n;
|
2011-02-04 12:57:53 +01:00
|
|
|
|
|
2012-12-28 14:03:16 +01:00
|
|
|
|
for (n=0; n < blob->nsigs; n++)
|
2003-08-05 17:11:04 +00:00
|
|
|
|
{
|
2012-12-28 17:17:56 +01:00
|
|
|
|
blob->sigs[n] = sigstatus? sigstatus[n+1] : 0;
|
2003-08-05 17:11:04 +00:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2012-12-28 14:03:16 +01:00
|
|
|
|
|
2003-08-05 17:11:04 +00:00
|
|
|
|
static int
|
2012-12-28 14:03:16 +01:00
|
|
|
|
pgp_create_blob_keyblock (KEYBOXBLOB blob,
|
|
|
|
|
const unsigned char *image, size_t imagelen)
|
2003-08-05 17:11:04 +00:00
|
|
|
|
{
|
|
|
|
|
struct membuf *a = blob->buf;
|
|
|
|
|
int n;
|
|
|
|
|
u32 kbstart = a->len;
|
|
|
|
|
|
2012-12-28 14:03:16 +01:00
|
|
|
|
add_fixup (blob, 8, kbstart);
|
2003-08-05 17:11:04 +00:00
|
|
|
|
|
2012-12-28 14:03:16 +01:00
|
|
|
|
for (n = 0; n < blob->nuids; n++)
|
|
|
|
|
add_fixup (blob, blob->uids[n].off_addr, kbstart + blob->uids[n].off);
|
|
|
|
|
|
|
|
|
|
put_membuf (a, image, imagelen);
|
2003-08-05 17:11:04 +00:00
|
|
|
|
|
2012-12-28 14:03:16 +01:00
|
|
|
|
add_fixup (blob, 12, a->len - kbstart);
|
2003-08-05 17:11:04 +00:00
|
|
|
|
return 0;
|
|
|
|
|
}
|
2011-02-04 12:57:53 +01:00
|
|
|
|
|
2003-08-05 17:11:04 +00:00
|
|
|
|
|
|
|
|
|
|
|
|
|
|
#ifdef KEYBOX_WITH_X509
|
2011-02-04 12:57:53 +01:00
|
|
|
|
/*
|
2003-08-05 17:11:04 +00:00
|
|
|
|
X.509 specific stuff
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
/* Write the raw certificate out */
|
|
|
|
|
static int
|
2003-12-17 12:27:21 +00:00
|
|
|
|
x509_create_blob_cert (KEYBOXBLOB blob, ksba_cert_t cert)
|
2003-08-05 17:11:04 +00:00
|
|
|
|
{
|
|
|
|
|
struct membuf *a = blob->buf;
|
|
|
|
|
const unsigned char *image;
|
|
|
|
|
size_t length;
|
|
|
|
|
u32 kbstart = a->len;
|
|
|
|
|
|
|
|
|
|
/* Store our offset for later fixup */
|
|
|
|
|
add_fixup (blob, 8, kbstart);
|
|
|
|
|
|
|
|
|
|
image = ksba_cert_get_image (cert, &length);
|
|
|
|
|
if (!image)
|
|
|
|
|
return gpg_error (GPG_ERR_GENERAL);
|
|
|
|
|
put_membuf (a, image, length);
|
|
|
|
|
|
|
|
|
|
add_fixup (blob, 12, a->len - kbstart);
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
2011-02-04 12:57:53 +01:00
|
|
|
|
|
2003-08-05 17:11:04 +00:00
|
|
|
|
#endif /*KEYBOX_WITH_X509*/
|
|
|
|
|
|
|
|
|
|
/* Write a stored keyID out to the buffer */
|
|
|
|
|
static void
|
|
|
|
|
write_stored_kid (KEYBOXBLOB blob, int seqno)
|
|
|
|
|
{
|
|
|
|
|
struct keyid_list *r;
|
2011-02-04 12:57:53 +01:00
|
|
|
|
|
|
|
|
|
for ( r = blob->temp_kids; r; r = r->next )
|
2003-08-05 17:11:04 +00:00
|
|
|
|
{
|
|
|
|
|
if (r->seqno == seqno )
|
|
|
|
|
{
|
|
|
|
|
put_membuf (blob->buf, r->kid, 8);
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
never_reached ();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Release a list of key IDs */
|
|
|
|
|
static void
|
|
|
|
|
release_kid_list (struct keyid_list *kl)
|
|
|
|
|
{
|
|
|
|
|
struct keyid_list *r, *r2;
|
2011-02-04 12:57:53 +01:00
|
|
|
|
|
|
|
|
|
for ( r = kl; r; r = r2 )
|
2003-08-05 17:11:04 +00:00
|
|
|
|
{
|
|
|
|
|
r2 = r->next;
|
|
|
|
|
xfree (r);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2019-03-14 08:54:59 +01:00
|
|
|
|
/* Create a new blob header. If WANT_FPR32 is set a version 2 blob is
|
|
|
|
|
* created. */
|
2003-08-05 17:11:04 +00:00
|
|
|
|
static int
|
2019-03-14 08:54:59 +01:00
|
|
|
|
create_blob_header (KEYBOXBLOB blob, int blobtype, int as_ephemeral,
|
|
|
|
|
int want_fpr32)
|
2003-08-05 17:11:04 +00:00
|
|
|
|
{
|
|
|
|
|
struct membuf *a = blob->buf;
|
|
|
|
|
int i;
|
|
|
|
|
|
|
|
|
|
put32 ( a, 0 ); /* blob length, needs fixup */
|
2011-02-04 12:57:53 +01:00
|
|
|
|
put8 ( a, blobtype);
|
2019-03-14 08:54:59 +01:00
|
|
|
|
put8 ( a, want_fpr32? 2:1 ); /* blob type version */
|
2003-08-05 17:11:04 +00:00
|
|
|
|
put16 ( a, as_ephemeral? 2:0 ); /* blob flags */
|
|
|
|
|
|
|
|
|
|
put32 ( a, 0 ); /* offset to the raw data, needs fixup */
|
|
|
|
|
put32 ( a, 0 ); /* length of the raw data, needs fixup */
|
|
|
|
|
|
|
|
|
|
put16 ( a, blob->nkeys );
|
2019-03-14 08:54:59 +01:00
|
|
|
|
if (want_fpr32)
|
|
|
|
|
put16 ( a, 32 + 2 + 2 + 20); /* size of key info */
|
|
|
|
|
else
|
|
|
|
|
put16 ( a, 20 + 4 + 2 + 2 ); /* size of key info */
|
2003-08-05 17:11:04 +00:00
|
|
|
|
for ( i=0; i < blob->nkeys; i++ )
|
|
|
|
|
{
|
2019-03-14 08:54:59 +01:00
|
|
|
|
if (want_fpr32)
|
|
|
|
|
{
|
|
|
|
|
put_membuf (a, blob->keys[i].fpr, blob->keys[i].fprlen);
|
|
|
|
|
blob->keys[i].off_kid_addr = a->len;
|
|
|
|
|
if (blob->keys[i].fprlen == 32)
|
|
|
|
|
put16 ( a, (blob->keys[i].flags | 0x80));
|
|
|
|
|
else
|
|
|
|
|
put16 ( a, blob->keys[i].flags);
|
|
|
|
|
put16 ( a, 0 ); /* reserved */
|
|
|
|
|
/* FIXME: Put the real grip here instead of the filler. */
|
|
|
|
|
put_membuf (a, "\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0\0", 20);
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
log_assert (blob->keys[i].fprlen <= 20);
|
|
|
|
|
put_membuf (a, blob->keys[i].fpr, 20);
|
|
|
|
|
blob->keys[i].off_kid_addr = a->len;
|
|
|
|
|
put32 ( a, 0 ); /* offset to keyid, fixed up later */
|
|
|
|
|
put16 ( a, blob->keys[i].flags );
|
|
|
|
|
put16 ( a, 0 ); /* reserved */
|
|
|
|
|
}
|
2003-08-05 17:11:04 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
put16 (a, blob->seriallen); /*fixme: check that it fits into 16 bits*/
|
|
|
|
|
if (blob->serial)
|
|
|
|
|
put_membuf (a, blob->serial, blob->seriallen);
|
|
|
|
|
|
|
|
|
|
put16 ( a, blob->nuids );
|
|
|
|
|
put16 ( a, 4 + 4 + 2 + 1 + 1 ); /* size of uid info */
|
|
|
|
|
for (i=0; i < blob->nuids; i++)
|
|
|
|
|
{
|
|
|
|
|
blob->uids[i].off_addr = a->len;
|
|
|
|
|
put32 ( a, 0 ); /* offset to userid, fixed up later */
|
|
|
|
|
put32 ( a, blob->uids[i].len );
|
|
|
|
|
put16 ( a, blob->uids[i].flags );
|
|
|
|
|
put8 ( a, 0 ); /* validity */
|
|
|
|
|
put8 ( a, 0 ); /* reserved */
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
put16 ( a, blob->nsigs );
|
|
|
|
|
put16 ( a, 4 ); /* size of sig info */
|
|
|
|
|
for (i=0; i < blob->nsigs; i++)
|
|
|
|
|
{
|
|
|
|
|
put32 ( a, blob->sigs[i]);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
put8 ( a, 0 ); /* assigned ownertrust */
|
|
|
|
|
put8 ( a, 0 ); /* validity of all user IDs */
|
|
|
|
|
put16 ( a, 0 ); /* reserved */
|
|
|
|
|
put32 ( a, 0 ); /* time of next recheck */
|
|
|
|
|
put32 ( a, 0 ); /* newest timestamp (none) */
|
|
|
|
|
put32 ( a, make_timestamp() ); /* creation time */
|
|
|
|
|
put32 ( a, 0 ); /* size of reserved space */
|
|
|
|
|
/* reserved space (which is currently of size 0) */
|
|
|
|
|
|
2017-02-20 16:19:50 -05:00
|
|
|
|
/* space where we write keyIDs and other stuff so that the
|
2003-08-05 17:11:04 +00:00
|
|
|
|
pointers can actually point to somewhere */
|
2019-03-14 08:54:59 +01:00
|
|
|
|
if (blobtype == KEYBOX_BLOBTYPE_PGP && !want_fpr32)
|
2003-08-05 17:11:04 +00:00
|
|
|
|
{
|
2019-03-14 08:54:59 +01:00
|
|
|
|
/* For version 1 blobs, we need to store the keyids for all v3
|
|
|
|
|
* keys because those key IDs are not part of the fingerprint.
|
|
|
|
|
* While we are doing that, we fixup all the keyID offsets. For
|
|
|
|
|
* version 2 blobs (which can't carry v3 keys) we compute the
|
|
|
|
|
* keyids in the fly because they are just stripped down
|
|
|
|
|
* fingerprints. */
|
2003-08-05 17:11:04 +00:00
|
|
|
|
for (i=0; i < blob->nkeys; i++ )
|
|
|
|
|
{
|
2011-02-04 12:57:53 +01:00
|
|
|
|
if (blob->keys[i].off_kid)
|
2003-08-05 17:11:04 +00:00
|
|
|
|
{ /* this is a v3 one */
|
|
|
|
|
add_fixup (blob, blob->keys[i].off_kid_addr, a->len);
|
|
|
|
|
write_stored_kid (blob, blob->keys[i].off_kid);
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{ /* the better v4 key IDs - just store an offset 8 bytes back */
|
|
|
|
|
add_fixup (blob, blob->keys[i].off_kid_addr,
|
2011-02-04 12:57:53 +01:00
|
|
|
|
blob->keys[i].off_kid_addr - 8);
|
2003-08-05 17:11:04 +00:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
2011-02-04 12:57:53 +01:00
|
|
|
|
|
2014-10-31 12:15:34 +01:00
|
|
|
|
if (blobtype == KEYBOX_BLOBTYPE_X509)
|
2003-08-05 17:11:04 +00:00
|
|
|
|
{
|
|
|
|
|
/* We don't want to point to ASN.1 encoded UserIDs (DNs) but to
|
2018-10-24 15:56:18 -04:00
|
|
|
|
the utf-8 string representation of them */
|
2003-08-05 17:11:04 +00:00
|
|
|
|
for (i=0; i < blob->nuids; i++ )
|
|
|
|
|
{
|
2011-02-04 12:57:53 +01:00
|
|
|
|
if (blob->uids[i].name)
|
2003-08-05 17:11:04 +00:00
|
|
|
|
{ /* this is a v3 one */
|
|
|
|
|
add_fixup (blob, blob->uids[i].off_addr, a->len);
|
|
|
|
|
put_membuf (blob->buf, blob->uids[i].name, blob->uids[i].len);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
static int
|
|
|
|
|
create_blob_trailer (KEYBOXBLOB blob)
|
|
|
|
|
{
|
2008-10-20 13:53:23 +00:00
|
|
|
|
(void)blob;
|
|
|
|
|
return 0;
|
2003-08-05 17:11:04 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
static int
|
|
|
|
|
create_blob_finish (KEYBOXBLOB blob)
|
|
|
|
|
{
|
|
|
|
|
struct membuf *a = blob->buf;
|
2005-06-16 08:12:03 +00:00
|
|
|
|
unsigned char *p;
|
|
|
|
|
unsigned char *pp;
|
2003-08-05 17:11:04 +00:00
|
|
|
|
size_t n;
|
|
|
|
|
|
2013-01-08 18:15:49 +01:00
|
|
|
|
/* Write a placeholder for the checksum */
|
|
|
|
|
put_membuf (a, NULL, 20);
|
2011-02-04 12:57:53 +01:00
|
|
|
|
|
2003-08-05 17:11:04 +00:00
|
|
|
|
/* get the memory area */
|
2005-06-16 08:12:03 +00:00
|
|
|
|
n = 0; /* (Just to avoid compiler warning.) */
|
2003-08-05 17:11:04 +00:00
|
|
|
|
p = get_membuf (a, &n);
|
|
|
|
|
if (!p)
|
|
|
|
|
return gpg_error (GPG_ERR_ENOMEM);
|
|
|
|
|
assert (n >= 20);
|
|
|
|
|
|
|
|
|
|
/* fixup the length */
|
|
|
|
|
add_fixup (blob, 0, n);
|
|
|
|
|
|
|
|
|
|
/* do the fixups */
|
|
|
|
|
if (blob->fixup_out_of_core)
|
2016-06-30 17:09:59 +02:00
|
|
|
|
{
|
|
|
|
|
xfree (p);
|
|
|
|
|
return gpg_error (GPG_ERR_ENOMEM);
|
|
|
|
|
}
|
2003-08-05 17:11:04 +00:00
|
|
|
|
|
|
|
|
|
{
|
2016-06-30 17:09:59 +02:00
|
|
|
|
struct fixup_list *fl, *next;
|
|
|
|
|
for (fl = blob->fixups; fl; fl = next)
|
2003-08-05 17:11:04 +00:00
|
|
|
|
{
|
|
|
|
|
assert (fl->off+4 <= n);
|
|
|
|
|
p[fl->off+0] = fl->val >> 24;
|
|
|
|
|
p[fl->off+1] = fl->val >> 16;
|
|
|
|
|
p[fl->off+2] = fl->val >> 8;
|
|
|
|
|
p[fl->off+3] = fl->val;
|
2016-06-30 17:09:59 +02:00
|
|
|
|
next = fl->next;
|
|
|
|
|
xfree (fl);
|
2003-08-05 17:11:04 +00:00
|
|
|
|
}
|
2016-06-30 17:09:59 +02:00
|
|
|
|
blob->fixups = NULL;
|
2003-08-05 17:11:04 +00:00
|
|
|
|
}
|
|
|
|
|
|
2013-01-08 18:15:49 +01:00
|
|
|
|
/* Compute and store the SHA-1 checksum. */
|
|
|
|
|
gcry_md_hash_buffer (GCRY_MD_SHA1, p + n - 20, p, n - 20);
|
2003-08-05 17:11:04 +00:00
|
|
|
|
|
|
|
|
|
pp = xtrymalloc (n);
|
|
|
|
|
if ( !pp )
|
2016-06-30 17:09:59 +02:00
|
|
|
|
{
|
|
|
|
|
xfree (p);
|
|
|
|
|
return gpg_error_from_syserror ();
|
|
|
|
|
}
|
2003-08-05 17:11:04 +00:00
|
|
|
|
memcpy (pp , p, n);
|
2016-06-30 17:09:59 +02:00
|
|
|
|
xfree (p);
|
2003-08-05 17:11:04 +00:00
|
|
|
|
blob->blob = pp;
|
|
|
|
|
blob->bloblen = n;
|
2011-02-04 12:57:53 +01:00
|
|
|
|
|
2003-08-05 17:11:04 +00:00
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
2012-12-28 17:17:56 +01:00
|
|
|
|
|
2003-08-05 17:11:04 +00:00
|
|
|
|
|
2012-12-28 14:03:16 +01:00
|
|
|
|
gpg_error_t
|
|
|
|
|
_keybox_create_openpgp_blob (KEYBOXBLOB *r_blob,
|
|
|
|
|
keybox_openpgp_info_t info,
|
|
|
|
|
const unsigned char *image,
|
|
|
|
|
size_t imagelen,
|
|
|
|
|
int as_ephemeral)
|
2003-08-05 17:11:04 +00:00
|
|
|
|
{
|
2012-12-28 14:03:16 +01:00
|
|
|
|
gpg_error_t err;
|
2003-08-05 17:11:04 +00:00
|
|
|
|
KEYBOXBLOB blob;
|
2019-03-14 08:54:59 +01:00
|
|
|
|
int need_fpr32 = 0;
|
2003-08-05 17:11:04 +00:00
|
|
|
|
|
|
|
|
|
*r_blob = NULL;
|
2012-12-28 14:03:16 +01:00
|
|
|
|
|
2019-03-14 08:54:59 +01:00
|
|
|
|
|
|
|
|
|
/* Check whether we need a blob with 32 bit fingerprints. We could
|
|
|
|
|
* use this always but for backward compatiblity we do this only for
|
|
|
|
|
* v5 keys. */
|
|
|
|
|
if (info->primary.version == 5)
|
|
|
|
|
need_fpr32 = 1;
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
struct _keybox_openpgp_key_info *kinfo;
|
|
|
|
|
for (kinfo = &info->subkeys; kinfo; kinfo = kinfo->next)
|
|
|
|
|
if (kinfo->version == 5)
|
|
|
|
|
{
|
|
|
|
|
need_fpr32 = 1;
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2003-08-05 17:11:04 +00:00
|
|
|
|
blob = xtrycalloc (1, sizeof *blob);
|
|
|
|
|
if (!blob)
|
2008-05-06 14:03:36 +00:00
|
|
|
|
return gpg_error_from_syserror ();
|
2003-08-05 17:11:04 +00:00
|
|
|
|
|
2012-12-28 14:03:16 +01:00
|
|
|
|
blob->nkeys = 1 + info->nsubkeys;
|
|
|
|
|
blob->keys = xtrycalloc (blob->nkeys, sizeof *blob->keys );
|
|
|
|
|
if (!blob->keys)
|
2003-08-05 17:11:04 +00:00
|
|
|
|
{
|
2012-12-28 14:03:16 +01:00
|
|
|
|
err = gpg_error_from_syserror ();
|
|
|
|
|
goto leave;
|
2003-08-05 17:11:04 +00:00
|
|
|
|
}
|
2014-08-18 12:55:29 +02:00
|
|
|
|
|
2012-12-28 14:03:16 +01:00
|
|
|
|
blob->nuids = info->nuids;
|
2014-08-18 12:55:29 +02:00
|
|
|
|
if (blob->nuids)
|
2012-12-28 14:03:16 +01:00
|
|
|
|
{
|
2014-08-18 12:55:29 +02:00
|
|
|
|
blob->uids = xtrycalloc (blob->nuids, sizeof *blob->uids );
|
|
|
|
|
if (!blob->uids)
|
|
|
|
|
{
|
|
|
|
|
err = gpg_error_from_syserror ();
|
|
|
|
|
goto leave;
|
|
|
|
|
}
|
2012-12-28 14:03:16 +01:00
|
|
|
|
}
|
2014-08-18 12:55:29 +02:00
|
|
|
|
|
2012-12-28 14:03:16 +01:00
|
|
|
|
blob->nsigs = info->nsigs;
|
2014-08-18 12:55:29 +02:00
|
|
|
|
if (blob->nsigs)
|
2003-08-05 17:11:04 +00:00
|
|
|
|
{
|
2014-08-18 12:55:29 +02:00
|
|
|
|
blob->sigs = xtrycalloc (blob->nsigs, sizeof *blob->sigs );
|
|
|
|
|
if (!blob->sigs)
|
|
|
|
|
{
|
|
|
|
|
err = gpg_error_from_syserror ();
|
|
|
|
|
goto leave;
|
|
|
|
|
}
|
2003-08-05 17:11:04 +00:00
|
|
|
|
}
|
|
|
|
|
|
2012-12-28 14:03:16 +01:00
|
|
|
|
err = pgp_create_key_part (blob, info);
|
|
|
|
|
if (err)
|
2003-08-05 17:11:04 +00:00
|
|
|
|
goto leave;
|
2012-12-28 14:03:16 +01:00
|
|
|
|
pgp_create_uid_part (blob, info);
|
2017-03-30 15:18:45 +02:00
|
|
|
|
pgp_create_sig_part (blob, NULL);
|
2011-02-04 12:57:53 +01:00
|
|
|
|
|
2003-08-05 17:11:04 +00:00
|
|
|
|
init_membuf (&blob->bufbuf, 1024);
|
|
|
|
|
blob->buf = &blob->bufbuf;
|
2019-03-14 08:54:59 +01:00
|
|
|
|
err = create_blob_header (blob, KEYBOX_BLOBTYPE_PGP,
|
|
|
|
|
as_ephemeral, need_fpr32);
|
2012-12-28 14:03:16 +01:00
|
|
|
|
if (err)
|
2003-08-05 17:11:04 +00:00
|
|
|
|
goto leave;
|
2012-12-28 14:03:16 +01:00
|
|
|
|
err = pgp_create_blob_keyblock (blob, image, imagelen);
|
|
|
|
|
if (err)
|
2003-08-05 17:11:04 +00:00
|
|
|
|
goto leave;
|
2012-12-28 14:03:16 +01:00
|
|
|
|
err = create_blob_trailer (blob);
|
|
|
|
|
if (err)
|
2003-08-05 17:11:04 +00:00
|
|
|
|
goto leave;
|
2012-12-28 14:03:16 +01:00
|
|
|
|
err = create_blob_finish (blob);
|
|
|
|
|
if (err)
|
2003-08-05 17:11:04 +00:00
|
|
|
|
goto leave;
|
|
|
|
|
|
|
|
|
|
leave:
|
|
|
|
|
release_kid_list (blob->temp_kids);
|
|
|
|
|
blob->temp_kids = NULL;
|
2012-12-28 14:03:16 +01:00
|
|
|
|
if (err)
|
|
|
|
|
_keybox_release_blob (blob);
|
2003-08-05 17:11:04 +00:00
|
|
|
|
else
|
2012-12-28 14:03:16 +01:00
|
|
|
|
*r_blob = blob;
|
|
|
|
|
return err;
|
2003-08-05 17:11:04 +00:00
|
|
|
|
}
|
2012-12-28 14:03:16 +01:00
|
|
|
|
|
2003-08-05 17:11:04 +00:00
|
|
|
|
|
|
|
|
|
#ifdef KEYBOX_WITH_X509
|
|
|
|
|
|
2008-03-13 08:46:08 +00:00
|
|
|
|
/* Return an allocated string with the email address extracted from a
|
|
|
|
|
DN. Note hat we use this code also in ../sm/keylist.c. */
|
2003-08-05 17:11:04 +00:00
|
|
|
|
static char *
|
|
|
|
|
x509_email_kludge (const char *name)
|
|
|
|
|
{
|
2008-03-13 08:46:08 +00:00
|
|
|
|
const char *p, *string;
|
2003-08-05 17:11:04 +00:00
|
|
|
|
unsigned char *buf;
|
|
|
|
|
int n;
|
|
|
|
|
|
2008-03-13 08:46:08 +00:00
|
|
|
|
string = name;
|
|
|
|
|
for (;;)
|
|
|
|
|
{
|
|
|
|
|
p = strstr (string, "1.2.840.113549.1.9.1=#");
|
|
|
|
|
if (!p)
|
|
|
|
|
return NULL;
|
|
|
|
|
if (p == name || (p > string+1 && p[-1] == ',' && p[-2] != '\\'))
|
|
|
|
|
{
|
|
|
|
|
name = p + 22;
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
string = p + 22;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2003-08-05 17:11:04 +00:00
|
|
|
|
/* This looks pretty much like an email address in the subject's DN
|
|
|
|
|
we use this to add an additional user ID entry. This way,
|
2008-03-13 08:46:08 +00:00
|
|
|
|
OpenSSL generated keys get a nicer and usable listing. */
|
2003-08-05 17:11:04 +00:00
|
|
|
|
for (n=0, p=name; hexdigitp (p) && hexdigitp (p+1); p +=2, n++)
|
|
|
|
|
;
|
2008-03-13 08:46:08 +00:00
|
|
|
|
if (!n)
|
2003-08-05 17:11:04 +00:00
|
|
|
|
return NULL;
|
|
|
|
|
buf = xtrymalloc (n+3);
|
|
|
|
|
if (!buf)
|
|
|
|
|
return NULL; /* oops, out of core */
|
|
|
|
|
*buf = '<';
|
2008-03-13 08:46:08 +00:00
|
|
|
|
for (n=1, p=name; hexdigitp (p); p +=2, n++)
|
2003-08-05 17:11:04 +00:00
|
|
|
|
buf[n] = xtoi_2 (p);
|
|
|
|
|
buf[n++] = '>';
|
|
|
|
|
buf[n] = 0;
|
2008-03-13 08:46:08 +00:00
|
|
|
|
return (char*)buf;
|
2003-08-05 17:11:04 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* Note: We should move calculation of the digest into libksba and
|
|
|
|
|
remove that parameter */
|
|
|
|
|
int
|
2003-12-17 12:27:21 +00:00
|
|
|
|
_keybox_create_x509_blob (KEYBOXBLOB *r_blob, ksba_cert_t cert,
|
2003-08-05 17:11:04 +00:00
|
|
|
|
unsigned char *sha1_digest, int as_ephemeral)
|
|
|
|
|
{
|
|
|
|
|
int i, rc = 0;
|
|
|
|
|
KEYBOXBLOB blob;
|
2005-06-16 08:12:03 +00:00
|
|
|
|
unsigned char *sn;
|
|
|
|
|
char *p;
|
|
|
|
|
char **names = NULL;
|
2003-08-05 17:11:04 +00:00
|
|
|
|
size_t max_names;
|
|
|
|
|
|
|
|
|
|
*r_blob = NULL;
|
|
|
|
|
blob = xtrycalloc (1, sizeof *blob);
|
|
|
|
|
if( !blob )
|
2008-05-06 14:03:36 +00:00
|
|
|
|
return gpg_error_from_syserror ();
|
2003-08-05 17:11:04 +00:00
|
|
|
|
|
2005-06-16 08:12:03 +00:00
|
|
|
|
sn = ksba_cert_get_serial (cert);
|
|
|
|
|
if (sn)
|
2003-08-05 17:11:04 +00:00
|
|
|
|
{
|
|
|
|
|
size_t n, len;
|
2005-06-16 08:12:03 +00:00
|
|
|
|
n = gcry_sexp_canon_len (sn, 0, NULL, NULL);
|
2003-08-05 17:11:04 +00:00
|
|
|
|
if (n < 2)
|
|
|
|
|
{
|
2005-06-16 08:12:03 +00:00
|
|
|
|
xfree (sn);
|
2003-08-05 17:11:04 +00:00
|
|
|
|
return gpg_error (GPG_ERR_GENERAL);
|
|
|
|
|
}
|
2005-06-16 08:12:03 +00:00
|
|
|
|
blob->serialbuf = sn;
|
|
|
|
|
sn++; n--; /* skip '(' */
|
|
|
|
|
for (len=0; n && *sn && *sn != ':' && digitp (sn); n--, sn++)
|
|
|
|
|
len = len*10 + atoi_1 (sn);
|
|
|
|
|
if (*sn != ':')
|
2003-08-05 17:11:04 +00:00
|
|
|
|
{
|
|
|
|
|
xfree (blob->serialbuf);
|
|
|
|
|
blob->serialbuf = NULL;
|
|
|
|
|
return gpg_error (GPG_ERR_GENERAL);
|
|
|
|
|
}
|
2005-06-16 08:12:03 +00:00
|
|
|
|
sn++;
|
|
|
|
|
blob->serial = sn;
|
2003-08-05 17:11:04 +00:00
|
|
|
|
blob->seriallen = len;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
blob->nkeys = 1;
|
|
|
|
|
|
|
|
|
|
/* create list of names */
|
|
|
|
|
blob->nuids = 0;
|
|
|
|
|
max_names = 100;
|
|
|
|
|
names = xtrymalloc (max_names * sizeof *names);
|
|
|
|
|
if (!names)
|
|
|
|
|
{
|
2008-05-06 14:03:36 +00:00
|
|
|
|
rc = gpg_error_from_syserror ();
|
2003-08-05 17:11:04 +00:00
|
|
|
|
goto leave;
|
|
|
|
|
}
|
2011-02-04 12:57:53 +01:00
|
|
|
|
|
2003-08-05 17:11:04 +00:00
|
|
|
|
p = ksba_cert_get_issuer (cert, 0);
|
|
|
|
|
if (!p)
|
|
|
|
|
{
|
|
|
|
|
rc = gpg_error (GPG_ERR_MISSING_VALUE);
|
|
|
|
|
goto leave;
|
|
|
|
|
}
|
|
|
|
|
names[blob->nuids++] = p;
|
|
|
|
|
for (i=0; (p = ksba_cert_get_subject (cert, i)); i++)
|
|
|
|
|
{
|
|
|
|
|
if (blob->nuids >= max_names)
|
|
|
|
|
{
|
2005-06-16 08:12:03 +00:00
|
|
|
|
char **tmp;
|
2011-02-04 12:57:53 +01:00
|
|
|
|
|
2003-08-05 17:11:04 +00:00
|
|
|
|
max_names += 100;
|
|
|
|
|
tmp = xtryrealloc (names, max_names * sizeof *names);
|
|
|
|
|
if (!tmp)
|
|
|
|
|
{
|
2008-05-06 14:03:36 +00:00
|
|
|
|
rc = gpg_error_from_syserror ();
|
2003-08-05 17:11:04 +00:00
|
|
|
|
goto leave;
|
|
|
|
|
}
|
2010-07-23 12:30:34 +00:00
|
|
|
|
names = tmp;
|
2003-08-05 17:11:04 +00:00
|
|
|
|
}
|
|
|
|
|
names[blob->nuids++] = p;
|
|
|
|
|
if (!i && (p=x509_email_kludge (p)))
|
|
|
|
|
names[blob->nuids++] = p; /* due to !i we don't need to check bounds*/
|
|
|
|
|
}
|
2011-02-04 12:57:53 +01:00
|
|
|
|
|
2003-08-05 17:11:04 +00:00
|
|
|
|
/* space for signature information */
|
2011-02-04 12:57:53 +01:00
|
|
|
|
blob->nsigs = 1;
|
2003-08-05 17:11:04 +00:00
|
|
|
|
|
|
|
|
|
blob->keys = xtrycalloc (blob->nkeys, sizeof *blob->keys );
|
|
|
|
|
blob->uids = xtrycalloc (blob->nuids, sizeof *blob->uids );
|
|
|
|
|
blob->sigs = xtrycalloc (blob->nsigs, sizeof *blob->sigs );
|
|
|
|
|
if (!blob->keys || !blob->uids || !blob->sigs)
|
|
|
|
|
{
|
|
|
|
|
rc = gpg_error (GPG_ERR_ENOMEM);
|
|
|
|
|
goto leave;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
memcpy (blob->keys[0].fpr, sha1_digest, 20);
|
|
|
|
|
blob->keys[0].off_kid = 0; /* We don't have keyids */
|
|
|
|
|
blob->keys[0].flags = 0;
|
|
|
|
|
|
|
|
|
|
/* issuer and subject names */
|
|
|
|
|
for (i=0; i < blob->nuids; i++)
|
|
|
|
|
{
|
|
|
|
|
blob->uids[i].name = names[i];
|
|
|
|
|
blob->uids[i].len = strlen(names[i]);
|
|
|
|
|
names[i] = NULL;
|
|
|
|
|
blob->uids[i].flags = 0;
|
|
|
|
|
blob->uids[i].validity = 0;
|
|
|
|
|
}
|
|
|
|
|
xfree (names);
|
|
|
|
|
names = NULL;
|
|
|
|
|
|
|
|
|
|
/* signatures */
|
|
|
|
|
blob->sigs[0] = 0; /* not yet checked */
|
|
|
|
|
|
|
|
|
|
/* Create a temporary buffer for further processing */
|
|
|
|
|
init_membuf (&blob->bufbuf, 1024);
|
|
|
|
|
blob->buf = &blob->bufbuf;
|
|
|
|
|
/* write out what we already have */
|
2019-03-14 08:54:59 +01:00
|
|
|
|
rc = create_blob_header (blob, KEYBOX_BLOBTYPE_X509, as_ephemeral, 0);
|
2003-08-05 17:11:04 +00:00
|
|
|
|
if (rc)
|
|
|
|
|
goto leave;
|
|
|
|
|
rc = x509_create_blob_cert (blob, cert);
|
|
|
|
|
if (rc)
|
|
|
|
|
goto leave;
|
|
|
|
|
rc = create_blob_trailer (blob);
|
|
|
|
|
if (rc)
|
|
|
|
|
goto leave;
|
|
|
|
|
rc = create_blob_finish ( blob );
|
|
|
|
|
if (rc)
|
|
|
|
|
goto leave;
|
|
|
|
|
|
2011-02-04 12:57:53 +01:00
|
|
|
|
|
2003-08-05 17:11:04 +00:00
|
|
|
|
leave:
|
|
|
|
|
release_kid_list (blob->temp_kids);
|
|
|
|
|
blob->temp_kids = NULL;
|
2014-04-15 16:40:48 +02:00
|
|
|
|
if (names)
|
2003-08-05 17:11:04 +00:00
|
|
|
|
{
|
|
|
|
|
for (i=0; i < blob->nuids; i++)
|
2011-02-04 12:57:53 +01:00
|
|
|
|
xfree (names[i]);
|
2014-04-15 16:40:48 +02:00
|
|
|
|
xfree (names);
|
2003-08-05 17:11:04 +00:00
|
|
|
|
}
|
|
|
|
|
if (rc)
|
|
|
|
|
{
|
|
|
|
|
_keybox_release_blob (blob);
|
|
|
|
|
*r_blob = NULL;
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
*r_blob = blob;
|
|
|
|
|
}
|
|
|
|
|
return rc;
|
|
|
|
|
}
|
|
|
|
|
#endif /*KEYBOX_WITH_X509*/
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
int
|
2005-06-16 08:12:03 +00:00
|
|
|
|
_keybox_new_blob (KEYBOXBLOB *r_blob,
|
|
|
|
|
unsigned char *image, size_t imagelen, off_t off)
|
2003-08-05 17:11:04 +00:00
|
|
|
|
{
|
|
|
|
|
KEYBOXBLOB blob;
|
2011-02-04 12:57:53 +01:00
|
|
|
|
|
2003-08-05 17:11:04 +00:00
|
|
|
|
*r_blob = NULL;
|
|
|
|
|
blob = xtrycalloc (1, sizeof *blob);
|
|
|
|
|
if (!blob)
|
2008-05-06 14:03:36 +00:00
|
|
|
|
return gpg_error_from_syserror ();
|
2003-08-05 17:11:04 +00:00
|
|
|
|
|
|
|
|
|
blob->blob = image;
|
|
|
|
|
blob->bloblen = imagelen;
|
|
|
|
|
blob->fileoffset = off;
|
|
|
|
|
*r_blob = blob;
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
2004-04-26 08:09:25 +00:00
|
|
|
|
|
2003-08-05 17:11:04 +00:00
|
|
|
|
void
|
|
|
|
|
_keybox_release_blob (KEYBOXBLOB blob)
|
|
|
|
|
{
|
|
|
|
|
int i;
|
|
|
|
|
if (!blob)
|
|
|
|
|
return;
|
2016-06-30 17:09:59 +02:00
|
|
|
|
if (blob->buf)
|
|
|
|
|
{
|
|
|
|
|
size_t len;
|
|
|
|
|
xfree (get_membuf (blob->buf, &len));
|
|
|
|
|
}
|
2003-08-05 17:11:04 +00:00
|
|
|
|
xfree (blob->keys );
|
|
|
|
|
xfree (blob->serialbuf);
|
|
|
|
|
for (i=0; i < blob->nuids; i++)
|
|
|
|
|
xfree (blob->uids[i].name);
|
|
|
|
|
xfree (blob->uids );
|
|
|
|
|
xfree (blob->sigs );
|
|
|
|
|
xfree (blob->blob );
|
|
|
|
|
xfree (blob );
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
2005-06-16 08:12:03 +00:00
|
|
|
|
const unsigned char *
|
2003-08-05 17:11:04 +00:00
|
|
|
|
_keybox_get_blob_image ( KEYBOXBLOB blob, size_t *n )
|
|
|
|
|
{
|
|
|
|
|
*n = blob->bloblen;
|
|
|
|
|
return blob->blob;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
off_t
|
|
|
|
|
_keybox_get_blob_fileoffset (KEYBOXBLOB blob)
|
|
|
|
|
{
|
|
|
|
|
return blob->fileoffset;
|
|
|
|
|
}
|
|
|
|
|
|
2004-04-26 08:09:25 +00:00
|
|
|
|
|
|
|
|
|
|
|
|
|
|
void
|
2014-10-09 19:10:32 +02:00
|
|
|
|
_keybox_update_header_blob (KEYBOXBLOB blob, int for_openpgp)
|
2004-04-26 08:09:25 +00:00
|
|
|
|
{
|
2014-10-31 12:15:34 +01:00
|
|
|
|
if (blob->bloblen >= 32 && blob->blob[4] == KEYBOX_BLOBTYPE_HEADER)
|
2004-04-26 08:09:25 +00:00
|
|
|
|
{
|
|
|
|
|
u32 val = make_timestamp ();
|
|
|
|
|
|
2019-07-18 13:59:52 +02:00
|
|
|
|
/* Update the last maintenance run timestamp. */
|
2004-04-26 08:09:25 +00:00
|
|
|
|
blob->blob[20] = (val >> 24);
|
|
|
|
|
blob->blob[20+1] = (val >> 16);
|
|
|
|
|
blob->blob[20+2] = (val >> 8);
|
|
|
|
|
blob->blob[20+3] = (val );
|
2014-10-09 19:10:32 +02:00
|
|
|
|
|
|
|
|
|
if (for_openpgp)
|
|
|
|
|
blob->blob[7] |= 0x02; /* OpenPGP data may be available. */
|
2004-04-26 08:09:25 +00:00
|
|
|
|
}
|
|
|
|
|
}
|