mirror of https://gitee.com/openkylin/gnupg2.git
341 lines
8.4 KiB
C
341 lines
8.4 KiB
C
/* g13tuple.c - Tuple handling
|
|
* Copyright (C) 2009 Free Software Foundation, Inc.
|
|
* Copyright (C) 2009, 2015, 2016 Werner Koch
|
|
*
|
|
* 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
|
|
* the Free Software Foundation; either version 3 of the License, or
|
|
* (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
|
|
* along with this program; if not, see <https://www.gnu.org/licenses/>.
|
|
*/
|
|
|
|
#include <config.h>
|
|
#include <stdio.h>
|
|
#include <stdlib.h>
|
|
#include <string.h>
|
|
#include <errno.h>
|
|
#include <assert.h>
|
|
|
|
#include "g13.h"
|
|
#include "g13tuple.h"
|
|
#include "keyblob.h" /* Required for dump_tupledesc. */
|
|
|
|
|
|
/* Definition of the tuple descriptor object. */
|
|
struct tupledesc_s
|
|
{
|
|
unsigned char *data; /* The tuple data. */
|
|
size_t datalen; /* The length of the data. */
|
|
size_t pos; /* The current position as used by next_tuple. */
|
|
int refcount; /* Number of references hold. */
|
|
};
|
|
|
|
|
|
|
|
/* Append the TAG and the VALUE to the MEMBUF. There is no error
|
|
checking here; this is instead done while getting the value back
|
|
from the membuf. */
|
|
void
|
|
append_tuple (membuf_t *membuf, int tag, const void *value, size_t length)
|
|
{
|
|
unsigned char buf[2];
|
|
|
|
assert (tag >= 0 && tag <= 0xffff);
|
|
assert (length <= 0xffff);
|
|
|
|
buf[0] = tag >> 8;
|
|
buf[1] = tag;
|
|
put_membuf (membuf, buf, 2);
|
|
buf[0] = length >> 8;
|
|
buf[1] = length;
|
|
put_membuf (membuf, buf, 2);
|
|
if (length)
|
|
put_membuf (membuf, value, length);
|
|
}
|
|
|
|
|
|
/* Append the unsigned integer VALUE under TAG to MEMBUF. We make
|
|
* sure that the most significant bit is always cleared to explicitly
|
|
* flag the value as unsigned. */
|
|
void
|
|
append_tuple_uint (membuf_t *membuf, int tag, unsigned long long value)
|
|
{
|
|
unsigned char buf[16];
|
|
unsigned char *p;
|
|
unsigned int len;
|
|
|
|
p = buf + sizeof buf;
|
|
len = 0;
|
|
do
|
|
{
|
|
if (p == buf)
|
|
BUG () ;
|
|
*--p = (value & 0xff);
|
|
value >>= 8;
|
|
len++;
|
|
}
|
|
while (value);
|
|
|
|
/* Prepend a zero byte if the first byte has its MSB set. */
|
|
if ((*p & 0x80))
|
|
{
|
|
if (p == buf)
|
|
BUG () ;
|
|
*--p = 0;
|
|
len++;
|
|
}
|
|
|
|
append_tuple (membuf, tag, p, len);
|
|
}
|
|
|
|
|
|
/* Create a tuple object by moving the ownership of (DATA,DATALEN) to
|
|
* a new object. Returns 0 on success and stores the new object at
|
|
* R_TUPLEHD. The return object must be released using
|
|
* destroy_tuples(). */
|
|
gpg_error_t
|
|
create_tupledesc (tupledesc_t *r_desc, void *data, size_t datalen)
|
|
{
|
|
if (datalen < 5 || memcmp (data, "\x00\x00\x00\x01\x01", 5))
|
|
return gpg_error (GPG_ERR_NOT_SUPPORTED);
|
|
|
|
*r_desc = xtrymalloc (sizeof **r_desc);
|
|
if (!*r_desc)
|
|
return gpg_error_from_syserror ();
|
|
(*r_desc)->data = data;
|
|
(*r_desc)->datalen = datalen;
|
|
(*r_desc)->pos = 0;
|
|
(*r_desc)->refcount = 1;
|
|
return 0;
|
|
}
|
|
|
|
/* Unref a tuple descriptor and if the refcount is down to 0 release
|
|
its allocated storage. */
|
|
void
|
|
destroy_tupledesc (tupledesc_t tupledesc)
|
|
{
|
|
if (!tupledesc)
|
|
return;
|
|
|
|
if (!--tupledesc->refcount)
|
|
{
|
|
xfree (tupledesc->data);
|
|
xfree (tupledesc);
|
|
}
|
|
}
|
|
|
|
|
|
tupledesc_t
|
|
ref_tupledesc (tupledesc_t tupledesc)
|
|
{
|
|
if (tupledesc)
|
|
tupledesc->refcount++;
|
|
return tupledesc;
|
|
}
|
|
|
|
|
|
/* Return a pointer to the memory used to store the tuples. This is
|
|
* the data originally provided to create_tupledesc. It is higly
|
|
* recommended that the callers uses ref_tupledesc before calling this
|
|
* function and unref_tupledesc when the return data will not anymore
|
|
* be used. */
|
|
const void *
|
|
get_tupledesc_data (tupledesc_t tupledesc, size_t *r_datalen)
|
|
{
|
|
*r_datalen = tupledesc->datalen;
|
|
return tupledesc->data;
|
|
}
|
|
|
|
/* Find the first tuple with tag TAG. On success return a pointer to
|
|
its value and store the length of the value at R_LENGTH. If no
|
|
tuple was found return NULL. For use by next_tuple, the last
|
|
position is stored in the descriptor. */
|
|
const void *
|
|
find_tuple (tupledesc_t tupledesc, unsigned int tag, size_t *r_length)
|
|
{
|
|
const unsigned char *s;
|
|
const unsigned char *s_end; /* Points right behind the data. */
|
|
unsigned int t;
|
|
size_t n;
|
|
|
|
s = tupledesc->data;
|
|
if (!s)
|
|
return NULL;
|
|
s_end = s + tupledesc->datalen;
|
|
while (s < s_end)
|
|
{
|
|
/* We use addresses for the overflow check to avoid undefined
|
|
behaviour. size_t should work with all flat memory models. */
|
|
if ((size_t)s+3 >= (size_t)s_end || (size_t)s + 3 < (size_t)s)
|
|
break;
|
|
t = s[0] << 8;
|
|
t |= s[1];
|
|
n = s[2] << 8;
|
|
n |= s[3];
|
|
s += 4;
|
|
if ((size_t)s + n > (size_t)s_end || (size_t)s + n < (size_t)s)
|
|
break;
|
|
if (t == tag)
|
|
{
|
|
tupledesc->pos = (s + n) - tupledesc->data;
|
|
*r_length = n;
|
|
return s;
|
|
}
|
|
s += n;
|
|
}
|
|
return NULL;
|
|
}
|
|
|
|
|
|
/* Helper for find_tuple_uint and others. */
|
|
static gpg_error_t
|
|
convert_uint (const unsigned char *s, size_t n, unsigned long long *r_value)
|
|
{
|
|
unsigned long long value = 0;
|
|
|
|
*r_value = 0;
|
|
|
|
if (!s)
|
|
return gpg_error (GPG_ERR_NOT_FOUND);
|
|
if (!n || (*s & 0x80)) /* No bytes or negative. */
|
|
return gpg_error (GPG_ERR_ERANGE);
|
|
if (n && !*s) /* Skip a leading zero. */
|
|
{
|
|
n--;
|
|
s++;
|
|
}
|
|
if (n > sizeof value)
|
|
return gpg_error (GPG_ERR_ERANGE);
|
|
for (; n; n--, s++)
|
|
{
|
|
value <<= 8;
|
|
value |= *s;
|
|
}
|
|
*r_value = value;
|
|
return 0;
|
|
}
|
|
|
|
|
|
/* Similar to find-tuple but expects an unsigned int value and stores
|
|
* that at R_VALUE. If the tag was not found GPG_ERR_NOT_FOUND is
|
|
* returned and 0 stored at R_VALUE. If the value cannot be converted
|
|
* to an unsigned integer GPG_ERR_ERANGE is returned. */
|
|
gpg_error_t
|
|
find_tuple_uint (tupledesc_t tupledesc, unsigned int tag,
|
|
unsigned long long *r_value)
|
|
{
|
|
const unsigned char *s;
|
|
size_t n;
|
|
|
|
s = find_tuple (tupledesc, tag, &n);
|
|
return convert_uint (s, n, r_value);
|
|
}
|
|
|
|
|
|
const void *
|
|
next_tuple (tupledesc_t tupledesc, unsigned int *r_tag, size_t *r_length)
|
|
{
|
|
const unsigned char *s;
|
|
const unsigned char *s_end; /* Points right behind the data. */
|
|
unsigned int t;
|
|
size_t n;
|
|
|
|
s = tupledesc->data;
|
|
if (!s)
|
|
return NULL;
|
|
s_end = s + tupledesc->datalen;
|
|
s += tupledesc->pos;
|
|
if (s < s_end
|
|
&& !((size_t)s + 3 >= (size_t)s_end || (size_t)s + 3 < (size_t)s))
|
|
{
|
|
t = s[0] << 8;
|
|
t |= s[1];
|
|
n = s[2] << 8;
|
|
n |= s[3];
|
|
s += 4;
|
|
if (!((size_t)s + n > (size_t)s_end || (size_t)s + n < (size_t)s))
|
|
{
|
|
tupledesc->pos = (s + n) - tupledesc->data;
|
|
*r_tag = t;
|
|
*r_length = n;
|
|
return s;
|
|
}
|
|
}
|
|
|
|
return NULL;
|
|
}
|
|
|
|
|
|
/* Return true if BUF has only printable characters. */
|
|
static int
|
|
all_printable (const void *buf, size_t buflen)
|
|
{
|
|
const unsigned char *s;
|
|
|
|
for (s=buf ; buflen; s++, buflen--)
|
|
if (*s < 32 || *s > 126)
|
|
return 0;
|
|
return 1;
|
|
}
|
|
|
|
|
|
/* Print information about TUPLES to the log stream. */
|
|
void
|
|
dump_tupledesc (tupledesc_t tuples)
|
|
{
|
|
size_t n;
|
|
unsigned int tag;
|
|
const void *value;
|
|
unsigned long long uint;
|
|
|
|
log_info ("keyblob dump:\n");
|
|
tag = KEYBLOB_TAG_BLOBVERSION;
|
|
value = find_tuple (tuples, tag, &n);
|
|
while (value)
|
|
{
|
|
log_info (" tag: %-5u len: %-2u value: ", tag, (unsigned int)n);
|
|
if (!n)
|
|
log_printf ("[none]\n");
|
|
else
|
|
{
|
|
switch (tag)
|
|
{
|
|
case KEYBLOB_TAG_ENCKEY:
|
|
case KEYBLOB_TAG_MACKEY:
|
|
log_printf ("[confidential]\n");
|
|
break;
|
|
|
|
case KEYBLOB_TAG_ALGOSTR:
|
|
if (n < 100 && all_printable (value, n))
|
|
log_printf ("%.*s\n", (int)n, (const char*)value);
|
|
else
|
|
log_printhex ("", value, n);
|
|
break;
|
|
|
|
case KEYBLOB_TAG_CONT_NSEC:
|
|
case KEYBLOB_TAG_ENC_NSEC:
|
|
case KEYBLOB_TAG_ENC_OFF:
|
|
if (!convert_uint (value, n, &uint))
|
|
log_printf ("%llu\n", uint);
|
|
else
|
|
log_printhex ("", value, n);
|
|
break;
|
|
|
|
default:
|
|
log_printhex ("", value, n);
|
|
break;
|
|
}
|
|
}
|
|
value = next_tuple (tuples, &tag, &n);
|
|
}
|
|
}
|