1
0
mirror of git://git.gnupg.org/gnupg.git synced 2025-01-03 12:11:33 +01:00

common: New helper functions for OpenPGP curve OIDs.

* common/openpgp-oid.c (openpgp_oidbuf_to_str): Factor most code out
to ...
(openpgp_oidbuf_to_str): new.
(openpgp_oidbuf_is_ed25519): New.
(openpgp_oidbuf_is_cv25519): New.
--

At some places it is more convenient (and faster) to directly work on
buffers and avoid the way via opaque MPIs.  These 3 new functions
allow for that.

Signed-off-by: Werner Koch <wk@gnupg.org>
(cherry picked from commit 4a1558d0c7190cf13d35385e47291a7aa121be3e)
This commit is contained in:
Werner Koch 2019-01-29 18:19:05 +01:00
parent 9fd6ba268f
commit dddbb26155
No known key found for this signature in database
GPG Key ID: E3FDFF218E45B72B
3 changed files with 66 additions and 34 deletions

View File

@ -184,48 +184,36 @@ openpgp_oid_from_str (const char *string, gcry_mpi_t *r_mpi)
} }
/* Return a malloced string represenation of the OID in the opaque MPI /* Return a malloced string representation of the OID in the buffer
A. In case of an error NULL is returned and ERRNO is set. */ * (BUF,LEN). In case of an error NULL is returned and ERRNO is set.
* As per OpenPGP spec the first byte of the buffer is the length of
* the rest; the function performs a consistency check. */
char * char *
openpgp_oid_to_str (gcry_mpi_t a) openpgp_oidbuf_to_str (const unsigned char *buf, size_t len)
{ {
const unsigned char *buf;
size_t length;
unsigned int lengthi;
char *string, *p; char *string, *p;
int n = 0; int n = 0;
unsigned long val, valmask; unsigned long val, valmask;
valmask = (unsigned long)0xfe << (8 * (sizeof (valmask) - 1)); valmask = (unsigned long)0xfe << (8 * (sizeof (valmask) - 1));
if (!a
|| !gcry_mpi_get_flag (a, GCRYMPI_FLAG_OPAQUE)
|| !(buf = gcry_mpi_get_opaque (a, &lengthi)))
{
gpg_err_set_errno (EINVAL);
return NULL;
}
buf = gcry_mpi_get_opaque (a, &lengthi);
length = (lengthi+7)/8;
/* The first bytes gives the length; check consistency. */ /* The first bytes gives the length; check consistency. */
if (!length || buf[0] != length -1)
if (!len || buf[0] != len -1)
{ {
gpg_err_set_errno (EINVAL); gpg_err_set_errno (EINVAL);
return NULL; return NULL;
} }
/* Skip length byte. */ /* Skip length byte. */
length--; len--;
buf++; buf++;
/* To calculate the length of the string we can safely assume an /* To calculate the length of the string we can safely assume an
upper limit of 3 decimal characters per byte. Two extra bytes upper limit of 3 decimal characters per byte. Two extra bytes
account for the special first octect */ account for the special first octect */
string = p = xtrymalloc (length*(1+3)+2+1); string = p = xtrymalloc (len*(1+3)+2+1);
if (!string) if (!string)
return NULL; return NULL;
if (!length) if (!len)
{ {
*p = 0; *p = 0;
return string; return string;
@ -237,7 +225,7 @@ openpgp_oid_to_str (gcry_mpi_t a)
p += sprintf (p, "1.%d", buf[n]-40); p += sprintf (p, "1.%d", buf[n]-40);
else { else {
val = buf[n] & 0x7f; val = buf[n] & 0x7f;
while ( (buf[n]&0x80) && ++n < length ) while ( (buf[n]&0x80) && ++n < len )
{ {
if ( (val & valmask) ) if ( (val & valmask) )
goto badoid; /* Overflow. */ goto badoid; /* Overflow. */
@ -250,10 +238,10 @@ openpgp_oid_to_str (gcry_mpi_t a)
sprintf (p, "2.%lu", val); sprintf (p, "2.%lu", val);
p += strlen (p); p += strlen (p);
} }
for (n++; n < length; n++) for (n++; n < len; n++)
{ {
val = buf[n] & 0x7f; val = buf[n] & 0x7f;
while ( (buf[n]&0x80) && ++n < length ) while ( (buf[n]&0x80) && ++n < len )
{ {
if ( (val & valmask) ) if ( (val & valmask) )
goto badoid; /* Overflow. */ goto badoid; /* Overflow. */
@ -278,6 +266,35 @@ openpgp_oid_to_str (gcry_mpi_t a)
} }
/* Return a malloced string representation of the OID in the opaque
* MPI A. In case of an error NULL is returned and ERRNO is set. */
char *
openpgp_oid_to_str (gcry_mpi_t a)
{
const unsigned char *buf;
unsigned int lengthi;
if (!a
|| !gcry_mpi_get_flag (a, GCRYMPI_FLAG_OPAQUE)
|| !(buf = gcry_mpi_get_opaque (a, &lengthi)))
{
gpg_err_set_errno (EINVAL);
return NULL;
}
buf = gcry_mpi_get_opaque (a, &lengthi);
return openpgp_oidbuf_to_str (buf, (lengthi+7)/8);
}
/* Return true if (BUF,LEN) represents the OID for Ed25519. */
int
openpgp_oidbuf_is_ed25519 (const void *buf, size_t len)
{
return (buf && len == DIM (oid_ed25519)
&& !memcmp (buf, oid_ed25519, DIM (oid_ed25519)));
}
/* Return true if A represents the OID for Ed25519. */ /* Return true if A represents the OID for Ed25519. */
int int
@ -285,32 +302,36 @@ openpgp_oid_is_ed25519 (gcry_mpi_t a)
{ {
const unsigned char *buf; const unsigned char *buf;
unsigned int nbits; unsigned int nbits;
size_t n;
if (!a || !gcry_mpi_get_flag (a, GCRYMPI_FLAG_OPAQUE)) if (!a || !gcry_mpi_get_flag (a, GCRYMPI_FLAG_OPAQUE))
return 0; return 0;
buf = gcry_mpi_get_opaque (a, &nbits); buf = gcry_mpi_get_opaque (a, &nbits);
n = (nbits+7)/8; return openpgp_oidbuf_is_ed25519 (buf, (nbits+7)/8);
return (n == DIM (oid_ed25519)
&& !memcmp (buf, oid_ed25519, DIM (oid_ed25519)));
} }
/* Return true if (BUF,LEN) represents the OID for Curve25519. */
int
openpgp_oidbuf_is_cv25519 (const void *buf, size_t len)
{
return (buf && len == DIM (oid_cv25519)
&& !memcmp (buf, oid_cv25519, DIM (oid_cv25519)));
}
/* Return true if the MPI A represents the OID for Curve25519. */
int int
openpgp_oid_is_cv25519 (gcry_mpi_t a) openpgp_oid_is_cv25519 (gcry_mpi_t a)
{ {
const unsigned char *buf; const unsigned char *buf;
unsigned int nbits; unsigned int nbits;
size_t n;
if (!a || !gcry_mpi_get_flag (a, GCRYMPI_FLAG_OPAQUE)) if (!a || !gcry_mpi_get_flag (a, GCRYMPI_FLAG_OPAQUE))
return 0; return 0;
buf = gcry_mpi_get_opaque (a, &nbits); buf = gcry_mpi_get_opaque (a, &nbits);
n = (nbits+7)/8; return openpgp_oidbuf_is_cv25519 (buf, (nbits+7)/8);
return (n == DIM (oid_cv25519)
&& !memcmp (buf, oid_cv25519, DIM (oid_cv25519)));
} }

View File

@ -142,7 +142,15 @@ test_openpgp_oid_to_str (void)
fail (idx, 0); fail (idx, 0);
xfree (string); xfree (string);
gcry_mpi_release (a); gcry_mpi_release (a);
}
/* Again using the buffer variant. */
string = openpgp_oidbuf_to_str (samples[idx].der, samples[idx].der[0]+1);
if (!string)
fail (idx, gpg_error_from_syserror ());
if (strcmp (string, samples[idx].string))
fail (idx, 0);
xfree (string);
}
} }

View File

@ -219,8 +219,11 @@ size_t percent_unescape_inplace (char *string, int nulrepl);
/*-- openpgp-oid.c --*/ /*-- openpgp-oid.c --*/
gpg_error_t openpgp_oid_from_str (const char *string, gcry_mpi_t *r_mpi); gpg_error_t openpgp_oid_from_str (const char *string, gcry_mpi_t *r_mpi);
char *openpgp_oidbuf_to_str (const unsigned char *buf, size_t len);
char *openpgp_oid_to_str (gcry_mpi_t a); char *openpgp_oid_to_str (gcry_mpi_t a);
int openpgp_oidbuf_is_ed25519 (const void *buf, size_t len);
int openpgp_oid_is_ed25519 (gcry_mpi_t a); int openpgp_oid_is_ed25519 (gcry_mpi_t a);
int openpgp_oidbuf_is_cv25519 (const void *buf, size_t len);
int openpgp_oid_is_cv25519 (gcry_mpi_t a); int openpgp_oid_is_cv25519 (gcry_mpi_t a);
const char *openpgp_curve_to_oid (const char *name, unsigned int *r_nbits); const char *openpgp_curve_to_oid (const char *name, unsigned int *r_nbits);
const char *openpgp_oid_to_curve (const char *oid, int canon); const char *openpgp_oid_to_curve (const char *oid, int canon);