2003-06-18 21:56:13 +02:00
|
|
|
/* pkglue.c - public key operations glue code
|
|
|
|
* Copyright (C) 2000, 2003 Free Software Foundation, Inc.
|
|
|
|
*
|
|
|
|
* 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 21:49:40 +02:00
|
|
|
* the Free Software Foundation; either version 3 of the License, or
|
2003-06-18 21:56:13 +02: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
|
2007-07-04 21:49:40 +02:00
|
|
|
* along with this program; if not, see <http://www.gnu.org/licenses/>.
|
2003-06-18 21:56:13 +02:00
|
|
|
*/
|
|
|
|
|
|
|
|
#include <config.h>
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <string.h>
|
|
|
|
#include <errno.h>
|
|
|
|
#include <assert.h>
|
|
|
|
|
|
|
|
#include "gpg.h"
|
|
|
|
#include "util.h"
|
|
|
|
#include "pkglue.h"
|
|
|
|
|
|
|
|
|
2003-08-25 23:12:43 +02:00
|
|
|
static gcry_mpi_t
|
|
|
|
mpi_from_sexp (gcry_sexp_t sexp, const char * item)
|
|
|
|
{
|
|
|
|
gcry_sexp_t list;
|
|
|
|
gcry_mpi_t data;
|
2011-06-13 14:35:30 +02:00
|
|
|
|
2003-08-25 23:12:43 +02:00
|
|
|
list = gcry_sexp_find_token (sexp, item, 0);
|
|
|
|
assert (list);
|
2011-06-13 14:35:30 +02:00
|
|
|
data = gcry_sexp_nth_mpi (list, 1, GCRYMPI_FMT_USG);
|
2003-08-25 23:12:43 +02:00
|
|
|
assert (data);
|
|
|
|
gcry_sexp_release (list);
|
|
|
|
return data;
|
|
|
|
}
|
|
|
|
|
2003-06-18 21:56:13 +02:00
|
|
|
|
|
|
|
/****************
|
|
|
|
* Emulate our old PK interface here - sometime in the future we might
|
|
|
|
* change the internal design to directly fit to libgcrypt.
|
|
|
|
*/
|
|
|
|
int
|
|
|
|
pk_sign (int algo, gcry_mpi_t * data, gcry_mpi_t hash, gcry_mpi_t * skey)
|
|
|
|
{
|
2003-08-25 23:12:43 +02:00
|
|
|
gcry_sexp_t s_sig, s_hash, s_skey;
|
2003-06-18 21:56:13 +02:00
|
|
|
int rc;
|
|
|
|
|
|
|
|
/* make a sexp from skey */
|
|
|
|
if (algo == GCRY_PK_DSA)
|
|
|
|
{
|
|
|
|
rc = gcry_sexp_build (&s_skey, NULL,
|
|
|
|
"(private-key(dsa(p%m)(q%m)(g%m)(y%m)(x%m)))",
|
|
|
|
skey[0], skey[1], skey[2], skey[3], skey[4]);
|
|
|
|
}
|
2009-09-25 19:09:31 +02:00
|
|
|
else if (algo == GCRY_PK_RSA || algo == GCRY_PK_RSA_S)
|
2003-08-07 09:05:38 +02:00
|
|
|
{
|
|
|
|
rc = gcry_sexp_build (&s_skey, NULL,
|
|
|
|
"(private-key(rsa(n%m)(e%m)(d%m)(p%m)(q%m)(u%m)))",
|
|
|
|
skey[0], skey[1], skey[2], skey[3], skey[4],
|
|
|
|
skey[5]);
|
|
|
|
}
|
2003-06-18 21:56:13 +02:00
|
|
|
else if (algo == GCRY_PK_ELG || algo == GCRY_PK_ELG_E)
|
|
|
|
{
|
|
|
|
rc = gcry_sexp_build (&s_skey, NULL,
|
|
|
|
"(private-key(elg(p%m)(g%m)(y%m)(x%m)))",
|
|
|
|
skey[0], skey[1], skey[2], skey[3]);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
return GPG_ERR_PUBKEY_ALGO;
|
|
|
|
|
|
|
|
if (rc)
|
|
|
|
BUG ();
|
|
|
|
|
|
|
|
/* put hash into a S-Exp s_hash */
|
|
|
|
if (gcry_sexp_build (&s_hash, NULL, "%m", hash))
|
|
|
|
BUG ();
|
|
|
|
|
|
|
|
rc = gcry_pk_sign (&s_sig, s_hash, s_skey);
|
|
|
|
gcry_sexp_release (s_hash);
|
|
|
|
gcry_sexp_release (s_skey);
|
|
|
|
|
|
|
|
if (rc)
|
|
|
|
;
|
2009-09-25 19:09:31 +02:00
|
|
|
else if (algo == GCRY_PK_RSA || algo == GCRY_PK_RSA_S)
|
2003-08-25 23:12:43 +02:00
|
|
|
data[0] = mpi_from_sexp (s_sig, "s");
|
2003-06-18 21:56:13 +02:00
|
|
|
else
|
|
|
|
{
|
2003-08-25 23:12:43 +02:00
|
|
|
data[0] = mpi_from_sexp (s_sig, "r");
|
|
|
|
data[1] = mpi_from_sexp (s_sig, "s");
|
2003-06-18 21:56:13 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
gcry_sexp_release (s_sig);
|
|
|
|
return rc;
|
|
|
|
}
|
|
|
|
|
|
|
|
/****************
|
|
|
|
* Emulate our old PK interface here - sometime in the future we might
|
|
|
|
* change the internal design to directly fit to libgcrypt.
|
|
|
|
*/
|
|
|
|
int
|
|
|
|
pk_verify (int algo, gcry_mpi_t hash, gcry_mpi_t * data, gcry_mpi_t * pkey)
|
|
|
|
{
|
|
|
|
gcry_sexp_t s_sig, s_hash, s_pkey;
|
|
|
|
int rc;
|
|
|
|
|
|
|
|
/* make a sexp from pkey */
|
|
|
|
if (algo == GCRY_PK_DSA)
|
|
|
|
{
|
|
|
|
rc = gcry_sexp_build (&s_pkey, NULL,
|
|
|
|
"(public-key(dsa(p%m)(q%m)(g%m)(y%m)))",
|
|
|
|
pkey[0], pkey[1], pkey[2], pkey[3]);
|
|
|
|
}
|
|
|
|
else if (algo == GCRY_PK_ELG || algo == GCRY_PK_ELG_E)
|
|
|
|
{
|
|
|
|
rc = gcry_sexp_build (&s_pkey, NULL,
|
|
|
|
"(public-key(elg(p%m)(g%m)(y%m)))",
|
|
|
|
pkey[0], pkey[1], pkey[2]);
|
|
|
|
}
|
2009-09-25 19:09:31 +02:00
|
|
|
else if (algo == GCRY_PK_RSA || algo == GCRY_PK_RSA_S)
|
2003-06-18 21:56:13 +02:00
|
|
|
{
|
|
|
|
rc = gcry_sexp_build (&s_pkey, NULL,
|
|
|
|
"(public-key(rsa(n%m)(e%m)))", pkey[0], pkey[1]);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
return GPG_ERR_PUBKEY_ALGO;
|
|
|
|
|
|
|
|
if (rc)
|
2009-02-24 12:47:25 +01:00
|
|
|
BUG (); /* gcry_sexp_build should never fail. */
|
2003-06-18 21:56:13 +02:00
|
|
|
|
|
|
|
/* put hash into a S-Exp s_hash */
|
|
|
|
if (gcry_sexp_build (&s_hash, NULL, "%m", hash))
|
2009-02-24 12:47:25 +01:00
|
|
|
BUG (); /* gcry_sexp_build should never fail. */
|
2003-06-18 21:56:13 +02:00
|
|
|
|
2009-02-24 12:47:25 +01:00
|
|
|
/* Put data into a S-Exp s_sig. */
|
|
|
|
s_sig = NULL;
|
2003-06-18 21:56:13 +02:00
|
|
|
if (algo == GCRY_PK_DSA)
|
|
|
|
{
|
2003-08-07 09:05:38 +02:00
|
|
|
if (!data[0] || !data[1])
|
|
|
|
rc = gpg_error (GPG_ERR_BAD_MPI);
|
|
|
|
else
|
|
|
|
rc = gcry_sexp_build (&s_sig, NULL,
|
|
|
|
"(sig-val(dsa(r%m)(s%m)))", data[0], data[1]);
|
2003-06-18 21:56:13 +02:00
|
|
|
}
|
|
|
|
else if (algo == GCRY_PK_ELG || algo == GCRY_PK_ELG_E)
|
|
|
|
{
|
2003-08-07 09:05:38 +02:00
|
|
|
if (!data[0] || !data[1])
|
|
|
|
rc = gpg_error (GPG_ERR_BAD_MPI);
|
|
|
|
else
|
|
|
|
rc = gcry_sexp_build (&s_sig, NULL,
|
|
|
|
"(sig-val(elg(r%m)(s%m)))", data[0], data[1]);
|
2003-06-18 21:56:13 +02:00
|
|
|
}
|
2009-09-25 19:09:31 +02:00
|
|
|
else if (algo == GCRY_PK_RSA || algo == GCRY_PK_RSA_S)
|
2003-06-18 21:56:13 +02:00
|
|
|
{
|
2003-08-07 09:05:38 +02:00
|
|
|
if (!data[0])
|
|
|
|
rc = gpg_error (GPG_ERR_BAD_MPI);
|
|
|
|
else
|
|
|
|
rc = gcry_sexp_build (&s_sig, NULL, "(sig-val(rsa(s%m)))", data[0]);
|
2003-06-18 21:56:13 +02:00
|
|
|
}
|
|
|
|
else
|
|
|
|
BUG ();
|
|
|
|
|
2009-02-24 12:47:25 +01:00
|
|
|
if (!rc)
|
|
|
|
rc = gcry_pk_verify (s_sig, s_hash, s_pkey);
|
2003-06-18 21:56:13 +02:00
|
|
|
|
|
|
|
gcry_sexp_release (s_sig);
|
|
|
|
gcry_sexp_release (s_hash);
|
|
|
|
gcry_sexp_release (s_pkey);
|
|
|
|
return rc;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/****************
|
|
|
|
* Emulate our old PK interface here - sometime in the future we might
|
|
|
|
* change the internal design to directly fit to libgcrypt.
|
|
|
|
*/
|
|
|
|
int
|
|
|
|
pk_encrypt (int algo, gcry_mpi_t * resarr, gcry_mpi_t data, gcry_mpi_t * pkey)
|
|
|
|
{
|
|
|
|
gcry_sexp_t s_ciph, s_data, s_pkey;
|
|
|
|
int rc;
|
|
|
|
|
|
|
|
/* make a sexp from pkey */
|
|
|
|
if (algo == GCRY_PK_ELG || algo == GCRY_PK_ELG_E)
|
|
|
|
{
|
|
|
|
rc = gcry_sexp_build (&s_pkey, NULL,
|
|
|
|
"(public-key(elg(p%m)(g%m)(y%m)))",
|
|
|
|
pkey[0], pkey[1], pkey[2]);
|
|
|
|
}
|
2009-09-25 19:09:31 +02:00
|
|
|
else if (algo == GCRY_PK_RSA || algo == GCRY_PK_RSA_E)
|
2003-07-03 20:08:16 +02:00
|
|
|
{
|
|
|
|
rc = gcry_sexp_build (&s_pkey, NULL,
|
|
|
|
"(public-key(rsa(n%m)(e%m)))",
|
|
|
|
pkey[0], pkey[1]);
|
|
|
|
}
|
2003-06-18 21:56:13 +02:00
|
|
|
else
|
|
|
|
return GPG_ERR_PUBKEY_ALGO;
|
|
|
|
|
|
|
|
if (rc)
|
|
|
|
BUG ();
|
|
|
|
|
|
|
|
/* put the data into a simple list */
|
|
|
|
if (gcry_sexp_build (&s_data, NULL, "%m", data))
|
|
|
|
BUG ();
|
|
|
|
|
|
|
|
/* pass it to libgcrypt */
|
|
|
|
rc = gcry_pk_encrypt (&s_ciph, s_data, s_pkey);
|
|
|
|
gcry_sexp_release (s_data);
|
|
|
|
gcry_sexp_release (s_pkey);
|
|
|
|
|
|
|
|
if (rc)
|
|
|
|
;
|
|
|
|
else
|
|
|
|
{ /* add better error handling or make gnupg use S-Exp directly */
|
2003-08-25 23:12:43 +02:00
|
|
|
resarr[0] = mpi_from_sexp (s_ciph, "a");
|
2009-09-25 19:09:31 +02:00
|
|
|
if (algo != GCRY_PK_RSA && algo != GCRY_PK_RSA_E)
|
2003-08-25 23:12:43 +02:00
|
|
|
resarr[1] = mpi_from_sexp (s_ciph, "b");
|
2003-06-18 21:56:13 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
gcry_sexp_release (s_ciph);
|
|
|
|
return rc;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/****************
|
|
|
|
* Emulate our old PK interface here - sometime in the future we might
|
|
|
|
* change the internal design to directly fit to libgcrypt.
|
|
|
|
*/
|
|
|
|
int
|
|
|
|
pk_decrypt (int algo, gcry_mpi_t * result, gcry_mpi_t * data,
|
|
|
|
gcry_mpi_t * skey)
|
|
|
|
{
|
|
|
|
gcry_sexp_t s_skey, s_data, s_plain;
|
|
|
|
int rc;
|
|
|
|
|
|
|
|
*result = NULL;
|
|
|
|
/* make a sexp from skey */
|
|
|
|
if (algo == GCRY_PK_ELG || algo == GCRY_PK_ELG_E)
|
|
|
|
{
|
|
|
|
rc = gcry_sexp_build (&s_skey, NULL,
|
|
|
|
"(private-key(elg(p%m)(g%m)(y%m)(x%m)))",
|
|
|
|
skey[0], skey[1], skey[2], skey[3]);
|
|
|
|
}
|
2009-09-25 19:09:31 +02:00
|
|
|
else if (algo == GCRY_PK_RSA || algo == GCRY_PK_RSA_E)
|
2003-06-18 21:56:13 +02:00
|
|
|
{
|
|
|
|
rc = gcry_sexp_build (&s_skey, NULL,
|
|
|
|
"(private-key(rsa(n%m)(e%m)(d%m)(p%m)(q%m)(u%m)))",
|
|
|
|
skey[0], skey[1], skey[2], skey[3], skey[4],
|
|
|
|
skey[5]);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
return GPG_ERR_PUBKEY_ALGO;
|
|
|
|
|
|
|
|
if (rc)
|
|
|
|
BUG ();
|
|
|
|
|
|
|
|
/* put data into a S-Exp s_data */
|
|
|
|
if (algo == GCRY_PK_ELG || algo == GCRY_PK_ELG_E)
|
|
|
|
{
|
2003-08-07 09:05:38 +02:00
|
|
|
if (!data[0] || !data[1])
|
|
|
|
rc = gpg_error (GPG_ERR_BAD_MPI);
|
|
|
|
else
|
|
|
|
rc = gcry_sexp_build (&s_data, NULL,
|
|
|
|
"(enc-val(elg(a%m)(b%m)))", data[0], data[1]);
|
2003-06-18 21:56:13 +02:00
|
|
|
}
|
2009-09-25 19:09:31 +02:00
|
|
|
else if (algo == GCRY_PK_RSA || algo == GCRY_PK_RSA_E)
|
2003-06-18 21:56:13 +02:00
|
|
|
{
|
2003-08-07 09:05:38 +02:00
|
|
|
if (!data[0])
|
|
|
|
rc = gpg_error (GPG_ERR_BAD_MPI);
|
|
|
|
else
|
|
|
|
rc = gcry_sexp_build (&s_data, NULL, "(enc-val(rsa(a%m)))", data[0]);
|
2003-06-18 21:56:13 +02:00
|
|
|
}
|
|
|
|
else
|
|
|
|
BUG ();
|
|
|
|
|
|
|
|
if (rc)
|
|
|
|
BUG ();
|
|
|
|
|
|
|
|
rc = gcry_pk_decrypt (&s_plain, s_data, s_skey);
|
|
|
|
gcry_sexp_release (s_skey);
|
|
|
|
gcry_sexp_release (s_data);
|
|
|
|
if (rc)
|
|
|
|
return rc;
|
|
|
|
|
2011-06-13 14:35:30 +02:00
|
|
|
*result = gcry_sexp_nth_mpi (s_plain, 0, GCRYMPI_FMT_USG);
|
2003-06-18 21:56:13 +02:00
|
|
|
gcry_sexp_release (s_plain);
|
|
|
|
if (!*result)
|
|
|
|
return -1; /* oops */
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
2003-07-03 20:08:16 +02:00
|
|
|
|
|
|
|
|
2003-08-05 19:11:04 +02:00
|
|
|
/* Check whether SKEY is a suitable secret key. */
|
|
|
|
int
|
|
|
|
pk_check_secret_key (int algo, gcry_mpi_t *skey)
|
|
|
|
{
|
|
|
|
gcry_sexp_t s_skey;
|
|
|
|
int rc;
|
2003-07-03 20:08:16 +02:00
|
|
|
|
2003-08-05 19:11:04 +02:00
|
|
|
if (algo == GCRY_PK_DSA)
|
|
|
|
{
|
|
|
|
rc = gcry_sexp_build (&s_skey, NULL,
|
|
|
|
"(private-key(dsa(p%m)(q%m)(g%m)(y%m)(x%m)))",
|
|
|
|
skey[0], skey[1], skey[2], skey[3], skey[4]);
|
|
|
|
}
|
|
|
|
else if (algo == GCRY_PK_ELG || algo == GCRY_PK_ELG_E)
|
|
|
|
{
|
|
|
|
rc = gcry_sexp_build (&s_skey, NULL,
|
|
|
|
"(private-key(elg(p%m)(g%m)(y%m)(x%m)))",
|
|
|
|
skey[0], skey[1], skey[2], skey[3]);
|
|
|
|
}
|
2009-09-25 19:09:31 +02:00
|
|
|
else if (algo == GCRY_PK_RSA
|
|
|
|
|| algo == GCRY_PK_RSA_S || algo == GCRY_PK_RSA_E)
|
2003-08-05 19:11:04 +02:00
|
|
|
{
|
|
|
|
rc = gcry_sexp_build (&s_skey, NULL,
|
|
|
|
"(private-key(rsa(n%m)(e%m)(d%m)(p%m)(q%m)(u%m)))",
|
|
|
|
skey[0], skey[1], skey[2], skey[3], skey[4],
|
|
|
|
skey[5]);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
return GPG_ERR_PUBKEY_ALGO;
|
2003-07-03 20:08:16 +02:00
|
|
|
|
2003-08-05 19:11:04 +02:00
|
|
|
if (!rc)
|
|
|
|
{
|
|
|
|
rc = gcry_pk_testkey (s_skey);
|
|
|
|
gcry_sexp_release (s_skey);
|
|
|
|
}
|
|
|
|
return rc;
|
|
|
|
}
|