2004-12-02 07:48:09 +00:00
|
|
|
|
/* app-p15.c - The pkcs#15 card application.
|
2005-04-27 12:09:21 +00:00
|
|
|
|
* Copyright (C) 2005 Free Software Foundation, Inc.
|
2021-04-29 15:26:36 +02:00
|
|
|
|
* Copyright (C) 2020, 2021 g10 Code GmbH
|
2004-12-02 07:48:09 +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
|
2004-12-02 07:48:09 +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/>.
|
2020-03-30 21:18:12 +02:00
|
|
|
|
* SPDX-License-Identifier: GPL-3.0-or-later
|
2004-12-02 07:48:09 +00:00
|
|
|
|
*/
|
|
|
|
|
|
2006-05-23 16:19:43 +00:00
|
|
|
|
/* Information pertaining to the BELPIC developer card samples:
|
|
|
|
|
|
|
|
|
|
Unblock PUK: "222222111111"
|
|
|
|
|
Reset PIN: "333333111111")
|
|
|
|
|
|
2011-02-04 12:57:53 +01:00
|
|
|
|
e.g. the APDUs 00:20:00:02:08:2C:33:33:33:11:11:11:FF
|
2006-05-23 16:19:43 +00:00
|
|
|
|
and 00:24:01:01:08:24:12:34:FF:FF:FF:FF:FF
|
|
|
|
|
should change the PIN into 1234.
|
|
|
|
|
*/
|
|
|
|
|
|
2004-12-02 07:48:09 +00:00
|
|
|
|
#include <config.h>
|
|
|
|
|
#include <errno.h>
|
|
|
|
|
#include <stdio.h>
|
|
|
|
|
#include <stdlib.h>
|
|
|
|
|
#include <string.h>
|
|
|
|
|
#include <time.h>
|
|
|
|
|
|
|
|
|
|
#include "scdaemon.h"
|
|
|
|
|
|
|
|
|
|
#include "iso7816.h"
|
2020-04-07 20:38:15 +02:00
|
|
|
|
#include "../common/i18n.h"
|
2017-03-07 20:21:23 +09:00
|
|
|
|
#include "../common/tlv.h"
|
2021-04-29 15:26:36 +02:00
|
|
|
|
#include "../common/host2net.h"
|
|
|
|
|
#include "../common/openpgpdefs.h"
|
2005-04-27 12:09:21 +00:00
|
|
|
|
#include "apdu.h" /* fixme: we should move the card detection to a
|
|
|
|
|
separate file */
|
|
|
|
|
|
2021-04-29 15:26:36 +02:00
|
|
|
|
|
|
|
|
|
static const char oid_kp_codeSigning[] = "1.3.6.1.5.5.7.3.3";
|
|
|
|
|
static const char oid_kp_timeStamping[] = "1.3.6.1.5.5.7.3.8";
|
|
|
|
|
static const char oid_kp_ocspSigning[] = "1.3.6.1.5.5.7.3.9";
|
|
|
|
|
static const char oid_kp_ms_documentSigning[] = "1.3.6.1.4.1.311.10.3.12";
|
|
|
|
|
static const char oid_kp_ms_old_documentSigning[] = "1.3.6.1.4.1.311.3.10.3.12";
|
|
|
|
|
|
|
|
|
|
static const char oid_kp_emailProtection[]= "1.3.6.1.5.5.7.3.4";
|
|
|
|
|
|
|
|
|
|
static const char oid_kp_serverAuth[] = "1.3.6.1.5.5.7.3.1";
|
|
|
|
|
static const char oid_kp_clientAuth[] = "1.3.6.1.5.5.7.3.2";
|
|
|
|
|
static const char oid_kp_ms_smartcardLogon[] = "1.3.6.1.4.1.311.20.2.2";
|
|
|
|
|
|
|
|
|
|
static const char oid_kp_anyExtendedKeyUsage[] = "2.5.29.37.0";
|
|
|
|
|
|
|
|
|
|
static const char oid_kp_gpgUsageCert[] = "1.3.6.1.4.1.11591.2.6.1";
|
|
|
|
|
static const char oid_kp_gpgUsageSign[] = "1.3.6.1.4.1.11591.2.6.2";
|
|
|
|
|
static const char oid_kp_gpgUsageEncr[] = "1.3.6.1.4.1.11591.2.6.3";
|
|
|
|
|
static const char oid_kp_gpgUsageAuth[] = "1.3.6.1.4.1.11591.2.6.4";
|
|
|
|
|
|
2005-04-27 12:09:21 +00:00
|
|
|
|
/* Types of cards we know and which needs special treatment. */
|
|
|
|
|
typedef enum
|
|
|
|
|
{
|
|
|
|
|
CARD_TYPE_UNKNOWN,
|
|
|
|
|
CARD_TYPE_TCOS,
|
2005-09-06 18:42:13 +00:00
|
|
|
|
CARD_TYPE_MICARDO,
|
2020-03-30 21:18:12 +02:00
|
|
|
|
CARD_TYPE_CARDOS_50,
|
2021-04-29 15:26:36 +02:00
|
|
|
|
CARD_TYPE_CARDOS_53,
|
2021-06-18 17:32:01 +02:00
|
|
|
|
CARD_TYPE_AET, /* A.E.T. Europe JCOP card. */
|
2005-09-06 18:42:13 +00:00
|
|
|
|
CARD_TYPE_BELPIC /* Belgian eID card specs. */
|
2011-02-04 12:57:53 +01:00
|
|
|
|
}
|
2005-09-06 18:42:13 +00:00
|
|
|
|
card_type_t;
|
2005-04-27 12:09:21 +00:00
|
|
|
|
|
2020-04-07 16:00:11 +02:00
|
|
|
|
/* The OS of card as specified by card_type_t is not always
|
|
|
|
|
* sufficient. Thus we also distinguish the actual product build upon
|
|
|
|
|
* the given OS. */
|
|
|
|
|
typedef enum
|
|
|
|
|
{
|
|
|
|
|
CARD_PRODUCT_UNKNOWN,
|
2021-04-29 15:26:36 +02:00
|
|
|
|
CARD_PRODUCT_RSCS, /* Rohde&Schwarz Cybersecurity */
|
2022-04-13 13:06:27 +02:00
|
|
|
|
CARD_PRODUCT_DTRUST, /* D-Trust GmbH (bundesdruckerei.de) */
|
|
|
|
|
CARD_PRODUCT_GENUA /* GeNUA mbH */
|
2020-04-07 16:00:11 +02:00
|
|
|
|
}
|
|
|
|
|
card_product_t;
|
|
|
|
|
|
|
|
|
|
|
2005-04-27 12:09:21 +00:00
|
|
|
|
/* A list card types with ATRs noticed with these cards. */
|
2005-06-16 08:12:03 +00:00
|
|
|
|
#define X(a) ((unsigned char const *)(a))
|
2011-02-04 12:57:53 +01:00
|
|
|
|
static struct
|
2005-04-27 12:09:21 +00:00
|
|
|
|
{
|
|
|
|
|
size_t atrlen;
|
2005-06-16 08:12:03 +00:00
|
|
|
|
unsigned char const *atr;
|
2005-04-27 12:09:21 +00:00
|
|
|
|
card_type_t type;
|
|
|
|
|
} card_atr_list[] = {
|
2005-06-16 08:12:03 +00:00
|
|
|
|
{ 19, X("\x3B\xBA\x13\x00\x81\x31\x86\x5D\x00\x64\x05\x0A\x02\x01\x31\x80"
|
2011-02-04 12:57:53 +01:00
|
|
|
|
"\x90\x00\x8B"),
|
2005-04-27 12:09:21 +00:00
|
|
|
|
CARD_TYPE_TCOS }, /* SLE44 */
|
2005-06-16 08:12:03 +00:00
|
|
|
|
{ 19, X("\x3B\xBA\x14\x00\x81\x31\x86\x5D\x00\x64\x05\x14\x02\x02\x31\x80"
|
2011-02-04 12:57:53 +01:00
|
|
|
|
"\x90\x00\x91"),
|
2005-04-27 12:09:21 +00:00
|
|
|
|
CARD_TYPE_TCOS }, /* SLE66S */
|
2005-06-16 08:12:03 +00:00
|
|
|
|
{ 19, X("\x3B\xBA\x96\x00\x81\x31\x86\x5D\x00\x64\x05\x60\x02\x03\x31\x80"
|
|
|
|
|
"\x90\x00\x66"),
|
2005-04-27 12:09:21 +00:00
|
|
|
|
CARD_TYPE_TCOS }, /* SLE66P */
|
2005-06-16 08:12:03 +00:00
|
|
|
|
{ 27, X("\x3B\xFF\x94\x00\xFF\x80\xB1\xFE\x45\x1F\x03\x00\x68\xD2\x76\x00"
|
|
|
|
|
"\x00\x28\xFF\x05\x1E\x31\x80\x00\x90\x00\x23"),
|
2005-04-27 12:09:21 +00:00
|
|
|
|
CARD_TYPE_MICARDO }, /* German BMI card */
|
2005-06-16 08:12:03 +00:00
|
|
|
|
{ 19, X("\x3B\x6F\x00\xFF\x00\x68\xD2\x76\x00\x00\x28\xFF\x05\x1E\x31\x80"
|
|
|
|
|
"\x00\x90\x00"),
|
2005-04-27 19:47:53 +00:00
|
|
|
|
CARD_TYPE_MICARDO }, /* German BMI card (ATR due to reader problem) */
|
2005-06-16 08:12:03 +00:00
|
|
|
|
{ 26, X("\x3B\xFE\x94\x00\xFF\x80\xB1\xFA\x45\x1F\x03\x45\x73\x74\x45\x49"
|
|
|
|
|
"\x44\x20\x76\x65\x72\x20\x31\x2E\x30\x43"),
|
2005-04-27 12:09:21 +00:00
|
|
|
|
CARD_TYPE_MICARDO }, /* EstEID (Estonian Big Brother card) */
|
2020-03-30 21:18:12 +02:00
|
|
|
|
{ 11, X("\x3b\xd2\x18\x00\x81\x31\xfe\x58\xc9\x01\x14"),
|
|
|
|
|
CARD_TYPE_CARDOS_50 }, /* CardOS 5.0 */
|
2021-04-29 15:26:36 +02:00
|
|
|
|
{ 11, X("\x3b\xd2\x18\x00\x81\x31\xfe\x58\xc9\x03\x16"),
|
|
|
|
|
CARD_TYPE_CARDOS_53 }, /* CardOS 5.3 */
|
2021-06-18 17:32:01 +02:00
|
|
|
|
{ 24, X("\x3b\xfe\x18\x00\x00\x80\x31\xfe\x45\x53\x43\x45"
|
|
|
|
|
"\x36\x30\x2d\x43\x44\x30\x38\x31\x2d\x6e\x46\xa9"),
|
|
|
|
|
CARD_TYPE_AET },
|
2005-04-27 12:09:21 +00:00
|
|
|
|
{ 0 }
|
|
|
|
|
};
|
2005-06-16 08:12:03 +00:00
|
|
|
|
#undef X
|
2005-04-27 12:09:21 +00:00
|
|
|
|
|
|
|
|
|
|
2021-04-29 15:26:36 +02:00
|
|
|
|
/* Macro to test for CardOS 5.0 and 5.3. */
|
|
|
|
|
#define IS_CARDOS_5(a) ((a)->app_local->card_type == CARD_TYPE_CARDOS_50 \
|
|
|
|
|
|| (a)->app_local->card_type == CARD_TYPE_CARDOS_53)
|
|
|
|
|
|
|
|
|
|
/* The default PKCS-15 home DF */
|
|
|
|
|
#define DEFAULT_HOME_DF 0x5015
|
|
|
|
|
|
2005-09-05 14:36:36 +00:00
|
|
|
|
/* The AID of PKCS15. */
|
|
|
|
|
static char const pkcs15_aid[] = { 0xA0, 0, 0, 0, 0x63,
|
|
|
|
|
0x50, 0x4B, 0x43, 0x53, 0x2D, 0x31, 0x35 };
|
|
|
|
|
|
|
|
|
|
/* The Belgian eID variant - they didn't understood why a shared AID
|
|
|
|
|
is useful for a standard. Oh well. */
|
|
|
|
|
static char const pkcs15be_aid[] = { 0xA0, 0, 0, 0x01, 0x77,
|
|
|
|
|
0x50, 0x4B, 0x43, 0x53, 0x2D, 0x31, 0x35 };
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* The PIN types as defined in pkcs#15 v1.1 */
|
2011-02-04 12:57:53 +01:00
|
|
|
|
typedef enum
|
2005-04-27 12:09:21 +00:00
|
|
|
|
{
|
|
|
|
|
PIN_TYPE_BCD = 0,
|
|
|
|
|
PIN_TYPE_ASCII_NUMERIC = 1,
|
|
|
|
|
PIN_TYPE_UTF8 = 2,
|
|
|
|
|
PIN_TYPE_HALF_NIBBLE_BCD = 3,
|
|
|
|
|
PIN_TYPE_ISO9564_1 = 4
|
|
|
|
|
} pin_type_t;
|
|
|
|
|
|
2021-06-22 11:11:46 +02:00
|
|
|
|
/* The AuthenticationTypes as defined in pkcs#15 v1.1 (6.8.1) */
|
|
|
|
|
typedef enum
|
|
|
|
|
{
|
|
|
|
|
AUTH_TYPE_PIN = -1,
|
|
|
|
|
AUTH_TYPE_BIOMETRIC = 0,
|
|
|
|
|
AUTH_TYPE_AUTHKEY = 1,
|
|
|
|
|
AUTH_TYPE_EXTERNAL = 2,
|
|
|
|
|
} auth_type_t;
|
2005-04-27 12:09:21 +00:00
|
|
|
|
|
|
|
|
|
/* A bit array with for the key usage flags from the
|
|
|
|
|
commonKeyAttributes. */
|
|
|
|
|
struct keyusage_flags_s
|
|
|
|
|
{
|
|
|
|
|
unsigned int encrypt: 1;
|
|
|
|
|
unsigned int decrypt: 1;
|
|
|
|
|
unsigned int sign: 1;
|
|
|
|
|
unsigned int sign_recover: 1;
|
|
|
|
|
unsigned int wrap: 1;
|
|
|
|
|
unsigned int unwrap: 1;
|
|
|
|
|
unsigned int verify: 1;
|
|
|
|
|
unsigned int verify_recover: 1;
|
|
|
|
|
unsigned int derive: 1;
|
|
|
|
|
unsigned int non_repudiation: 1;
|
|
|
|
|
};
|
|
|
|
|
typedef struct keyusage_flags_s keyusage_flags_t;
|
|
|
|
|
|
|
|
|
|
|
2021-04-29 15:26:36 +02:00
|
|
|
|
/* A bit array with for the key access flags from the
|
|
|
|
|
commonKeyAttributes. */
|
|
|
|
|
struct keyaccess_flags_s
|
|
|
|
|
{
|
|
|
|
|
unsigned int any:1; /* Any access flag set. */
|
|
|
|
|
unsigned int sensitive:1;
|
|
|
|
|
unsigned int extractable:1;
|
|
|
|
|
unsigned int always_sensitive:1;
|
|
|
|
|
unsigned int never_extractable:1;
|
|
|
|
|
unsigned int local:1;
|
|
|
|
|
};
|
|
|
|
|
typedef struct keyaccess_flags_s keyaccess_flags_t;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* A bit array with for the gpg usage flags. */
|
|
|
|
|
struct gpgusage_flags_s
|
|
|
|
|
{
|
|
|
|
|
unsigned int any:1; /* Any of the next flags are set. */
|
|
|
|
|
unsigned int cert:1; /* 1.3.6.1.4.1.11591.2.6.1 */
|
|
|
|
|
unsigned int sign:1; /* 1.3.6.1.4.1.11591.2.6.2 */
|
|
|
|
|
unsigned int encr:1; /* 1.3.6.1.4.1.11591.2.6.3 */
|
|
|
|
|
unsigned int auth:1; /* 1.3.6.1.4.1.11591.2.6.4 */
|
|
|
|
|
};
|
|
|
|
|
typedef struct gpgusage_flags_s gpgusage_flags_t;
|
|
|
|
|
|
2005-04-27 12:09:21 +00:00
|
|
|
|
|
|
|
|
|
/* This is an object to store information about a Certificate
|
|
|
|
|
Directory File (CDF) in a format suitable for further processing by
|
|
|
|
|
us. To keep memory management, simple we use a linked list of
|
|
|
|
|
items; i.e. one such object represents one certificate and the list
|
|
|
|
|
the entire CDF. */
|
2011-02-04 12:57:53 +01:00
|
|
|
|
struct cdf_object_s
|
2005-04-27 12:09:21 +00:00
|
|
|
|
{
|
|
|
|
|
/* Link to next item when used in a linked list. */
|
2011-02-04 12:57:53 +01:00
|
|
|
|
struct cdf_object_s *next;
|
2005-04-27 12:09:21 +00:00
|
|
|
|
|
2020-03-31 19:55:15 +02:00
|
|
|
|
/* Flags to indicate whether fields are valid. */
|
|
|
|
|
unsigned int have_off:1;
|
|
|
|
|
|
2020-03-31 10:40:21 +02:00
|
|
|
|
/* Length and allocated buffer with the Id of this object.
|
|
|
|
|
* This field is used for X.509 in PKCS#11 to make it easier to
|
|
|
|
|
* match a private key with a certificate. */
|
2005-04-27 12:09:21 +00:00
|
|
|
|
size_t objidlen;
|
|
|
|
|
unsigned char *objid;
|
|
|
|
|
|
2021-04-29 15:26:36 +02:00
|
|
|
|
/* Length and allocated buffer with the authId of this object or
|
|
|
|
|
NULL if no authID is known. */
|
|
|
|
|
size_t authidlen;
|
|
|
|
|
unsigned char *authid;
|
|
|
|
|
|
|
|
|
|
/* NULL or the malloced label of this object. */
|
|
|
|
|
char *label;
|
|
|
|
|
|
|
|
|
|
/* To avoid reading and parsing a certificate more than once, we
|
|
|
|
|
* cache the ksba object. */
|
|
|
|
|
ksba_cert_t cert;
|
2011-02-04 12:57:53 +01:00
|
|
|
|
|
2005-04-27 12:09:21 +00:00
|
|
|
|
/* The offset and length of the object. They are only valid if
|
|
|
|
|
HAVE_OFF is true and set to 0 if HAVE_OFF is false. */
|
|
|
|
|
unsigned long off, len;
|
|
|
|
|
|
|
|
|
|
/* The length of the path as given in the CDF and the path itself.
|
|
|
|
|
path[0] is the top DF (usually 0x3f00). The path will never be
|
|
|
|
|
empty. */
|
|
|
|
|
size_t pathlen;
|
|
|
|
|
unsigned short path[1];
|
|
|
|
|
};
|
|
|
|
|
typedef struct cdf_object_s *cdf_object_t;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* This is an object to store information about a Private Key
|
|
|
|
|
Directory File (PrKDF) in a format suitable for further processing
|
|
|
|
|
by us. To keep memory management, simple we use a linked list of
|
|
|
|
|
items; i.e. one such object represents one certificate and the list
|
|
|
|
|
the entire PrKDF. */
|
2011-02-04 12:57:53 +01:00
|
|
|
|
struct prkdf_object_s
|
2005-04-27 12:09:21 +00:00
|
|
|
|
{
|
|
|
|
|
/* Link to next item when used in a linked list. */
|
2011-02-04 12:57:53 +01:00
|
|
|
|
struct prkdf_object_s *next;
|
2005-04-27 12:09:21 +00:00
|
|
|
|
|
2020-03-31 19:55:15 +02:00
|
|
|
|
/* Flags to indicate whether fields are valid. */
|
|
|
|
|
unsigned int keygrip_valid:1;
|
|
|
|
|
unsigned int key_reference_valid:1;
|
|
|
|
|
unsigned int have_off:1;
|
2021-04-29 15:26:36 +02:00
|
|
|
|
unsigned int have_keytime:1;
|
2020-03-31 19:55:15 +02:00
|
|
|
|
|
2020-04-01 20:31:21 +02:00
|
|
|
|
/* Flag indicating that the corresponding PIN has already been
|
2021-04-29 15:26:36 +02:00
|
|
|
|
* verified. Note that for cards which are able to return the
|
|
|
|
|
* verification stus, this flag is not used. */
|
2020-04-01 20:31:21 +02:00
|
|
|
|
unsigned int pin_verified:1;
|
|
|
|
|
|
2021-04-29 15:26:36 +02:00
|
|
|
|
/* PKCS#15 info whether this is an EC key. Default is RSA. Note
|
|
|
|
|
* that there is also a KEYALGO field which is derived from the
|
|
|
|
|
* publick key via Libgcrypt. */
|
|
|
|
|
unsigned int is_ecc:1;
|
|
|
|
|
|
2020-03-31 19:55:15 +02:00
|
|
|
|
/* The key's usage flags. */
|
|
|
|
|
keyusage_flags_t usageflags;
|
|
|
|
|
|
2021-04-29 15:26:36 +02:00
|
|
|
|
/* The key's access flags. */
|
|
|
|
|
keyaccess_flags_t accessflags;
|
|
|
|
|
|
|
|
|
|
/* Extended key usage flags. Only used if .valid is set. This
|
|
|
|
|
* information is computed from an associated certificate15. */
|
|
|
|
|
struct {
|
|
|
|
|
unsigned int valid:1;
|
|
|
|
|
unsigned int sign:1;
|
|
|
|
|
unsigned int encr:1;
|
|
|
|
|
unsigned int auth:1;
|
|
|
|
|
} extusage;
|
|
|
|
|
|
|
|
|
|
/* OpenPGP key features for this key. This is taken from special
|
|
|
|
|
* extended key usage flags different from those tracked in EXTUSAGE
|
|
|
|
|
* above. There is also no valid flag as in EXTUSAGE. */
|
|
|
|
|
gpgusage_flags_t gpgusage;
|
|
|
|
|
|
2020-03-31 19:55:15 +02:00
|
|
|
|
/* The keygrip of the key. This is used as a cache. */
|
|
|
|
|
char keygrip[2*KEYGRIP_LEN+1];
|
|
|
|
|
|
2021-04-29 15:26:36 +02:00
|
|
|
|
/* A malloced algorithm string or NULL if not known. */
|
|
|
|
|
char *keyalgostr;
|
|
|
|
|
|
2020-03-31 19:55:15 +02:00
|
|
|
|
/* The Gcrypt algo identifier for the key. It is valid if the
|
2021-04-29 15:26:36 +02:00
|
|
|
|
* keygrip is also valid. See also is_ecc above. */
|
2020-03-31 19:55:15 +02:00
|
|
|
|
int keyalgo;
|
|
|
|
|
|
|
|
|
|
/* The length of the key in bits (e.g. for RSA the length of the
|
|
|
|
|
* modulus). It is valid if the keygrip is also valid. */
|
|
|
|
|
unsigned int keynbits;
|
|
|
|
|
|
2021-04-29 15:26:36 +02:00
|
|
|
|
/* The creation time of the key or 0 if not known. */
|
|
|
|
|
u32 keytime;
|
|
|
|
|
|
2020-04-07 20:38:15 +02:00
|
|
|
|
/* Malloced CN from the Subject-DN of the corresponding certificate
|
|
|
|
|
* or NULL if not known. */
|
|
|
|
|
char *common_name;
|
|
|
|
|
|
|
|
|
|
/* Malloced SerialNumber from the Subject-DN of the corresponding
|
|
|
|
|
* certificate or NULL if not known. */
|
|
|
|
|
char *serial_number;
|
|
|
|
|
|
2021-04-29 15:26:36 +02:00
|
|
|
|
/* KDF/KEK parameter for OpenPGP's ECDH. First byte is zero if not
|
|
|
|
|
* availabale. .*/
|
|
|
|
|
unsigned char ecdh_kdf[4];
|
|
|
|
|
|
2005-04-27 12:09:21 +00:00
|
|
|
|
/* Length and allocated buffer with the Id of this object. */
|
|
|
|
|
size_t objidlen;
|
|
|
|
|
unsigned char *objid;
|
|
|
|
|
|
|
|
|
|
/* Length and allocated buffer with the authId of this object or
|
|
|
|
|
NULL if no authID is known. */
|
|
|
|
|
size_t authidlen;
|
|
|
|
|
unsigned char *authid;
|
|
|
|
|
|
2021-04-29 15:26:36 +02:00
|
|
|
|
/* NULL or the malloced label of this object. */
|
|
|
|
|
char *label;
|
|
|
|
|
|
2011-02-04 12:57:53 +01:00
|
|
|
|
/* The keyReference and a flag telling whether it is valid. */
|
2005-04-27 12:09:21 +00:00
|
|
|
|
unsigned long key_reference;
|
|
|
|
|
|
|
|
|
|
/* The offset and length of the object. They are only valid if
|
2020-03-31 19:55:15 +02:00
|
|
|
|
* HAVE_OFF is true otherwise they are set to 0. */
|
2005-04-27 12:09:21 +00:00
|
|
|
|
unsigned long off, len;
|
|
|
|
|
|
|
|
|
|
/* The length of the path as given in the PrKDF and the path itself.
|
|
|
|
|
path[0] is the top DF (usually 0x3f00). */
|
|
|
|
|
size_t pathlen;
|
|
|
|
|
unsigned short path[1];
|
|
|
|
|
};
|
|
|
|
|
typedef struct prkdf_object_s *prkdf_object_t;
|
2021-04-29 15:26:36 +02:00
|
|
|
|
typedef struct prkdf_object_s *pukdf_object_t;
|
2005-04-27 12:09:21 +00:00
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* This is an object to store information about a Authentication
|
|
|
|
|
Object Directory File (AODF) in a format suitable for further
|
|
|
|
|
processing by us. To keep memory management, simple we use a linked
|
|
|
|
|
list of items; i.e. one such object represents one authentication
|
|
|
|
|
object and the list the entire AOKDF. */
|
2011-02-04 12:57:53 +01:00
|
|
|
|
struct aodf_object_s
|
2005-04-27 12:09:21 +00:00
|
|
|
|
{
|
|
|
|
|
/* Link to next item when used in a linked list. */
|
2011-02-04 12:57:53 +01:00
|
|
|
|
struct aodf_object_s *next;
|
2005-04-27 12:09:21 +00:00
|
|
|
|
|
2020-03-31 19:55:15 +02:00
|
|
|
|
/* Flags to indicate whether fields are valid. */
|
|
|
|
|
unsigned int have_off:1;
|
|
|
|
|
|
2005-04-27 12:09:21 +00:00
|
|
|
|
/* Length and allocated buffer with the Id of this object. */
|
|
|
|
|
size_t objidlen;
|
|
|
|
|
unsigned char *objid;
|
|
|
|
|
|
|
|
|
|
/* Length and allocated buffer with the authId of this object or
|
|
|
|
|
NULL if no authID is known. */
|
|
|
|
|
size_t authidlen;
|
|
|
|
|
unsigned char *authid;
|
|
|
|
|
|
2021-04-29 15:26:36 +02:00
|
|
|
|
/* NULL or the malloced label of this object. */
|
|
|
|
|
char *label;
|
|
|
|
|
|
2020-03-31 19:55:15 +02:00
|
|
|
|
/* The file ID of this AODF. */
|
|
|
|
|
unsigned short fid;
|
|
|
|
|
|
2021-06-22 11:11:46 +02:00
|
|
|
|
/* The type of this authentication object. */
|
|
|
|
|
auth_type_t auth_type;
|
|
|
|
|
|
|
|
|
|
/* Info used for AUTH_TYPE_PIN: */
|
|
|
|
|
|
2005-04-27 12:09:21 +00:00
|
|
|
|
/* The PIN Flags. */
|
2011-02-04 12:57:53 +01:00
|
|
|
|
struct
|
2005-04-27 12:09:21 +00:00
|
|
|
|
{
|
|
|
|
|
unsigned int case_sensitive: 1;
|
|
|
|
|
unsigned int local: 1;
|
|
|
|
|
unsigned int change_disabled: 1;
|
|
|
|
|
unsigned int unblock_disabled: 1;
|
|
|
|
|
unsigned int initialized: 1;
|
|
|
|
|
unsigned int needs_padding: 1;
|
|
|
|
|
unsigned int unblocking_pin: 1;
|
|
|
|
|
unsigned int so_pin: 1;
|
|
|
|
|
unsigned int disable_allowed: 1;
|
|
|
|
|
unsigned int integrity_protected: 1;
|
|
|
|
|
unsigned int confidentiality_protected: 1;
|
|
|
|
|
unsigned int exchange_ref_data: 1;
|
|
|
|
|
} pinflags;
|
|
|
|
|
|
|
|
|
|
/* The PIN Type. */
|
|
|
|
|
pin_type_t pintype;
|
|
|
|
|
|
|
|
|
|
/* The minimum length of a PIN. */
|
|
|
|
|
unsigned long min_length;
|
|
|
|
|
|
|
|
|
|
/* The stored length of a PIN. */
|
|
|
|
|
unsigned long stored_length;
|
|
|
|
|
|
|
|
|
|
/* The maximum length of a PIN and a flag telling whether it is valid. */
|
|
|
|
|
unsigned long max_length;
|
|
|
|
|
int max_length_valid;
|
|
|
|
|
|
2011-02-04 12:57:53 +01:00
|
|
|
|
/* The pinReference and a flag telling whether it is valid. */
|
2005-04-27 12:09:21 +00:00
|
|
|
|
unsigned long pin_reference;
|
|
|
|
|
int pin_reference_valid;
|
|
|
|
|
|
|
|
|
|
/* The padChar and a flag telling whether it is valid. */
|
|
|
|
|
char pad_char;
|
|
|
|
|
int pad_char_valid;
|
|
|
|
|
|
|
|
|
|
/* The offset and length of the object. They are only valid if
|
|
|
|
|
HAVE_OFF is true and set to 0 if HAVE_OFF is false. */
|
|
|
|
|
unsigned long off, len;
|
|
|
|
|
|
|
|
|
|
/* The length of the path as given in the Aodf and the path itself.
|
|
|
|
|
path[0] is the top DF (usually 0x3f00). PATH is optional and thus
|
|
|
|
|
may be NULL. Malloced.*/
|
|
|
|
|
size_t pathlen;
|
|
|
|
|
unsigned short *path;
|
2021-06-22 11:11:46 +02:00
|
|
|
|
|
|
|
|
|
/* Info used for AUTH_TYPE_AUTHKEY: */
|
|
|
|
|
|
2005-04-27 12:09:21 +00:00
|
|
|
|
};
|
|
|
|
|
typedef struct aodf_object_s *aodf_object_t;
|
2004-12-02 07:48:09 +00:00
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* Context local to this application. */
|
2011-02-04 12:57:53 +01:00
|
|
|
|
struct app_local_s
|
2004-12-02 07:48:09 +00:00
|
|
|
|
{
|
2005-04-27 12:09:21 +00:00
|
|
|
|
/* The home DF. Note, that we don't yet support a multilevel
|
2015-11-16 12:41:46 +01:00
|
|
|
|
hierarchy. Thus we assume this is directly below the MF. */
|
2005-04-27 12:09:21 +00:00
|
|
|
|
unsigned short home_df;
|
|
|
|
|
|
2020-04-07 16:00:11 +02:00
|
|
|
|
/* The type of the card's OS. */
|
2005-04-27 12:09:21 +00:00
|
|
|
|
card_type_t card_type;
|
|
|
|
|
|
2020-04-07 16:00:11 +02:00
|
|
|
|
/* The vendor's product. */
|
|
|
|
|
card_product_t card_product;
|
|
|
|
|
|
2022-04-11 17:48:45 +02:00
|
|
|
|
/* Flag indicating that extended_mode is not supported. */
|
2021-06-18 17:32:01 +02:00
|
|
|
|
unsigned int no_extended_mode : 1;
|
|
|
|
|
|
2005-09-05 14:36:36 +00:00
|
|
|
|
/* Flag indicating whether we may use direct path selection. */
|
2021-06-18 17:32:01 +02:00
|
|
|
|
unsigned int direct_path_selection : 1;
|
2005-09-05 14:36:36 +00:00
|
|
|
|
|
2021-04-29 15:26:36 +02:00
|
|
|
|
/* Flag indicating whether the card has any key with a gpgusage set. */
|
2021-06-18 17:32:01 +02:00
|
|
|
|
unsigned int any_gpgusage : 1;
|
2021-04-29 15:26:36 +02:00
|
|
|
|
|
2005-04-27 12:09:21 +00:00
|
|
|
|
/* Structure with the EFIDs of the objects described in the ODF
|
|
|
|
|
file. */
|
2004-12-02 07:48:09 +00:00
|
|
|
|
struct
|
|
|
|
|
{
|
|
|
|
|
unsigned short private_keys;
|
|
|
|
|
unsigned short public_keys;
|
|
|
|
|
unsigned short trusted_public_keys;
|
|
|
|
|
unsigned short secret_keys;
|
|
|
|
|
unsigned short certificates;
|
|
|
|
|
unsigned short trusted_certificates;
|
|
|
|
|
unsigned short useful_certificates;
|
|
|
|
|
unsigned short data_objects;
|
|
|
|
|
unsigned short auth_objects;
|
2011-02-04 12:57:53 +01:00
|
|
|
|
} odf;
|
2004-12-02 07:48:09 +00:00
|
|
|
|
|
2005-09-05 14:36:36 +00:00
|
|
|
|
/* The PKCS#15 serialnumber from EF(TokeiNFo) or NULL. Malloced. */
|
|
|
|
|
unsigned char *serialno;
|
|
|
|
|
size_t serialnolen;
|
|
|
|
|
|
2021-04-29 15:26:36 +02:00
|
|
|
|
/* The manufacturerID from the TokenInfo EF. Malloced or NULL. */
|
2020-04-03 11:21:32 +02:00
|
|
|
|
char *manufacturer_id;
|
|
|
|
|
|
2021-04-29 15:26:36 +02:00
|
|
|
|
/* The label from the TokenInfo EF. Malloced or NULL. */
|
|
|
|
|
char *token_label;
|
|
|
|
|
|
|
|
|
|
/* The tokenflags from the TokenInfo EF. Malloced or NULL. */
|
|
|
|
|
unsigned char *tokenflags;
|
|
|
|
|
unsigned int tokenflagslen;
|
|
|
|
|
|
2005-04-27 12:09:21 +00:00
|
|
|
|
/* Information on all certificates. */
|
|
|
|
|
cdf_object_t certificate_info;
|
|
|
|
|
/* Information on all trusted certificates. */
|
|
|
|
|
cdf_object_t trusted_certificate_info;
|
|
|
|
|
/* Information on all useful certificates. */
|
|
|
|
|
cdf_object_t useful_certificate_info;
|
|
|
|
|
|
2021-04-29 15:26:36 +02:00
|
|
|
|
/* Information on all public keys. */
|
|
|
|
|
prkdf_object_t public_key_info;
|
|
|
|
|
|
2005-04-27 12:09:21 +00:00
|
|
|
|
/* Information on all private keys. */
|
2021-04-29 15:26:36 +02:00
|
|
|
|
pukdf_object_t private_key_info;
|
2005-04-27 12:09:21 +00:00
|
|
|
|
|
|
|
|
|
/* Information on all authentication objects. */
|
|
|
|
|
aodf_object_t auth_object_info;
|
2004-12-02 07:48:09 +00:00
|
|
|
|
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
2005-04-27 12:09:21 +00:00
|
|
|
|
/*** Local prototypes. ***/
|
2021-04-29 15:26:36 +02:00
|
|
|
|
static gpg_error_t select_ef_by_path (app_t app, const unsigned short *path,
|
|
|
|
|
size_t pathlen);
|
2020-03-31 19:55:15 +02:00
|
|
|
|
static gpg_error_t keygrip_from_prkdf (app_t app, prkdf_object_t prkdf);
|
2005-04-27 12:09:21 +00:00
|
|
|
|
static gpg_error_t readcert_by_cdf (app_t app, cdf_object_t cdf,
|
|
|
|
|
unsigned char **r_cert, size_t *r_certlen);
|
2020-04-08 07:54:01 +02:00
|
|
|
|
static char *get_dispserialno (app_t app, prkdf_object_t prkdf);
|
2020-04-03 11:21:32 +02:00
|
|
|
|
static gpg_error_t do_getattr (app_t app, ctrl_t ctrl, const char *name);
|
2005-04-27 12:09:21 +00:00
|
|
|
|
|
|
|
|
|
|
2021-04-29 15:26:36 +02:00
|
|
|
|
|
|
|
|
|
static const char *
|
|
|
|
|
cardtype2str (card_type_t cardtype)
|
|
|
|
|
{
|
|
|
|
|
switch (cardtype)
|
|
|
|
|
{
|
|
|
|
|
case CARD_TYPE_UNKNOWN: return "";
|
|
|
|
|
case CARD_TYPE_TCOS: return "TCOS";
|
|
|
|
|
case CARD_TYPE_MICARDO: return "Micardo";
|
|
|
|
|
case CARD_TYPE_CARDOS_50: return "CardOS 5.0";
|
|
|
|
|
case CARD_TYPE_CARDOS_53: return "CardOS 5.3";
|
|
|
|
|
case CARD_TYPE_BELPIC: return "Belgian eID";
|
2021-06-18 17:32:01 +02:00
|
|
|
|
case CARD_TYPE_AET: return "AET";
|
2021-04-29 15:26:36 +02:00
|
|
|
|
}
|
|
|
|
|
return "";
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static const char *
|
|
|
|
|
cardproduct2str (card_product_t cardproduct)
|
|
|
|
|
{
|
|
|
|
|
switch (cardproduct)
|
|
|
|
|
{
|
|
|
|
|
case CARD_PRODUCT_UNKNOWN: return "";
|
|
|
|
|
case CARD_PRODUCT_RSCS: return "R&S";
|
|
|
|
|
case CARD_PRODUCT_DTRUST: return "D-Trust";
|
2022-04-13 13:06:27 +02:00
|
|
|
|
case CARD_PRODUCT_GENUA: return "GeNUA";
|
2021-04-29 15:26:36 +02:00
|
|
|
|
}
|
|
|
|
|
return "";
|
|
|
|
|
}
|
2005-04-27 12:09:21 +00:00
|
|
|
|
|
|
|
|
|
/* Release the CDF object A */
|
|
|
|
|
static void
|
|
|
|
|
release_cdflist (cdf_object_t a)
|
|
|
|
|
{
|
|
|
|
|
while (a)
|
|
|
|
|
{
|
|
|
|
|
cdf_object_t tmp = a->next;
|
2021-04-29 15:26:36 +02:00
|
|
|
|
ksba_free (a->cert);
|
2005-04-27 12:09:21 +00:00
|
|
|
|
xfree (a->objid);
|
2021-04-29 15:26:36 +02:00
|
|
|
|
xfree (a->authid);
|
|
|
|
|
xfree (a->label);
|
2005-04-27 12:09:21 +00:00
|
|
|
|
xfree (a);
|
|
|
|
|
a = tmp;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Release the PrKDF object A. */
|
|
|
|
|
static void
|
|
|
|
|
release_prkdflist (prkdf_object_t a)
|
|
|
|
|
{
|
|
|
|
|
while (a)
|
|
|
|
|
{
|
|
|
|
|
prkdf_object_t tmp = a->next;
|
2021-04-29 15:26:36 +02:00
|
|
|
|
xfree (a->keyalgostr);
|
2020-04-07 20:38:15 +02:00
|
|
|
|
xfree (a->common_name);
|
|
|
|
|
xfree (a->serial_number);
|
2005-04-27 12:09:21 +00:00
|
|
|
|
xfree (a->objid);
|
|
|
|
|
xfree (a->authid);
|
2021-04-29 15:26:36 +02:00
|
|
|
|
xfree (a->label);
|
2005-04-27 12:09:21 +00:00
|
|
|
|
xfree (a);
|
|
|
|
|
a = tmp;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2021-04-29 15:26:36 +02:00
|
|
|
|
static void
|
|
|
|
|
release_pukdflist (pukdf_object_t a)
|
|
|
|
|
{
|
|
|
|
|
release_prkdflist (a);
|
|
|
|
|
}
|
|
|
|
|
|
2005-04-27 12:09:21 +00:00
|
|
|
|
/* Release just one aodf object. */
|
|
|
|
|
void
|
|
|
|
|
release_aodf_object (aodf_object_t a)
|
|
|
|
|
{
|
|
|
|
|
if (a)
|
|
|
|
|
{
|
|
|
|
|
xfree (a->objid);
|
|
|
|
|
xfree (a->authid);
|
2021-04-29 15:26:36 +02:00
|
|
|
|
xfree (a->label);
|
2005-04-27 12:09:21 +00:00
|
|
|
|
xfree (a->path);
|
|
|
|
|
xfree (a);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Release the AODF list A. */
|
|
|
|
|
static void
|
|
|
|
|
release_aodflist (aodf_object_t a)
|
|
|
|
|
{
|
|
|
|
|
while (a)
|
|
|
|
|
{
|
|
|
|
|
aodf_object_t tmp = a->next;
|
|
|
|
|
release_aodf_object (a);
|
|
|
|
|
a = tmp;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2021-04-29 15:26:36 +02:00
|
|
|
|
static void
|
|
|
|
|
release_lists (app_t app)
|
|
|
|
|
{
|
|
|
|
|
release_cdflist (app->app_local->certificate_info);
|
|
|
|
|
app->app_local->certificate_info = NULL;
|
|
|
|
|
release_cdflist (app->app_local->trusted_certificate_info);
|
|
|
|
|
app->app_local->trusted_certificate_info = NULL;
|
|
|
|
|
release_cdflist (app->app_local->useful_certificate_info);
|
|
|
|
|
app->app_local->useful_certificate_info = NULL;
|
|
|
|
|
release_pukdflist (app->app_local->public_key_info);
|
|
|
|
|
app->app_local->public_key_info = NULL;
|
|
|
|
|
release_prkdflist (app->app_local->private_key_info);
|
|
|
|
|
app->app_local->private_key_info = NULL;
|
|
|
|
|
release_aodflist (app->app_local->auth_object_info);
|
|
|
|
|
app->app_local->auth_object_info = NULL;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
static void
|
|
|
|
|
release_tokeninfo (app_t app)
|
|
|
|
|
{
|
|
|
|
|
xfree (app->app_local->manufacturer_id);
|
|
|
|
|
app->app_local->manufacturer_id = NULL;
|
|
|
|
|
xfree (app->app_local->token_label);
|
|
|
|
|
app->app_local->token_label = NULL;
|
|
|
|
|
xfree (app->app_local->tokenflags);
|
|
|
|
|
app->app_local->tokenflags = NULL;
|
|
|
|
|
xfree (app->app_local->serialno);
|
|
|
|
|
app->app_local->serialno = NULL;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2005-04-27 12:09:21 +00:00
|
|
|
|
/* Release all local resources. */
|
|
|
|
|
static void
|
|
|
|
|
do_deinit (app_t app)
|
|
|
|
|
{
|
|
|
|
|
if (app && app->app_local)
|
|
|
|
|
{
|
2021-04-29 15:26:36 +02:00
|
|
|
|
release_lists (app);
|
|
|
|
|
release_tokeninfo (app);
|
2005-04-27 12:09:21 +00:00
|
|
|
|
xfree (app->app_local);
|
|
|
|
|
app->app_local = NULL;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2004-12-02 07:48:09 +00:00
|
|
|
|
|
2005-04-27 12:09:21 +00:00
|
|
|
|
/* Do a select and a read for the file with EFID. EFID_DESC is a
|
2004-12-02 07:48:09 +00:00
|
|
|
|
desctription of the EF to be used with error messages. On success
|
|
|
|
|
BUFFER and BUFLEN contain the entire content of the EF. The caller
|
2021-04-29 15:26:36 +02:00
|
|
|
|
must free BUFFER only on success. If EFID is 0 no seelct is done. */
|
2011-02-04 12:57:53 +01:00
|
|
|
|
static gpg_error_t
|
2021-04-29 15:26:36 +02:00
|
|
|
|
select_and_read_binary (app_t app, unsigned short efid, const char *efid_desc,
|
2004-12-02 07:48:09 +00:00
|
|
|
|
unsigned char **buffer, size_t *buflen)
|
|
|
|
|
{
|
|
|
|
|
gpg_error_t err;
|
2021-04-29 15:26:36 +02:00
|
|
|
|
int sw;
|
|
|
|
|
|
|
|
|
|
if (efid)
|
|
|
|
|
{
|
|
|
|
|
err = select_ef_by_path (app, &efid, 1);
|
|
|
|
|
if (err)
|
|
|
|
|
{
|
|
|
|
|
log_error ("p15: error selecting %s (0x%04X): %s\n",
|
|
|
|
|
efid_desc, efid, gpg_strerror (err));
|
|
|
|
|
return err;
|
|
|
|
|
}
|
|
|
|
|
}
|
2004-12-02 07:48:09 +00:00
|
|
|
|
|
2021-04-29 15:26:36 +02:00
|
|
|
|
err = iso7816_read_binary_ext (app_get_slot (app),
|
|
|
|
|
0, 0, 0, buffer, buflen, &sw);
|
2004-12-02 07:48:09 +00:00
|
|
|
|
if (err)
|
2021-04-29 15:26:36 +02:00
|
|
|
|
log_error ("p15: error reading %s (0x%04X): %s (sw=%04X)\n",
|
|
|
|
|
efid_desc, efid, gpg_strerror (err), sw);
|
|
|
|
|
return err;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* If EFID is not 0 do a select and then read the record RECNO.
|
|
|
|
|
* EFID_DESC is a description of the EF to be used with error
|
|
|
|
|
* messages. On success BUFFER and BUFLEN contain the entire content
|
|
|
|
|
* of the EF. The caller must free BUFFER only on success. */
|
|
|
|
|
static gpg_error_t
|
|
|
|
|
select_and_read_record (app_t app, unsigned short efid, int recno,
|
|
|
|
|
const char *efid_desc,
|
|
|
|
|
unsigned char **buffer, size_t *buflen, int *r_sw)
|
|
|
|
|
{
|
|
|
|
|
gpg_error_t err;
|
|
|
|
|
int sw;
|
|
|
|
|
|
|
|
|
|
if (r_sw)
|
|
|
|
|
*r_sw = 0x9000;
|
|
|
|
|
|
|
|
|
|
if (efid)
|
2004-12-02 07:48:09 +00:00
|
|
|
|
{
|
2021-04-29 15:26:36 +02:00
|
|
|
|
err = select_ef_by_path (app, &efid, 1);
|
|
|
|
|
if (err)
|
|
|
|
|
{
|
|
|
|
|
log_error ("p15: error selecting %s (0x%04X): %s\n",
|
|
|
|
|
efid_desc, efid, gpg_strerror (err));
|
|
|
|
|
if (r_sw)
|
|
|
|
|
*r_sw = sw;
|
|
|
|
|
return err;
|
|
|
|
|
}
|
2004-12-02 07:48:09 +00:00
|
|
|
|
}
|
2021-04-29 15:26:36 +02:00
|
|
|
|
|
|
|
|
|
err = iso7816_read_record_ext (app_get_slot (app),
|
|
|
|
|
recno, 1, 0, buffer, buflen, &sw);
|
2004-12-02 07:48:09 +00:00
|
|
|
|
if (err)
|
|
|
|
|
{
|
2021-04-29 15:26:36 +02:00
|
|
|
|
if (gpg_err_code (err) == GPG_ERR_NOT_FOUND)
|
|
|
|
|
;
|
|
|
|
|
else if (err && sw == SW_FILE_STRUCT)
|
|
|
|
|
;
|
|
|
|
|
else
|
|
|
|
|
log_error ("p15: error reading %s (0x%04X) record %d: %s (sw=%04X)\n",
|
|
|
|
|
efid_desc, efid, recno, gpg_strerror (err), sw);
|
|
|
|
|
if (r_sw)
|
|
|
|
|
*r_sw = sw;
|
2004-12-02 07:48:09 +00:00
|
|
|
|
return err;
|
|
|
|
|
}
|
2021-04-29 15:26:36 +02:00
|
|
|
|
/* On CardOS with a Linear TLV file structure the records starts
|
|
|
|
|
* with some tag (often the record number) followed by the length
|
|
|
|
|
* byte for this record. Detect and remove this prefix. */
|
|
|
|
|
if (*buflen > 2 && (*buffer)[0] != 0x30 && (*buffer)[1] == *buflen - 2)
|
|
|
|
|
{
|
|
|
|
|
memmove (*buffer, *buffer + 2, *buflen - 2);
|
|
|
|
|
*buflen = *buflen - 2;
|
|
|
|
|
}
|
|
|
|
|
|
2004-12-02 07:48:09 +00:00
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2005-04-27 19:47:53 +00:00
|
|
|
|
/* This function calls select file to read a file using a complete
|
2011-02-04 12:57:53 +01:00
|
|
|
|
path which may or may not start at the master file (MF). */
|
2005-04-27 12:09:21 +00:00
|
|
|
|
static gpg_error_t
|
|
|
|
|
select_ef_by_path (app_t app, const unsigned short *path, size_t pathlen)
|
|
|
|
|
{
|
|
|
|
|
gpg_error_t err;
|
|
|
|
|
int i, j;
|
|
|
|
|
|
2005-09-05 14:36:36 +00:00
|
|
|
|
if (!pathlen)
|
|
|
|
|
return gpg_error (GPG_ERR_INV_VALUE);
|
|
|
|
|
|
2021-04-29 15:26:36 +02:00
|
|
|
|
/* log_debug ("%s: path=", __func__); */
|
|
|
|
|
/* for (j=0; j < pathlen; j++) */
|
|
|
|
|
/* log_printf ("%s%04hX", j? "/":"", path[j]); */
|
|
|
|
|
/* log_printf ("%s\n",app->app_local->direct_path_selection?" (direct)":"");*/
|
2011-02-04 12:57:53 +01:00
|
|
|
|
|
2005-09-05 14:36:36 +00:00
|
|
|
|
if (app->app_local->direct_path_selection)
|
2005-04-27 12:09:21 +00:00
|
|
|
|
{
|
2021-04-29 15:26:36 +02:00
|
|
|
|
if (pathlen && *path == 0x3f00 )
|
|
|
|
|
{
|
|
|
|
|
if (pathlen == 1)
|
|
|
|
|
err = iso7816_select_mf (app_get_slot (app));
|
|
|
|
|
else
|
|
|
|
|
err = iso7816_select_path (app_get_slot (app), path+1, pathlen-1,
|
|
|
|
|
0);
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
err = iso7816_select_path (app_get_slot (app), path, pathlen,
|
|
|
|
|
app->app_local->home_df);
|
2005-04-27 12:09:21 +00:00
|
|
|
|
if (err)
|
|
|
|
|
{
|
2020-03-31 10:40:21 +02:00
|
|
|
|
log_error ("p15: error selecting path ");
|
2021-04-29 15:26:36 +02:00
|
|
|
|
goto err_print_path;
|
2005-04-27 12:09:21 +00:00
|
|
|
|
}
|
|
|
|
|
}
|
2021-06-18 17:32:01 +02:00
|
|
|
|
else if (pathlen > 1 && path[0] == 0x3fff)
|
|
|
|
|
{
|
|
|
|
|
err = iso7816_select_file (app_get_slot (app), 0x3f00, 0);
|
|
|
|
|
if (err)
|
|
|
|
|
{
|
|
|
|
|
log_error ("p15: error selecting part %d from path ", 0);
|
|
|
|
|
goto err_print_path;
|
|
|
|
|
}
|
|
|
|
|
path++;
|
|
|
|
|
pathlen--;
|
|
|
|
|
for (i=0; i < pathlen; i++)
|
|
|
|
|
{
|
|
|
|
|
err = iso7816_select_file (app_get_slot (app),
|
|
|
|
|
path[i], (i+1 == pathlen)? 2 : 1);
|
|
|
|
|
if (err)
|
|
|
|
|
{
|
|
|
|
|
log_error ("p15: error selecting part %d from path ", i);
|
|
|
|
|
goto err_print_path;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
2005-09-05 14:36:36 +00:00
|
|
|
|
else
|
|
|
|
|
{
|
2021-04-29 15:26:36 +02:00
|
|
|
|
if (pathlen && *path != 0x3f00 )
|
|
|
|
|
log_error ("p15: warning: relative path select not yet implemented\n");
|
|
|
|
|
|
|
|
|
|
/* FIXME: Use home_df. */
|
2005-09-05 14:36:36 +00:00
|
|
|
|
for (i=0; i < pathlen; i++)
|
|
|
|
|
{
|
2021-04-29 15:26:36 +02:00
|
|
|
|
err = iso7816_select_file (app_get_slot (app),
|
|
|
|
|
path[i], !(i+1 == pathlen));
|
2005-09-05 14:36:36 +00:00
|
|
|
|
if (err)
|
|
|
|
|
{
|
2020-03-31 10:40:21 +02:00
|
|
|
|
log_error ("p15: error selecting part %d from path ", i);
|
2021-04-29 15:26:36 +02:00
|
|
|
|
goto err_print_path;
|
2005-09-05 14:36:36 +00:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
2005-04-27 12:09:21 +00:00
|
|
|
|
return 0;
|
2021-04-29 15:26:36 +02:00
|
|
|
|
|
|
|
|
|
err_print_path:
|
|
|
|
|
if (pathlen && *path != 0x3f00 )
|
|
|
|
|
log_printf ("3F00/");
|
|
|
|
|
else
|
|
|
|
|
log_printf ("%04hX/", app->app_local->home_df);
|
|
|
|
|
for (j=0; j < pathlen; j++)
|
|
|
|
|
log_printf ("%s%04hX", j? "/":"", path[j]);
|
|
|
|
|
log_printf (": %s\n", gpg_strerror (err));
|
|
|
|
|
return err;
|
2005-04-27 12:09:21 +00:00
|
|
|
|
}
|
|
|
|
|
|
2021-04-29 15:26:36 +02:00
|
|
|
|
|
2005-04-27 12:09:21 +00:00
|
|
|
|
/* Parse a cert Id string (or a key Id string) and return the binary
|
|
|
|
|
object Id string in a newly allocated buffer stored at R_OBJID and
|
|
|
|
|
R_OBJIDLEN. On Error NULL will be stored there and an error code
|
|
|
|
|
returned. On success caller needs to free the buffer at R_OBJID. */
|
|
|
|
|
static gpg_error_t
|
|
|
|
|
parse_certid (app_t app, const char *certid,
|
|
|
|
|
unsigned char **r_objid, size_t *r_objidlen)
|
|
|
|
|
{
|
|
|
|
|
char tmpbuf[10];
|
|
|
|
|
const char *s;
|
|
|
|
|
size_t objidlen;
|
|
|
|
|
unsigned char *objid;
|
|
|
|
|
int i;
|
|
|
|
|
|
|
|
|
|
*r_objid = NULL;
|
|
|
|
|
*r_objidlen = 0;
|
|
|
|
|
|
2020-03-31 19:55:15 +02:00
|
|
|
|
if (certid[0] != 'P' && strlen (certid) == 40) /* This is a keygrip. */
|
|
|
|
|
{
|
2020-04-02 12:39:56 +02:00
|
|
|
|
prkdf_object_t prkdf;
|
2020-03-31 19:55:15 +02:00
|
|
|
|
|
2020-04-02 12:39:56 +02:00
|
|
|
|
for (prkdf = app->app_local->private_key_info;
|
|
|
|
|
prkdf; prkdf = prkdf->next)
|
|
|
|
|
if (!keygrip_from_prkdf (app, prkdf)
|
|
|
|
|
&& !strcmp (certid, prkdf->keygrip))
|
2020-03-31 19:55:15 +02:00
|
|
|
|
break;
|
2020-04-02 12:39:56 +02:00
|
|
|
|
if (!prkdf || !prkdf->objidlen || !prkdf->objid)
|
2005-04-27 12:09:21 +00:00
|
|
|
|
return gpg_error (GPG_ERR_NOT_FOUND);
|
2020-04-02 12:39:56 +02:00
|
|
|
|
objidlen = prkdf->objidlen;
|
2020-03-31 19:55:15 +02:00
|
|
|
|
objid = xtrymalloc (objidlen);
|
|
|
|
|
if (!objid)
|
|
|
|
|
return gpg_error_from_syserror ();
|
2020-04-02 12:39:56 +02:00
|
|
|
|
memcpy (objid, prkdf->objid, prkdf->objidlen);
|
2020-03-31 19:55:15 +02:00
|
|
|
|
}
|
|
|
|
|
else /* This is a usual keyref. */
|
|
|
|
|
{
|
2021-04-29 15:26:36 +02:00
|
|
|
|
if (app->app_local->home_df != DEFAULT_HOME_DF)
|
2020-03-31 19:55:15 +02:00
|
|
|
|
snprintf (tmpbuf, sizeof tmpbuf, "P15-%04X.",
|
|
|
|
|
(unsigned int)(app->app_local->home_df & 0xffff));
|
|
|
|
|
else
|
|
|
|
|
strcpy (tmpbuf, "P15.");
|
|
|
|
|
if (strncmp (certid, tmpbuf, strlen (tmpbuf)) )
|
|
|
|
|
{
|
|
|
|
|
if (!strncmp (certid, "P15.", 4)
|
|
|
|
|
|| (!strncmp (certid, "P15-", 4)
|
|
|
|
|
&& hexdigitp (certid+4)
|
|
|
|
|
&& hexdigitp (certid+5)
|
|
|
|
|
&& hexdigitp (certid+6)
|
|
|
|
|
&& hexdigitp (certid+7)
|
|
|
|
|
&& certid[8] == '.'))
|
|
|
|
|
return gpg_error (GPG_ERR_NOT_FOUND);
|
|
|
|
|
return gpg_error (GPG_ERR_INV_ID);
|
|
|
|
|
}
|
|
|
|
|
certid += strlen (tmpbuf);
|
|
|
|
|
for (s=certid, objidlen=0; hexdigitp (s); s++, objidlen++)
|
|
|
|
|
;
|
|
|
|
|
if (*s || !objidlen || (objidlen%2))
|
|
|
|
|
return gpg_error (GPG_ERR_INV_ID);
|
|
|
|
|
objidlen /= 2;
|
|
|
|
|
objid = xtrymalloc (objidlen);
|
|
|
|
|
if (!objid)
|
|
|
|
|
return gpg_error_from_syserror ();
|
|
|
|
|
for (s=certid, i=0; i < objidlen; i++, s+=2)
|
|
|
|
|
objid[i] = xtoi_2 (s);
|
|
|
|
|
}
|
|
|
|
|
|
2005-04-27 12:09:21 +00:00
|
|
|
|
*r_objid = objid;
|
|
|
|
|
*r_objidlen = objidlen;
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2021-04-29 15:26:36 +02:00
|
|
|
|
/* Find a certificate object by its object ID and store a pointer to
|
|
|
|
|
* it at R_CDF. */
|
2005-04-27 12:09:21 +00:00
|
|
|
|
static gpg_error_t
|
2021-04-29 15:26:36 +02:00
|
|
|
|
cdf_object_from_objid (app_t app, size_t objidlen, const unsigned char *objid,
|
|
|
|
|
cdf_object_t *r_cdf)
|
2005-04-27 12:09:21 +00:00
|
|
|
|
{
|
|
|
|
|
cdf_object_t cdf;
|
|
|
|
|
|
|
|
|
|
for (cdf = app->app_local->certificate_info; cdf; cdf = cdf->next)
|
|
|
|
|
if (cdf->objidlen == objidlen && !memcmp (cdf->objid, objid, objidlen))
|
|
|
|
|
break;
|
|
|
|
|
if (!cdf)
|
|
|
|
|
for (cdf = app->app_local->trusted_certificate_info; cdf; cdf = cdf->next)
|
|
|
|
|
if (cdf->objidlen == objidlen && !memcmp (cdf->objid, objid, objidlen))
|
|
|
|
|
break;
|
|
|
|
|
if (!cdf)
|
|
|
|
|
for (cdf = app->app_local->useful_certificate_info; cdf; cdf = cdf->next)
|
|
|
|
|
if (cdf->objidlen == objidlen && !memcmp (cdf->objid, objid, objidlen))
|
|
|
|
|
break;
|
|
|
|
|
if (!cdf)
|
|
|
|
|
return gpg_error (GPG_ERR_NOT_FOUND);
|
|
|
|
|
*r_cdf = cdf;
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2021-04-29 15:26:36 +02:00
|
|
|
|
/* Find a certificate object by its label and store a pointer to it at
|
|
|
|
|
* R_CDF. */
|
|
|
|
|
static gpg_error_t
|
|
|
|
|
cdf_object_from_label (app_t app, const char *label, cdf_object_t *r_cdf)
|
|
|
|
|
{
|
|
|
|
|
cdf_object_t cdf;
|
|
|
|
|
|
|
|
|
|
if (!label)
|
|
|
|
|
return gpg_error (GPG_ERR_NOT_FOUND);
|
|
|
|
|
|
|
|
|
|
for (cdf = app->app_local->certificate_info; cdf; cdf = cdf->next)
|
|
|
|
|
if (cdf->label && !strcmp (cdf->label, label))
|
|
|
|
|
break;
|
|
|
|
|
if (!cdf)
|
|
|
|
|
for (cdf = app->app_local->trusted_certificate_info; cdf; cdf = cdf->next)
|
|
|
|
|
if (cdf->label && !strcmp (cdf->label, label))
|
|
|
|
|
break;
|
|
|
|
|
if (!cdf)
|
|
|
|
|
for (cdf = app->app_local->useful_certificate_info; cdf; cdf = cdf->next)
|
|
|
|
|
if (cdf->label && !strcmp (cdf->label, label))
|
|
|
|
|
break;
|
|
|
|
|
if (!cdf)
|
|
|
|
|
return gpg_error (GPG_ERR_NOT_FOUND);
|
|
|
|
|
*r_cdf = cdf;
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* Find a certificate object by the certificate ID CERTID and store a
|
|
|
|
|
* pointer to it at R_CDF. */
|
|
|
|
|
static gpg_error_t
|
|
|
|
|
cdf_object_from_certid (app_t app, const char *certid, cdf_object_t *r_cdf)
|
|
|
|
|
{
|
|
|
|
|
gpg_error_t err;
|
|
|
|
|
size_t objidlen;
|
|
|
|
|
unsigned char *objid;
|
|
|
|
|
cdf_object_t cdf;
|
|
|
|
|
prkdf_object_t prkdf;
|
|
|
|
|
|
|
|
|
|
err = parse_certid (app, certid, &objid, &objidlen);
|
|
|
|
|
if (err)
|
|
|
|
|
return err;
|
|
|
|
|
|
|
|
|
|
err = cdf_object_from_objid (app, objidlen, objid, &cdf);
|
|
|
|
|
if (gpg_err_code (err) == GPG_ERR_NOT_FOUND)
|
|
|
|
|
{
|
|
|
|
|
/* Try again by finding the certid in the prkdf and matching by
|
|
|
|
|
* label. */
|
|
|
|
|
for (prkdf = app->app_local->private_key_info; prkdf; prkdf = prkdf->next)
|
|
|
|
|
if (prkdf->objidlen == objidlen
|
|
|
|
|
&& !memcmp (prkdf->objid, objid, objidlen))
|
|
|
|
|
break;
|
|
|
|
|
if (prkdf)
|
|
|
|
|
err = cdf_object_from_label (app, prkdf->label, &cdf);
|
|
|
|
|
}
|
|
|
|
|
xfree (objid);
|
|
|
|
|
if (err)
|
|
|
|
|
return err;
|
|
|
|
|
*r_cdf = cdf;
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2005-04-27 12:09:21 +00:00
|
|
|
|
/* Find a private key object by the key Id string KEYIDSTR and store a
|
|
|
|
|
pointer to it at R_PRKDF. */
|
|
|
|
|
static gpg_error_t
|
|
|
|
|
prkdf_object_from_keyidstr (app_t app, const char *keyidstr,
|
|
|
|
|
prkdf_object_t *r_prkdf)
|
|
|
|
|
{
|
|
|
|
|
gpg_error_t err;
|
|
|
|
|
size_t objidlen;
|
|
|
|
|
unsigned char *objid;
|
|
|
|
|
prkdf_object_t prkdf;
|
|
|
|
|
|
|
|
|
|
err = parse_certid (app, keyidstr, &objid, &objidlen);
|
|
|
|
|
if (err)
|
|
|
|
|
return err;
|
|
|
|
|
|
|
|
|
|
for (prkdf = app->app_local->private_key_info; prkdf; prkdf = prkdf->next)
|
|
|
|
|
if (prkdf->objidlen == objidlen && !memcmp (prkdf->objid, objid, objidlen))
|
|
|
|
|
break;
|
|
|
|
|
xfree (objid);
|
|
|
|
|
if (!prkdf)
|
|
|
|
|
return gpg_error (GPG_ERR_NOT_FOUND);
|
|
|
|
|
*r_prkdf = prkdf;
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
2004-12-02 07:48:09 +00:00
|
|
|
|
|
|
|
|
|
|
2005-04-27 12:09:21 +00:00
|
|
|
|
|
2004-12-02 07:48:09 +00:00
|
|
|
|
/* Read and parse the Object Directory File and store away the
|
2005-04-27 12:09:21 +00:00
|
|
|
|
pointers. ODF_FID shall contain the FID of the ODF.
|
2004-12-02 07:48:09 +00:00
|
|
|
|
|
|
|
|
|
Example of such a file:
|
|
|
|
|
|
|
|
|
|
A0 06 30 04 04 02 60 34 = Private Keys
|
2011-02-04 12:57:53 +01:00
|
|
|
|
A4 06 30 04 04 02 60 35 = Certificates
|
2020-03-30 21:18:12 +02:00
|
|
|
|
A5 06 30 04 04 02 60 36 = Trusted Certificates
|
|
|
|
|
A7 06 30 04 04 02 60 37 = Data Objects
|
|
|
|
|
A8 06 30 04 04 02 60 38 = Auth Objects
|
2011-02-04 12:57:53 +01:00
|
|
|
|
|
2005-04-27 12:09:21 +00:00
|
|
|
|
These are all PathOrObjects using the path CHOICE element. The
|
|
|
|
|
paths are octet strings of length 2. Using this Path CHOICE
|
|
|
|
|
element is recommended, so we only implement that for now.
|
2004-12-02 07:48:09 +00:00
|
|
|
|
*/
|
|
|
|
|
static gpg_error_t
|
2005-04-27 12:09:21 +00:00
|
|
|
|
read_ef_odf (app_t app, unsigned short odf_fid)
|
2004-12-02 07:48:09 +00:00
|
|
|
|
{
|
|
|
|
|
gpg_error_t err;
|
|
|
|
|
unsigned char *buffer, *p;
|
2020-03-30 21:18:12 +02:00
|
|
|
|
size_t buflen, n;
|
2004-12-02 07:48:09 +00:00
|
|
|
|
unsigned short value;
|
2005-04-27 12:09:21 +00:00
|
|
|
|
size_t offset;
|
2020-03-30 21:18:12 +02:00
|
|
|
|
unsigned short home_df = 0;
|
2004-12-02 07:48:09 +00:00
|
|
|
|
|
2021-04-29 15:26:36 +02:00
|
|
|
|
|
|
|
|
|
app->app_local->odf.private_keys = 0;
|
|
|
|
|
app->app_local->odf.public_keys = 0;
|
|
|
|
|
app->app_local->odf.trusted_public_keys = 0;
|
|
|
|
|
app->app_local->odf.secret_keys = 0;
|
|
|
|
|
app->app_local->odf.certificates = 0;
|
|
|
|
|
app->app_local->odf.trusted_certificates = 0;
|
|
|
|
|
app->app_local->odf.useful_certificates = 0;
|
|
|
|
|
app->app_local->odf.data_objects = 0;
|
|
|
|
|
app->app_local->odf.auth_objects = 0;
|
|
|
|
|
|
|
|
|
|
err = select_and_read_binary (app, odf_fid, "ODF",
|
|
|
|
|
&buffer, &buflen);
|
2004-12-02 07:48:09 +00:00
|
|
|
|
if (err)
|
|
|
|
|
return err;
|
|
|
|
|
|
2005-04-27 12:09:21 +00:00
|
|
|
|
if (buflen < 8)
|
2004-12-02 07:48:09 +00:00
|
|
|
|
{
|
2020-03-31 10:40:21 +02:00
|
|
|
|
log_error ("p15: error: ODF too short\n");
|
2004-12-02 07:48:09 +00:00
|
|
|
|
xfree (buffer);
|
|
|
|
|
return gpg_error (GPG_ERR_INV_OBJ);
|
|
|
|
|
}
|
2020-03-30 21:18:12 +02:00
|
|
|
|
|
|
|
|
|
home_df = app->app_local->home_df;
|
2005-04-27 12:09:21 +00:00
|
|
|
|
p = buffer;
|
|
|
|
|
while (buflen && *p && *p != 0xff)
|
2004-12-02 07:48:09 +00:00
|
|
|
|
{
|
2005-04-27 12:09:21 +00:00
|
|
|
|
if ( buflen >= 8
|
|
|
|
|
&& (p[0] & 0xf0) == 0xA0
|
|
|
|
|
&& !memcmp (p+1, "\x06\x30\x04\x04\x02", 5) )
|
|
|
|
|
{
|
|
|
|
|
offset = 6;
|
|
|
|
|
}
|
2011-02-04 12:57:53 +01:00
|
|
|
|
else if ( buflen >= 12
|
2005-04-27 12:09:21 +00:00
|
|
|
|
&& (p[0] & 0xf0) == 0xA0
|
2005-09-05 14:36:36 +00:00
|
|
|
|
&& !memcmp (p+1, "\x0a\x30\x08\x04\x06\x3F\x00", 7)
|
2020-03-30 21:18:12 +02:00
|
|
|
|
&& (!home_df || home_df == ((p[8]<<8)|p[9])) )
|
|
|
|
|
{
|
2021-04-29 15:26:36 +02:00
|
|
|
|
/* FIXME: Is this hack still required? */
|
2020-03-30 21:18:12 +02:00
|
|
|
|
/* If we do not know the home DF, we take it from the first
|
|
|
|
|
* ODF object. Here are sample values:
|
|
|
|
|
* a0 0a 30 08 0406 3f00 5015 4401
|
|
|
|
|
* a1 0a 30 08 0406 3f00 5015 4411
|
|
|
|
|
* a4 0a 30 08 0406 3f00 5015 4441
|
|
|
|
|
* a5 0a 30 08 0406 3f00 5015 4451
|
|
|
|
|
* a8 0a 30 08 0406 3f00 5015 4481
|
|
|
|
|
* 00000000 */
|
|
|
|
|
if (!home_df)
|
|
|
|
|
{
|
|
|
|
|
home_df = ((p[8]<<8)|p[9]);
|
|
|
|
|
app->app_local->home_df = home_df;
|
2020-03-31 10:40:21 +02:00
|
|
|
|
log_info ("p15: application directory detected as 0x%04hX\n",
|
2020-03-30 21:18:12 +02:00
|
|
|
|
home_df);
|
2020-03-31 11:57:19 +02:00
|
|
|
|
/* We assume that direct path selection is possible. */
|
|
|
|
|
app->app_local->direct_path_selection = 1;
|
2020-03-30 21:18:12 +02:00
|
|
|
|
}
|
|
|
|
|
|
2005-09-05 14:36:36 +00:00
|
|
|
|
/* We only allow a full path if all files are at the same
|
2020-03-30 21:18:12 +02:00
|
|
|
|
level and below the home directory. To extend this we
|
2005-09-05 14:36:36 +00:00
|
|
|
|
would need to make use of new data type capable of
|
|
|
|
|
keeping a full path. */
|
2005-04-27 12:09:21 +00:00
|
|
|
|
offset = 10;
|
|
|
|
|
}
|
|
|
|
|
else
|
2004-12-02 07:48:09 +00:00
|
|
|
|
{
|
2020-05-12 18:51:47 +02:00
|
|
|
|
log_printhex (p, buflen, "p15: ODF format not supported:");
|
2004-12-02 07:48:09 +00:00
|
|
|
|
xfree (buffer);
|
|
|
|
|
return gpg_error (GPG_ERR_INV_OBJ);
|
|
|
|
|
}
|
|
|
|
|
switch ((p[0] & 0x0f))
|
|
|
|
|
{
|
|
|
|
|
case 0: value = app->app_local->odf.private_keys; break;
|
|
|
|
|
case 1: value = app->app_local->odf.public_keys; break;
|
|
|
|
|
case 2: value = app->app_local->odf.trusted_public_keys; break;
|
|
|
|
|
case 3: value = app->app_local->odf.secret_keys; break;
|
|
|
|
|
case 4: value = app->app_local->odf.certificates; break;
|
|
|
|
|
case 5: value = app->app_local->odf.trusted_certificates; break;
|
|
|
|
|
case 6: value = app->app_local->odf.useful_certificates; break;
|
|
|
|
|
case 7: value = app->app_local->odf.data_objects; break;
|
|
|
|
|
case 8: value = app->app_local->odf.auth_objects; break;
|
|
|
|
|
default: value = 0; break;
|
|
|
|
|
}
|
|
|
|
|
if (value)
|
|
|
|
|
{
|
2020-03-31 10:40:21 +02:00
|
|
|
|
log_error ("p15: duplicate object type %d in ODF ignored\n",
|
|
|
|
|
(p[0]&0x0f));
|
2004-12-02 07:48:09 +00:00
|
|
|
|
continue;
|
|
|
|
|
}
|
2005-04-27 12:09:21 +00:00
|
|
|
|
value = ((p[offset] << 8) | p[offset+1]);
|
2004-12-02 07:48:09 +00:00
|
|
|
|
switch ((p[0] & 0x0f))
|
|
|
|
|
{
|
|
|
|
|
case 0: app->app_local->odf.private_keys = value; break;
|
|
|
|
|
case 1: app->app_local->odf.public_keys = value; break;
|
|
|
|
|
case 2: app->app_local->odf.trusted_public_keys = value; break;
|
|
|
|
|
case 3: app->app_local->odf.secret_keys = value; break;
|
|
|
|
|
case 4: app->app_local->odf.certificates = value; break;
|
|
|
|
|
case 5: app->app_local->odf.trusted_certificates = value; break;
|
|
|
|
|
case 6: app->app_local->odf.useful_certificates = value; break;
|
|
|
|
|
case 7: app->app_local->odf.data_objects = value; break;
|
|
|
|
|
case 8: app->app_local->odf.auth_objects = value; break;
|
2011-02-04 12:57:53 +01:00
|
|
|
|
default:
|
2020-03-31 10:40:21 +02:00
|
|
|
|
log_error ("p15: unknown object type %d in ODF ignored\n",
|
|
|
|
|
(p[0]&0x0f));
|
2004-12-02 07:48:09 +00:00
|
|
|
|
}
|
2005-04-27 12:09:21 +00:00
|
|
|
|
offset += 2;
|
|
|
|
|
|
|
|
|
|
if (buflen < offset)
|
|
|
|
|
break;
|
|
|
|
|
p += offset;
|
|
|
|
|
buflen -= offset;
|
2004-12-02 07:48:09 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (buflen)
|
2020-03-30 21:18:12 +02:00
|
|
|
|
{
|
|
|
|
|
/* Print a warning if non-null garbage is left over. */
|
|
|
|
|
for (n=0; n < buflen && !p[n]; n++)
|
|
|
|
|
;
|
|
|
|
|
if (n < buflen)
|
|
|
|
|
{
|
2020-03-31 10:40:21 +02:00
|
|
|
|
log_info ("p15: warning: garbage detected at end of ODF: ");
|
2020-05-12 18:51:47 +02:00
|
|
|
|
log_printhex (p, buflen, "");
|
2020-03-30 21:18:12 +02:00
|
|
|
|
}
|
|
|
|
|
}
|
2004-12-02 07:48:09 +00:00
|
|
|
|
|
|
|
|
|
xfree (buffer);
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2021-04-29 15:26:36 +02:00
|
|
|
|
/* Helper for the read_ef_foo functions to read the first record or
|
|
|
|
|
* the entire data. */
|
|
|
|
|
static gpg_error_t
|
|
|
|
|
read_first_record (app_t app, unsigned short fid, const char *fid_desc,
|
|
|
|
|
unsigned char **r_buffer, size_t *r_buflen,
|
|
|
|
|
int *r_use_read_record)
|
|
|
|
|
{
|
|
|
|
|
gpg_error_t err;
|
|
|
|
|
int sw;
|
|
|
|
|
|
|
|
|
|
*r_buffer = NULL;
|
|
|
|
|
*r_buflen = 0;
|
|
|
|
|
*r_use_read_record = 0;
|
|
|
|
|
|
|
|
|
|
if (!fid)
|
|
|
|
|
return gpg_error (GPG_ERR_NO_DATA); /* No such file. */
|
|
|
|
|
|
|
|
|
|
if (IS_CARDOS_5 (app))
|
|
|
|
|
{
|
|
|
|
|
*r_use_read_record = 1;
|
|
|
|
|
err = select_and_read_record (app, fid, 1, fid_desc,
|
|
|
|
|
r_buffer, r_buflen, &sw);
|
|
|
|
|
if (err && sw == SW_FILE_STRUCT)
|
|
|
|
|
{
|
|
|
|
|
*r_use_read_record = 0;
|
|
|
|
|
err = select_and_read_binary (app, 0, fid_desc, r_buffer, r_buflen);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
err = select_and_read_binary (app, fid, fid_desc, r_buffer, r_buflen);
|
|
|
|
|
|
|
|
|
|
/* We get a not_found state in read_record mode if the select
|
|
|
|
|
* succeeded but reading the record failed. Map that to no_data
|
|
|
|
|
* which is what the caller of the read_ef_foo functions expect. */
|
|
|
|
|
if (gpg_err_code (err) == GPG_ERR_NOT_FOUND)
|
|
|
|
|
err = gpg_error (GPG_ERR_NO_DATA);
|
|
|
|
|
|
|
|
|
|
return err;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2014-07-24 16:16:53 +02:00
|
|
|
|
/* Parse the BIT STRING with the keyUsageFlags from the
|
2005-04-27 12:09:21 +00:00
|
|
|
|
CommonKeyAttributes. */
|
|
|
|
|
static gpg_error_t
|
|
|
|
|
parse_keyusage_flags (const unsigned char *der, size_t derlen,
|
|
|
|
|
keyusage_flags_t *usageflags)
|
|
|
|
|
{
|
|
|
|
|
unsigned int bits, mask;
|
|
|
|
|
int i, unused, full;
|
|
|
|
|
|
|
|
|
|
memset (usageflags, 0, sizeof *usageflags);
|
|
|
|
|
if (!derlen)
|
|
|
|
|
return gpg_error (GPG_ERR_INV_OBJ);
|
2011-02-04 12:57:53 +01:00
|
|
|
|
|
2005-04-27 12:09:21 +00:00
|
|
|
|
unused = *der++; derlen--;
|
|
|
|
|
if ((!derlen && unused) || unused/8 > derlen)
|
|
|
|
|
return gpg_error (GPG_ERR_ENCODING_PROBLEM);
|
|
|
|
|
full = derlen - (unused+7)/8;
|
|
|
|
|
unused %= 8;
|
|
|
|
|
mask = 0;
|
|
|
|
|
for (i=1; unused; i <<= 1, unused--)
|
|
|
|
|
mask |= i;
|
2011-02-04 12:57:53 +01:00
|
|
|
|
|
2005-04-27 12:09:21 +00:00
|
|
|
|
/* First octet */
|
|
|
|
|
if (derlen)
|
|
|
|
|
{
|
|
|
|
|
bits = *der++; derlen--;
|
|
|
|
|
if (full)
|
|
|
|
|
full--;
|
2011-02-04 12:57:53 +01:00
|
|
|
|
else
|
2005-04-27 12:09:21 +00:00
|
|
|
|
{
|
|
|
|
|
bits &= ~mask;
|
2011-02-04 12:57:53 +01:00
|
|
|
|
mask = 0;
|
2005-04-27 12:09:21 +00:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
bits = 0;
|
|
|
|
|
if ((bits & 0x80)) usageflags->encrypt = 1;
|
|
|
|
|
if ((bits & 0x40)) usageflags->decrypt = 1;
|
2011-02-04 12:57:53 +01:00
|
|
|
|
if ((bits & 0x20)) usageflags->sign = 1;
|
2005-04-27 12:09:21 +00:00
|
|
|
|
if ((bits & 0x10)) usageflags->sign_recover = 1;
|
|
|
|
|
if ((bits & 0x08)) usageflags->wrap = 1;
|
2011-02-04 12:57:53 +01:00
|
|
|
|
if ((bits & 0x04)) usageflags->unwrap = 1;
|
|
|
|
|
if ((bits & 0x02)) usageflags->verify = 1;
|
2005-04-27 12:09:21 +00:00
|
|
|
|
if ((bits & 0x01)) usageflags->verify_recover = 1;
|
|
|
|
|
|
|
|
|
|
/* Second octet. */
|
|
|
|
|
if (derlen)
|
|
|
|
|
{
|
|
|
|
|
bits = *der++; derlen--;
|
|
|
|
|
if (full)
|
|
|
|
|
full--;
|
2011-02-04 12:57:53 +01:00
|
|
|
|
else
|
2005-04-27 12:09:21 +00:00
|
|
|
|
{
|
|
|
|
|
bits &= ~mask;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
bits = 0;
|
|
|
|
|
if ((bits & 0x80)) usageflags->derive = 1;
|
|
|
|
|
if ((bits & 0x40)) usageflags->non_repudiation = 1;
|
|
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
2004-12-02 07:48:09 +00:00
|
|
|
|
|
|
|
|
|
|
2021-04-29 15:26:36 +02:00
|
|
|
|
static void
|
|
|
|
|
dump_keyusage_flags (keyusage_flags_t usageflags)
|
|
|
|
|
{
|
|
|
|
|
const char *s = "";
|
|
|
|
|
|
|
|
|
|
log_info ("p15: usage=");
|
|
|
|
|
if (usageflags.encrypt)
|
|
|
|
|
log_printf ("%sencrypt", s), s = ",";
|
|
|
|
|
if (usageflags.decrypt)
|
|
|
|
|
log_printf ("%sdecrypt", s), s = ",";
|
|
|
|
|
if (usageflags.sign )
|
|
|
|
|
log_printf ("%ssign", s), s = ",";
|
|
|
|
|
if (usageflags.sign_recover)
|
|
|
|
|
log_printf ("%ssign_recover", s), s = ",";
|
|
|
|
|
if (usageflags.wrap )
|
|
|
|
|
log_printf ("%swrap", s), s = ",";
|
|
|
|
|
if (usageflags.unwrap )
|
|
|
|
|
log_printf ("%sunwrap", s), s = ",";
|
|
|
|
|
if (usageflags.verify )
|
|
|
|
|
log_printf ("%sverify", s), s = ",";
|
|
|
|
|
if (usageflags.verify_recover)
|
|
|
|
|
log_printf ("%sverify_recover", s), s = ",";
|
|
|
|
|
if (usageflags.derive )
|
|
|
|
|
log_printf ("%sderive", s), s = ",";
|
|
|
|
|
if (usageflags.non_repudiation)
|
|
|
|
|
log_printf ("%snon_repudiation", s), s = ",";
|
|
|
|
|
}
|
2004-12-02 07:48:09 +00:00
|
|
|
|
|
2021-04-29 15:26:36 +02:00
|
|
|
|
|
|
|
|
|
static void
|
|
|
|
|
dump_keyaccess_flags (keyaccess_flags_t accessflags)
|
2004-12-02 07:48:09 +00:00
|
|
|
|
{
|
2021-04-29 15:26:36 +02:00
|
|
|
|
const char *s = "";
|
|
|
|
|
|
|
|
|
|
log_info ("p15: access=");
|
|
|
|
|
if (accessflags.sensitive)
|
|
|
|
|
log_printf ("%ssensitive", s), s = ",";
|
|
|
|
|
if (accessflags.extractable)
|
|
|
|
|
log_printf ("%sextractable", s), s = ",";
|
|
|
|
|
if (accessflags.always_sensitive)
|
|
|
|
|
log_printf ("%salways_sensitive", s), s = ",";
|
|
|
|
|
if (accessflags.never_extractable)
|
|
|
|
|
log_printf ("%snever_extractable", s), s = ",";
|
|
|
|
|
if (accessflags.local)
|
|
|
|
|
log_printf ("%slocal", s), s = ",";
|
|
|
|
|
}
|
2011-02-04 12:57:53 +01:00
|
|
|
|
|
|
|
|
|
|
2021-04-29 15:26:36 +02:00
|
|
|
|
static void
|
|
|
|
|
dump_gpgusage_flags (gpgusage_flags_t gpgusage)
|
|
|
|
|
{
|
|
|
|
|
const char *s = "";
|
|
|
|
|
|
|
|
|
|
log_info ("p15: gpgusage=");
|
|
|
|
|
if (gpgusage.cert)
|
|
|
|
|
log_printf ("%scert", s), s = ",";
|
|
|
|
|
if (gpgusage.sign)
|
|
|
|
|
log_printf ("%ssign", s), s = ",";
|
|
|
|
|
if (gpgusage.encr)
|
|
|
|
|
log_printf ("%sencr", s), s = ",";
|
|
|
|
|
if (gpgusage.auth)
|
|
|
|
|
log_printf ("%sauth", s), s = ",";
|
|
|
|
|
}
|
2011-02-04 12:57:53 +01:00
|
|
|
|
|
2004-12-02 07:48:09 +00:00
|
|
|
|
|
2021-04-29 15:26:36 +02:00
|
|
|
|
/* Parse the BIT STRING with the keyAccessFlags from the
|
|
|
|
|
CommonKeyAttributes. */
|
|
|
|
|
static gpg_error_t
|
|
|
|
|
parse_keyaccess_flags (const unsigned char *der, size_t derlen,
|
|
|
|
|
keyaccess_flags_t *accessflags)
|
|
|
|
|
{
|
|
|
|
|
unsigned int bits, mask;
|
|
|
|
|
int i, unused, full;
|
2005-04-27 12:09:21 +00:00
|
|
|
|
|
2021-04-29 15:26:36 +02:00
|
|
|
|
memset (accessflags, 0, sizeof *accessflags);
|
|
|
|
|
if (!derlen)
|
|
|
|
|
return gpg_error (GPG_ERR_INV_OBJ);
|
2004-12-02 07:48:09 +00:00
|
|
|
|
|
2021-04-29 15:26:36 +02:00
|
|
|
|
unused = *der++; derlen--;
|
|
|
|
|
if ((!derlen && unused) || unused/8 > derlen)
|
|
|
|
|
return gpg_error (GPG_ERR_ENCODING_PROBLEM);
|
|
|
|
|
full = derlen - (unused+7)/8;
|
|
|
|
|
unused %= 8;
|
|
|
|
|
mask = 0;
|
|
|
|
|
for (i=1; unused; i <<= 1, unused--)
|
|
|
|
|
mask |= i;
|
|
|
|
|
|
|
|
|
|
/* First octet */
|
|
|
|
|
if (derlen)
|
|
|
|
|
{
|
|
|
|
|
bits = *der++; derlen--;
|
|
|
|
|
if (full)
|
|
|
|
|
full--;
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
bits &= ~mask;
|
|
|
|
|
mask = 0;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
bits = 0;
|
|
|
|
|
if ((bits & 0x10)) accessflags->local = 1;
|
|
|
|
|
if ((bits & 0x08)) accessflags->never_extractable = 1;
|
|
|
|
|
if ((bits & 0x04)) accessflags->always_sensitive = 1;
|
|
|
|
|
if ((bits & 0x02)) accessflags->extractable = 1;
|
|
|
|
|
if ((bits & 0x01)) accessflags->sensitive = 1;
|
|
|
|
|
|
|
|
|
|
accessflags->any = 1;
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* Parse the commonObjectAttributes and store a malloced authid at
|
|
|
|
|
* (r_authid,r_authidlen). (NULL,0) is stored on error or if no
|
|
|
|
|
* authid is found. IF R_LABEL is not NULL the label is stored there
|
|
|
|
|
* as a malloced string (spaces are replaced by underscores).
|
|
|
|
|
*
|
|
|
|
|
* Example data:
|
|
|
|
|
* 2 30 17: SEQUENCE { -- commonObjectAttributes
|
|
|
|
|
* 4 0C 8: UTF8String 'SK.CH.DS' -- label
|
|
|
|
|
* 14 03 2: BIT STRING 6 unused bits
|
|
|
|
|
* : '01'B (bit 0)
|
|
|
|
|
* 18 04 1: OCTET STRING --authid
|
|
|
|
|
* : 07
|
|
|
|
|
* : }
|
|
|
|
|
*/
|
|
|
|
|
static gpg_error_t
|
|
|
|
|
parse_common_obj_attr (unsigned char const **buffer, size_t *size,
|
|
|
|
|
unsigned char **r_authid, size_t *r_authidlen,
|
|
|
|
|
char **r_label)
|
|
|
|
|
{
|
|
|
|
|
gpg_error_t err;
|
|
|
|
|
int where;
|
|
|
|
|
int class, tag, constructed, ndef;
|
|
|
|
|
size_t objlen, hdrlen, nnn;
|
|
|
|
|
const unsigned char *ppp;
|
|
|
|
|
int ignore_eof = 0;
|
|
|
|
|
char *p;
|
|
|
|
|
|
|
|
|
|
*r_authid = NULL;
|
|
|
|
|
*r_authidlen = 0;
|
|
|
|
|
if (r_label)
|
|
|
|
|
*r_label = NULL;
|
|
|
|
|
|
|
|
|
|
where = __LINE__;
|
|
|
|
|
err = parse_ber_header (buffer, size, &class, &tag, &constructed,
|
|
|
|
|
&ndef, &objlen, &hdrlen);
|
|
|
|
|
if (!err && (objlen > *size || tag != TAG_SEQUENCE))
|
|
|
|
|
err = gpg_error (GPG_ERR_INV_OBJ);
|
|
|
|
|
if (err)
|
|
|
|
|
goto leave;
|
|
|
|
|
|
|
|
|
|
ppp = *buffer;
|
|
|
|
|
nnn = objlen;
|
|
|
|
|
*buffer += objlen;
|
|
|
|
|
*size -= objlen;
|
|
|
|
|
|
|
|
|
|
/* Search the optional AuthId. */
|
|
|
|
|
ignore_eof = 1;
|
|
|
|
|
where = __LINE__;
|
|
|
|
|
err = parse_ber_header (&ppp, &nnn, &class, &tag, &constructed,
|
|
|
|
|
&ndef, &objlen, &hdrlen);
|
|
|
|
|
if (!err && (objlen > nnn || class != CLASS_UNIVERSAL))
|
|
|
|
|
err = gpg_error (GPG_ERR_INV_OBJ);
|
|
|
|
|
if (err)
|
|
|
|
|
goto leave;
|
|
|
|
|
|
|
|
|
|
if (tag == TAG_UTF8_STRING)
|
|
|
|
|
{
|
|
|
|
|
if (r_label)
|
|
|
|
|
{
|
|
|
|
|
*r_label = xtrymalloc (objlen + 1);
|
|
|
|
|
if (!*r_label)
|
|
|
|
|
{
|
|
|
|
|
err = gpg_error_from_syserror ();
|
|
|
|
|
goto leave;
|
|
|
|
|
}
|
|
|
|
|
memcpy (*r_label, ppp, objlen);
|
|
|
|
|
(*r_label)[objlen] = 0;
|
|
|
|
|
/* We don't want spaces in the labels due to the properties
|
|
|
|
|
* of CHV-LABEL. */
|
|
|
|
|
for (p = *r_label; *p; p++)
|
|
|
|
|
if (ascii_isspace (*p))
|
|
|
|
|
*p = '_';
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
ppp += objlen;
|
|
|
|
|
nnn -= objlen;
|
|
|
|
|
|
|
|
|
|
where = __LINE__;
|
|
|
|
|
err = parse_ber_header (&ppp, &nnn, &class, &tag, &constructed,
|
|
|
|
|
&ndef, &objlen, &hdrlen);
|
|
|
|
|
if (!err && (objlen > nnn || class != CLASS_UNIVERSAL))
|
|
|
|
|
err = gpg_error (GPG_ERR_INV_OBJ);
|
|
|
|
|
if (err)
|
|
|
|
|
goto leave;
|
|
|
|
|
}
|
|
|
|
|
if (tag == TAG_BIT_STRING)
|
|
|
|
|
{
|
|
|
|
|
ppp += objlen; /* Skip the CommonObjectFlags. */
|
|
|
|
|
nnn -= objlen;
|
|
|
|
|
|
|
|
|
|
where = __LINE__;
|
|
|
|
|
err = parse_ber_header (&ppp, &nnn, &class, &tag, &constructed,
|
|
|
|
|
&ndef, &objlen, &hdrlen);
|
|
|
|
|
if (!err && (objlen > nnn || class != CLASS_UNIVERSAL))
|
|
|
|
|
err = gpg_error (GPG_ERR_INV_OBJ);
|
|
|
|
|
if (err)
|
|
|
|
|
goto leave;
|
|
|
|
|
}
|
|
|
|
|
if (tag == TAG_OCTET_STRING && objlen)
|
|
|
|
|
{
|
|
|
|
|
*r_authid = xtrymalloc (objlen);
|
|
|
|
|
if (!*r_authid)
|
|
|
|
|
{
|
|
|
|
|
err = gpg_error_from_syserror ();
|
|
|
|
|
goto leave;
|
|
|
|
|
}
|
|
|
|
|
memcpy (*r_authid, ppp, objlen);
|
|
|
|
|
*r_authidlen = objlen;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
leave:
|
|
|
|
|
if (ignore_eof && gpg_err_code (err) == GPG_ERR_EOF)
|
|
|
|
|
err = 0;
|
|
|
|
|
else if (err)
|
|
|
|
|
log_error ("p15: error parsing commonObjectAttributes at %d: %s\n",
|
|
|
|
|
where, gpg_strerror (err));
|
|
|
|
|
|
|
|
|
|
if (err && r_label)
|
|
|
|
|
{
|
|
|
|
|
xfree (*r_label);
|
|
|
|
|
*r_label = NULL;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return err;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* Parse the commonKeyAttributes. On success store the objid at
|
|
|
|
|
* (R_OBJID/R_OBJIDLEN), sets the key usage flags at USAGEFLAGS and
|
|
|
|
|
* the optiona key refrence at R_KEY_REFERENCE. The latter is only
|
|
|
|
|
* valid if true is also stored at R_KEY_REFERENCE_VALID.
|
|
|
|
|
*
|
|
|
|
|
* Example data:
|
|
|
|
|
*
|
|
|
|
|
* 21 30 12: SEQUENCE { -- commonKeyAttributes
|
|
|
|
|
* 23 04 1: OCTET STRING
|
|
|
|
|
* : 01
|
|
|
|
|
* 26 03 3: BIT STRING 6 unused bits
|
|
|
|
|
* : '1000000000'B (bit 9)
|
|
|
|
|
* 31 02 2: INTEGER 80 -- keyReference (optional)
|
|
|
|
|
* : }
|
|
|
|
|
*/
|
|
|
|
|
static gpg_error_t
|
|
|
|
|
parse_common_key_attr (unsigned char const **buffer, size_t *size,
|
|
|
|
|
unsigned char **r_objid, size_t *r_objidlen,
|
|
|
|
|
keyusage_flags_t *usageflags,
|
|
|
|
|
keyaccess_flags_t *accessflags,
|
|
|
|
|
unsigned long *r_key_reference,
|
|
|
|
|
int *r_key_reference_valid)
|
|
|
|
|
{
|
|
|
|
|
gpg_error_t err;
|
|
|
|
|
int where;
|
|
|
|
|
int class, tag, constructed, ndef;
|
|
|
|
|
size_t objlen, hdrlen, nnn;
|
|
|
|
|
const unsigned char *ppp;
|
|
|
|
|
int ignore_eof = 0;
|
|
|
|
|
unsigned long ul;
|
|
|
|
|
const unsigned char *objid = NULL;
|
|
|
|
|
size_t objidlen;
|
|
|
|
|
unsigned long key_reference = 0;
|
|
|
|
|
int key_reference_valid = 0;
|
|
|
|
|
|
|
|
|
|
*r_objid = NULL;
|
|
|
|
|
*r_objidlen = 0;
|
|
|
|
|
memset (usageflags, 0, sizeof *usageflags);
|
|
|
|
|
memset (accessflags, 0, sizeof *accessflags);
|
|
|
|
|
*r_key_reference_valid = 0;
|
|
|
|
|
|
|
|
|
|
where = __LINE__;
|
|
|
|
|
err = parse_ber_header (buffer, size, &class, &tag, &constructed,
|
|
|
|
|
&ndef, &objlen, &hdrlen);
|
|
|
|
|
if (!err && (objlen > *size || tag != TAG_SEQUENCE))
|
|
|
|
|
err = gpg_error (GPG_ERR_INV_OBJ);
|
|
|
|
|
if (err)
|
|
|
|
|
goto leave;
|
|
|
|
|
|
|
|
|
|
ppp = *buffer;
|
|
|
|
|
nnn = objlen;
|
|
|
|
|
*buffer += objlen;
|
|
|
|
|
*size -= objlen;
|
|
|
|
|
|
|
|
|
|
/* Get the Id. */
|
|
|
|
|
where = __LINE__;
|
|
|
|
|
err = parse_ber_header (&ppp, &nnn, &class, &tag, &constructed,
|
|
|
|
|
&ndef, &objlen, &hdrlen);
|
|
|
|
|
if (!err && (objlen > nnn
|
|
|
|
|
|| class != CLASS_UNIVERSAL || tag != TAG_OCTET_STRING))
|
|
|
|
|
err = gpg_error (GPG_ERR_INV_OBJ);
|
|
|
|
|
if (err)
|
|
|
|
|
goto leave;
|
|
|
|
|
|
|
|
|
|
objid = ppp;
|
|
|
|
|
objidlen = objlen;
|
|
|
|
|
ppp += objlen;
|
|
|
|
|
nnn -= objlen;
|
|
|
|
|
|
|
|
|
|
/* Get the KeyUsageFlags. */
|
|
|
|
|
where = __LINE__;
|
|
|
|
|
err = parse_ber_header (&ppp, &nnn, &class, &tag, &constructed,
|
|
|
|
|
&ndef, &objlen, &hdrlen);
|
|
|
|
|
if (!err && (objlen > nnn
|
|
|
|
|
|| class != CLASS_UNIVERSAL || tag != TAG_BIT_STRING))
|
|
|
|
|
err = gpg_error (GPG_ERR_INV_OBJ);
|
|
|
|
|
if (err)
|
|
|
|
|
goto leave;
|
|
|
|
|
|
|
|
|
|
err = parse_keyusage_flags (ppp, objlen, usageflags);
|
|
|
|
|
if (err)
|
|
|
|
|
goto leave;
|
|
|
|
|
ppp += objlen;
|
|
|
|
|
nnn -= objlen;
|
|
|
|
|
|
|
|
|
|
ignore_eof = 1; /* Remaining items are optional. */
|
|
|
|
|
|
|
|
|
|
/* Find the keyReference */
|
|
|
|
|
where = __LINE__;
|
|
|
|
|
err = parse_ber_header (&ppp, &nnn, &class, &tag, &constructed,
|
|
|
|
|
&ndef, &objlen, &hdrlen);
|
|
|
|
|
if (!err && objlen > nnn)
|
|
|
|
|
err = gpg_error (GPG_ERR_INV_OBJ);
|
|
|
|
|
if (err)
|
|
|
|
|
goto leave;
|
|
|
|
|
|
|
|
|
|
if (class == CLASS_UNIVERSAL && tag == TAG_BOOLEAN)
|
|
|
|
|
{
|
|
|
|
|
/* Skip the native element. */
|
|
|
|
|
ppp += objlen;
|
|
|
|
|
nnn -= objlen;
|
|
|
|
|
|
|
|
|
|
err = parse_ber_header (&ppp, &nnn, &class, &tag, &constructed,
|
|
|
|
|
&ndef, &objlen, &hdrlen);
|
|
|
|
|
if (!err && objlen > nnn)
|
|
|
|
|
err = gpg_error (GPG_ERR_INV_OBJ);
|
|
|
|
|
if (err)
|
|
|
|
|
goto leave;
|
|
|
|
|
}
|
|
|
|
|
if (class == CLASS_UNIVERSAL && tag == TAG_BIT_STRING)
|
|
|
|
|
{
|
|
|
|
|
/* These are the keyAccessFlags. */
|
|
|
|
|
err = parse_keyaccess_flags (ppp, objlen, accessflags);
|
|
|
|
|
if (err)
|
|
|
|
|
goto leave;
|
|
|
|
|
ppp += objlen;
|
|
|
|
|
nnn -= objlen;
|
|
|
|
|
|
|
|
|
|
err = parse_ber_header (&ppp, &nnn, &class, &tag, &constructed,
|
|
|
|
|
&ndef, &objlen, &hdrlen);
|
|
|
|
|
if (!err && objlen > nnn)
|
|
|
|
|
err = gpg_error (GPG_ERR_INV_OBJ);
|
|
|
|
|
if (err)
|
|
|
|
|
goto leave;
|
|
|
|
|
}
|
|
|
|
|
if (class == CLASS_UNIVERSAL && tag == TAG_INTEGER)
|
|
|
|
|
{
|
|
|
|
|
/* This is the keyReference. */
|
|
|
|
|
for (ul=0; objlen; objlen--)
|
|
|
|
|
{
|
|
|
|
|
ul <<= 8;
|
|
|
|
|
ul |= (*ppp++) & 0xff;
|
|
|
|
|
nnn--;
|
|
|
|
|
}
|
|
|
|
|
key_reference = ul;
|
|
|
|
|
key_reference_valid = 1;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
leave:
|
|
|
|
|
if (ignore_eof && gpg_err_code (err) == GPG_ERR_EOF)
|
|
|
|
|
err = 0;
|
|
|
|
|
|
|
|
|
|
if (!err)
|
|
|
|
|
{
|
|
|
|
|
if (!objid || !objidlen)
|
|
|
|
|
err = gpg_error (GPG_ERR_INV_OBJ);
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
*r_objid = xtrymalloc (objidlen);
|
|
|
|
|
if (!*r_objid)
|
|
|
|
|
err = gpg_error_from_syserror ();
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
memcpy (*r_objid, objid, objidlen);
|
|
|
|
|
*r_objidlen = objidlen;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
if (!err && key_reference_valid)
|
|
|
|
|
{
|
|
|
|
|
*r_key_reference = key_reference;
|
|
|
|
|
*r_key_reference_valid = 1;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (err)
|
|
|
|
|
log_error ("p15: error parsing commonKeyAttributes at %d: %s\n",
|
|
|
|
|
where, gpg_strerror (err));
|
|
|
|
|
return err;
|
|
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* Read and parse the Private Key Directory Files.
|
|
|
|
|
*
|
|
|
|
|
* Sample object:
|
|
|
|
|
* SEQUENCE {
|
|
|
|
|
* SEQUENCE { -- commonObjectAttributes
|
|
|
|
|
* UTF8String 'SK.CH.DS'
|
|
|
|
|
* BIT STRING 6 unused bits
|
|
|
|
|
* '01'B (bit 0) -- flags: non-modifiable,private
|
|
|
|
|
* OCTET STRING --authid
|
|
|
|
|
* 07
|
|
|
|
|
* }
|
|
|
|
|
* SEQUENCE { -- commonKeyAttributes
|
|
|
|
|
* OCTET STRING
|
|
|
|
|
* 01
|
|
|
|
|
* BIT STRING 6 unused bits
|
|
|
|
|
* '1000000000'B (bit 9) -- keyusage: non-repudiation
|
|
|
|
|
* INTEGER 80 -- keyReference (optional)
|
|
|
|
|
* }
|
|
|
|
|
* [1] { -- keyAttributes
|
|
|
|
|
* SEQUENCE { -- privateRSAKeyAttributes
|
|
|
|
|
* SEQUENCE { -- objectValue
|
|
|
|
|
* OCTET STRING --path
|
|
|
|
|
* 3F 00 40 16 00 50
|
|
|
|
|
* }
|
|
|
|
|
* INTEGER 1024 -- modulus
|
|
|
|
|
* }
|
|
|
|
|
* }
|
|
|
|
|
* }
|
|
|
|
|
*
|
|
|
|
|
* Sample part for EC objects:
|
|
|
|
|
* [1] { -- keyAttributes
|
|
|
|
|
* [1] { -- privateECkeyAttributes
|
|
|
|
|
* SEQUENCE { -- objectValue
|
|
|
|
|
* SEQUENCE { --path
|
|
|
|
|
* OCTET STRING 50 72 4B 03
|
|
|
|
|
* }
|
|
|
|
|
* INTEGER 33 -- Not in PKCS#15v1.1, need to buy 7816-15?
|
|
|
|
|
* }
|
|
|
|
|
* }
|
|
|
|
|
*/
|
|
|
|
|
static gpg_error_t
|
|
|
|
|
read_ef_prkdf (app_t app, unsigned short fid, prkdf_object_t *result)
|
|
|
|
|
{
|
|
|
|
|
gpg_error_t err;
|
|
|
|
|
unsigned char *buffer;
|
|
|
|
|
size_t buflen;
|
|
|
|
|
const unsigned char *p;
|
|
|
|
|
size_t n, objlen, hdrlen;
|
|
|
|
|
int class, tag, constructed, ndef;
|
|
|
|
|
prkdf_object_t prkdflist = NULL;
|
|
|
|
|
int i;
|
|
|
|
|
int recno = 1;
|
|
|
|
|
unsigned char *authid = NULL;
|
|
|
|
|
size_t authidlen = 0;
|
|
|
|
|
unsigned char *objid = NULL;
|
|
|
|
|
size_t objidlen = 0;
|
|
|
|
|
char *label = NULL;
|
|
|
|
|
int record_mode;
|
|
|
|
|
|
|
|
|
|
err = read_first_record (app, fid, "PrKDF", &buffer, &buflen, &record_mode);
|
|
|
|
|
if (err)
|
|
|
|
|
return err;
|
|
|
|
|
|
|
|
|
|
p = buffer;
|
|
|
|
|
n = buflen;
|
|
|
|
|
|
|
|
|
|
/* Loop over the records. We stop as soon as we detect a new record
|
|
|
|
|
starting with 0x00 or 0xff as these values are commonly used to
|
|
|
|
|
pad data blocks and are no valid ASN.1 encoding. Note the
|
|
|
|
|
special handling for record mode at the end of the loop. */
|
|
|
|
|
while (n && *p && *p != 0xff)
|
|
|
|
|
{
|
|
|
|
|
const unsigned char *pp;
|
|
|
|
|
size_t nn;
|
|
|
|
|
int where;
|
|
|
|
|
const char *errstr = NULL;
|
|
|
|
|
prkdf_object_t prkdf = NULL;
|
|
|
|
|
unsigned long ul;
|
|
|
|
|
keyusage_flags_t usageflags;
|
|
|
|
|
keyaccess_flags_t accessflags;
|
|
|
|
|
unsigned long key_reference = 0;
|
|
|
|
|
int key_reference_valid = 0;
|
|
|
|
|
int is_ecc = 0;
|
|
|
|
|
|
|
|
|
|
where = __LINE__;
|
|
|
|
|
err = parse_ber_header (&p, &n, &class, &tag, &constructed,
|
|
|
|
|
&ndef, &objlen, &hdrlen);
|
|
|
|
|
if (err)
|
|
|
|
|
;
|
|
|
|
|
else if (objlen > n)
|
|
|
|
|
err = gpg_error (GPG_ERR_INV_OBJ);
|
|
|
|
|
else if (class == CLASS_UNIVERSAL && tag == TAG_SEQUENCE)
|
|
|
|
|
; /* PrivateRSAKeyAttributes */
|
|
|
|
|
else if (class == CLASS_CONTEXT)
|
|
|
|
|
{
|
|
|
|
|
switch (tag)
|
|
|
|
|
{
|
|
|
|
|
case 0: is_ecc = 1; break; /* PrivateECKeyAttributes */
|
|
|
|
|
case 1: errstr = "DH key objects are not supported"; break;
|
|
|
|
|
case 2: errstr = "DSA key objects are not supported"; break;
|
|
|
|
|
case 3: errstr = "KEA key objects are not supported"; break;
|
|
|
|
|
default: errstr = "unknown privateKeyObject"; break;
|
|
|
|
|
}
|
|
|
|
|
if (errstr)
|
|
|
|
|
goto parse_error;
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
err = gpg_error (GPG_ERR_INV_OBJ);
|
|
|
|
|
goto parse_error;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (err)
|
|
|
|
|
{
|
|
|
|
|
log_error ("p15: error parsing PrKDF record: %s\n",
|
|
|
|
|
gpg_strerror (err));
|
|
|
|
|
goto leave;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
pp = p;
|
|
|
|
|
nn = objlen;
|
|
|
|
|
p += objlen;
|
|
|
|
|
n -= objlen;
|
2004-12-02 07:48:09 +00:00
|
|
|
|
|
2005-04-27 12:09:21 +00:00
|
|
|
|
/* Parse the commonObjectAttributes. */
|
|
|
|
|
where = __LINE__;
|
2021-04-29 15:26:36 +02:00
|
|
|
|
xfree (authid);
|
|
|
|
|
xfree (label);
|
|
|
|
|
err = parse_common_obj_attr (&pp, &nn, &authid, &authidlen, &label);
|
|
|
|
|
if (err)
|
|
|
|
|
goto parse_error;
|
|
|
|
|
|
|
|
|
|
/* Parse the commonKeyAttributes. */
|
|
|
|
|
where = __LINE__;
|
|
|
|
|
xfree (objid);
|
|
|
|
|
err = parse_common_key_attr (&pp, &nn,
|
|
|
|
|
&objid, &objidlen,
|
|
|
|
|
&usageflags, &accessflags,
|
|
|
|
|
&key_reference, &key_reference_valid);
|
|
|
|
|
if (err)
|
|
|
|
|
goto parse_error;
|
|
|
|
|
log_assert (objid);
|
|
|
|
|
|
|
|
|
|
/* Skip commonPrivateKeyAttributes. */
|
|
|
|
|
where = __LINE__;
|
2004-12-02 07:48:09 +00:00
|
|
|
|
err = parse_ber_header (&pp, &nn, &class, &tag, &constructed,
|
|
|
|
|
&ndef, &objlen, &hdrlen);
|
2021-04-29 15:26:36 +02:00
|
|
|
|
if (!err && objlen > nn)
|
2004-12-02 07:48:09 +00:00
|
|
|
|
err = gpg_error (GPG_ERR_INV_OBJ);
|
|
|
|
|
if (err)
|
2005-04-27 12:09:21 +00:00
|
|
|
|
goto parse_error;
|
2021-04-29 15:26:36 +02:00
|
|
|
|
if (class == CLASS_CONTEXT && tag == 0)
|
|
|
|
|
{
|
|
|
|
|
pp += objlen;
|
|
|
|
|
nn -= objlen;
|
2005-04-27 12:09:21 +00:00
|
|
|
|
|
2021-04-29 15:26:36 +02:00
|
|
|
|
where = __LINE__;
|
|
|
|
|
err = parse_ber_header (&pp, &nn, &class, &tag, &constructed,
|
|
|
|
|
&ndef, &objlen, &hdrlen);
|
|
|
|
|
}
|
|
|
|
|
/* Parse the keyAttributes. */
|
|
|
|
|
if (!err && (objlen > nn || class != CLASS_CONTEXT || tag != 1))
|
|
|
|
|
err = gpg_error (GPG_ERR_INV_OBJ);
|
|
|
|
|
if (err)
|
|
|
|
|
goto parse_error;
|
|
|
|
|
nn = objlen;
|
2005-04-27 12:09:21 +00:00
|
|
|
|
|
2021-04-29 15:26:36 +02:00
|
|
|
|
where = __LINE__;
|
|
|
|
|
err = parse_ber_header (&pp, &nn, &class, &tag, &constructed,
|
|
|
|
|
&ndef, &objlen, &hdrlen);
|
|
|
|
|
if (err)
|
2005-04-27 12:09:21 +00:00
|
|
|
|
;
|
2021-04-29 15:26:36 +02:00
|
|
|
|
else if (!err && objlen > nn)
|
|
|
|
|
err = gpg_error (GPG_ERR_INV_OBJ);
|
|
|
|
|
else if (class == CLASS_UNIVERSAL && tag == TAG_SEQUENCE)
|
|
|
|
|
; /* A typeAttribute always starts with a sequence. */
|
|
|
|
|
else
|
|
|
|
|
err = gpg_error (GPG_ERR_INV_OBJ);
|
|
|
|
|
if (err)
|
|
|
|
|
goto parse_error;
|
2005-04-27 12:09:21 +00:00
|
|
|
|
|
2021-04-29 15:26:36 +02:00
|
|
|
|
nn = objlen;
|
|
|
|
|
|
|
|
|
|
/* Check that the reference is a Path object. */
|
2005-04-27 12:09:21 +00:00
|
|
|
|
where = __LINE__;
|
2004-12-02 07:48:09 +00:00
|
|
|
|
err = parse_ber_header (&pp, &nn, &class, &tag, &constructed,
|
|
|
|
|
&ndef, &objlen, &hdrlen);
|
2021-04-29 15:26:36 +02:00
|
|
|
|
if (!err && objlen > nn)
|
2004-12-02 07:48:09 +00:00
|
|
|
|
err = gpg_error (GPG_ERR_INV_OBJ);
|
|
|
|
|
if (err)
|
2005-04-27 12:09:21 +00:00
|
|
|
|
goto parse_error;
|
2021-04-29 15:26:36 +02:00
|
|
|
|
if (class != CLASS_UNIVERSAL || tag != TAG_SEQUENCE)
|
|
|
|
|
{
|
|
|
|
|
errstr = "unsupported reference type";
|
|
|
|
|
goto parse_error;
|
|
|
|
|
}
|
|
|
|
|
nn = objlen;
|
2004-12-02 07:48:09 +00:00
|
|
|
|
|
2021-04-29 15:26:36 +02:00
|
|
|
|
/* Parse the Path object. */
|
|
|
|
|
where = __LINE__;
|
|
|
|
|
err = parse_ber_header (&pp, &nn, &class, &tag, &constructed,
|
|
|
|
|
&ndef, &objlen, &hdrlen);
|
|
|
|
|
if (!err && objlen > nn)
|
|
|
|
|
err = gpg_error (GPG_ERR_INV_OBJ);
|
|
|
|
|
if (err)
|
|
|
|
|
goto parse_error;
|
2004-12-02 07:48:09 +00:00
|
|
|
|
|
2021-04-29 15:26:36 +02:00
|
|
|
|
/* Make sure that the next element is a non zero path and of
|
|
|
|
|
even length (FID are two bytes each). */
|
|
|
|
|
if (class != CLASS_UNIVERSAL || tag != TAG_OCTET_STRING
|
|
|
|
|
|| !objlen || (objlen & 1) )
|
|
|
|
|
{
|
|
|
|
|
errstr = "invalid path reference";
|
|
|
|
|
goto parse_error;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Create a new PrKDF list item. */
|
|
|
|
|
prkdf = xtrycalloc (1, (sizeof *prkdf
|
|
|
|
|
- sizeof(unsigned short)
|
|
|
|
|
+ objlen/2 * sizeof(unsigned short)));
|
|
|
|
|
if (!prkdf)
|
|
|
|
|
{
|
|
|
|
|
err = gpg_error_from_syserror ();
|
|
|
|
|
goto leave;
|
|
|
|
|
}
|
|
|
|
|
prkdf->is_ecc = is_ecc;
|
|
|
|
|
|
|
|
|
|
prkdf->objidlen = objidlen;
|
|
|
|
|
prkdf->objid = objid;
|
|
|
|
|
objid = NULL;
|
|
|
|
|
if (authid)
|
|
|
|
|
{
|
|
|
|
|
prkdf->authidlen = authidlen;
|
|
|
|
|
prkdf->authid = authid;
|
|
|
|
|
authid = NULL;
|
|
|
|
|
}
|
|
|
|
|
if (label)
|
|
|
|
|
{
|
|
|
|
|
prkdf->label = label;
|
|
|
|
|
label = NULL;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
prkdf->pathlen = objlen/2;
|
|
|
|
|
for (i=0; i < prkdf->pathlen; i++, pp += 2, nn -= 2)
|
|
|
|
|
prkdf->path[i] = ((pp[0] << 8) | pp[1]);
|
|
|
|
|
|
|
|
|
|
prkdf->usageflags = usageflags;
|
|
|
|
|
prkdf->accessflags = accessflags;
|
|
|
|
|
prkdf->key_reference = key_reference;
|
|
|
|
|
prkdf->key_reference_valid = key_reference_valid;
|
|
|
|
|
|
|
|
|
|
if (nn)
|
|
|
|
|
{
|
|
|
|
|
/* An index and length follows. */
|
|
|
|
|
prkdf->have_off = 1;
|
|
|
|
|
where = __LINE__;
|
|
|
|
|
err = parse_ber_header (&pp, &nn, &class, &tag, &constructed,
|
|
|
|
|
&ndef, &objlen, &hdrlen);
|
|
|
|
|
if (!err && (objlen > nn
|
|
|
|
|
|| class != CLASS_UNIVERSAL || tag != TAG_INTEGER))
|
|
|
|
|
err = gpg_error (GPG_ERR_INV_OBJ);
|
|
|
|
|
if (err)
|
|
|
|
|
goto parse_error;
|
|
|
|
|
|
|
|
|
|
for (ul=0; objlen; objlen--)
|
|
|
|
|
{
|
|
|
|
|
ul <<= 8;
|
|
|
|
|
ul |= (*pp++) & 0xff;
|
|
|
|
|
nn--;
|
|
|
|
|
}
|
|
|
|
|
prkdf->off = ul;
|
|
|
|
|
|
|
|
|
|
where = __LINE__;
|
|
|
|
|
err = parse_ber_header (&pp, &nn, &class, &tag, &constructed,
|
|
|
|
|
&ndef, &objlen, &hdrlen);
|
|
|
|
|
if (!err && (objlen > nn
|
|
|
|
|
|| class != CLASS_CONTEXT || tag != 0))
|
|
|
|
|
err = gpg_error (GPG_ERR_INV_OBJ);
|
|
|
|
|
if (err)
|
|
|
|
|
goto parse_error;
|
|
|
|
|
|
|
|
|
|
for (ul=0; objlen; objlen--)
|
|
|
|
|
{
|
|
|
|
|
ul <<= 8;
|
|
|
|
|
ul |= (*pp++) & 0xff;
|
|
|
|
|
nn--;
|
|
|
|
|
}
|
|
|
|
|
prkdf->len = ul;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* The info is printed later in read_p15_info because we also
|
|
|
|
|
* want to look at the certificates. */
|
|
|
|
|
|
|
|
|
|
/* Put it into the list. */
|
|
|
|
|
prkdf->next = prkdflist;
|
|
|
|
|
prkdflist = prkdf;
|
|
|
|
|
prkdf = NULL;
|
|
|
|
|
goto next_record; /* Ready with this record. */
|
|
|
|
|
|
|
|
|
|
parse_error:
|
|
|
|
|
log_error ("p15: error parsing PrKDF record at %d: %s - skipped\n",
|
|
|
|
|
where, errstr? errstr : gpg_strerror (err));
|
|
|
|
|
if (prkdf)
|
|
|
|
|
{
|
|
|
|
|
xfree (prkdf->objid);
|
|
|
|
|
xfree (prkdf->authid);
|
|
|
|
|
xfree (prkdf->label);
|
|
|
|
|
xfree (prkdf);
|
|
|
|
|
}
|
|
|
|
|
err = 0;
|
|
|
|
|
|
|
|
|
|
next_record:
|
|
|
|
|
/* If the card uses a record oriented file structure, read the
|
|
|
|
|
* next record. Otherwise we keep on parsing the current buffer. */
|
|
|
|
|
recno++;
|
|
|
|
|
if (record_mode)
|
|
|
|
|
{
|
|
|
|
|
xfree (buffer); buffer = NULL;
|
|
|
|
|
err = select_and_read_record (app, 0, recno, "PrKDF",
|
|
|
|
|
&buffer, &buflen, NULL);
|
|
|
|
|
if (err) {
|
|
|
|
|
if (gpg_err_code (err) == GPG_ERR_NOT_FOUND)
|
|
|
|
|
err = 0;
|
|
|
|
|
goto leave;
|
|
|
|
|
}
|
|
|
|
|
p = buffer;
|
|
|
|
|
n = buflen;
|
|
|
|
|
}
|
|
|
|
|
} /* End looping over all records. */
|
|
|
|
|
|
|
|
|
|
leave:
|
|
|
|
|
xfree (authid);
|
|
|
|
|
xfree (label);
|
|
|
|
|
xfree (objid);
|
|
|
|
|
xfree (buffer);
|
|
|
|
|
if (err)
|
|
|
|
|
release_prkdflist (prkdflist);
|
|
|
|
|
else
|
|
|
|
|
*result = prkdflist;
|
|
|
|
|
return err;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* Read and parse the Public Keys Directory File. */
|
|
|
|
|
static gpg_error_t
|
|
|
|
|
read_ef_pukdf (app_t app, unsigned short fid, pukdf_object_t *result)
|
|
|
|
|
{
|
|
|
|
|
gpg_error_t err;
|
|
|
|
|
unsigned char *buffer;
|
|
|
|
|
size_t buflen;
|
|
|
|
|
const unsigned char *p;
|
|
|
|
|
size_t n, objlen, hdrlen;
|
|
|
|
|
int class, tag, constructed, ndef;
|
|
|
|
|
pukdf_object_t pukdflist = NULL;
|
|
|
|
|
int i;
|
|
|
|
|
int recno = 1;
|
|
|
|
|
unsigned char *authid = NULL;
|
|
|
|
|
size_t authidlen = 0;
|
|
|
|
|
unsigned char *objid = NULL;
|
|
|
|
|
size_t objidlen = 0;
|
|
|
|
|
char *label = NULL;
|
|
|
|
|
int record_mode;
|
|
|
|
|
|
|
|
|
|
err = read_first_record (app, fid, "PuKDF", &buffer, &buflen, &record_mode);
|
|
|
|
|
if (err)
|
|
|
|
|
return err;
|
|
|
|
|
|
|
|
|
|
p = buffer;
|
|
|
|
|
n = buflen;
|
|
|
|
|
|
|
|
|
|
/* Loop over the records. We stop as soon as we detect a new record
|
|
|
|
|
* starting with 0x00 or 0xff as these values are commonly used to
|
|
|
|
|
* pad data blocks and are no valid ASN.1 encoding. Note the
|
|
|
|
|
* special handling for record mode at the end of the loop. */
|
|
|
|
|
while (n && *p && *p != 0xff)
|
|
|
|
|
{
|
|
|
|
|
const unsigned char *pp;
|
|
|
|
|
size_t nn;
|
|
|
|
|
int where;
|
|
|
|
|
const char *errstr = NULL;
|
|
|
|
|
pukdf_object_t pukdf = NULL;
|
|
|
|
|
unsigned long ul;
|
|
|
|
|
keyusage_flags_t usageflags;
|
|
|
|
|
keyaccess_flags_t accessflags;
|
|
|
|
|
unsigned long key_reference = 0;
|
|
|
|
|
int key_reference_valid = 0;
|
|
|
|
|
|
|
|
|
|
where = __LINE__;
|
|
|
|
|
err = parse_ber_header (&p, &n, &class, &tag, &constructed,
|
2005-04-27 12:09:21 +00:00
|
|
|
|
&ndef, &objlen, &hdrlen);
|
2021-04-29 15:26:36 +02:00
|
|
|
|
if (err)
|
|
|
|
|
;
|
|
|
|
|
else if (objlen > n)
|
|
|
|
|
err = gpg_error (GPG_ERR_INV_OBJ);
|
|
|
|
|
else if (class == CLASS_UNIVERSAL && tag == TAG_SEQUENCE)
|
|
|
|
|
; /* PublicRSAKeyAttributes */
|
|
|
|
|
else if (class == CLASS_CONTEXT)
|
|
|
|
|
{
|
|
|
|
|
switch (tag)
|
|
|
|
|
{
|
|
|
|
|
case 0: break; /* EC key object */
|
|
|
|
|
case 1: errstr = "DH key objects are not supported"; break;
|
|
|
|
|
case 2: errstr = "DSA key objects are not supported"; break;
|
|
|
|
|
case 3: errstr = "KEA key objects are not supported"; break;
|
|
|
|
|
default: errstr = "unknown publicKeyObject"; break;
|
|
|
|
|
}
|
|
|
|
|
if (errstr)
|
|
|
|
|
goto parse_error;
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
2005-04-27 12:09:21 +00:00
|
|
|
|
err = gpg_error (GPG_ERR_INV_OBJ);
|
|
|
|
|
goto parse_error;
|
2021-04-29 15:26:36 +02:00
|
|
|
|
}
|
2005-04-27 12:09:21 +00:00
|
|
|
|
|
2021-04-29 15:26:36 +02:00
|
|
|
|
if (err)
|
|
|
|
|
{
|
|
|
|
|
log_error ("p15: error parsing PuKDF record: %s\n",
|
|
|
|
|
gpg_strerror (err));
|
|
|
|
|
goto leave;
|
|
|
|
|
}
|
2005-04-27 12:09:21 +00:00
|
|
|
|
|
2021-04-29 15:26:36 +02:00
|
|
|
|
pp = p;
|
|
|
|
|
nn = objlen;
|
|
|
|
|
p += objlen;
|
|
|
|
|
n -= objlen;
|
2004-12-02 07:48:09 +00:00
|
|
|
|
|
2021-04-29 15:26:36 +02:00
|
|
|
|
/* Parse the commonObjectAttributes. */
|
|
|
|
|
where = __LINE__;
|
|
|
|
|
xfree (authid);
|
|
|
|
|
xfree (label);
|
|
|
|
|
err = parse_common_obj_attr (&pp, &nn, &authid, &authidlen, &label);
|
|
|
|
|
if (err)
|
|
|
|
|
goto parse_error;
|
2004-12-02 07:48:09 +00:00
|
|
|
|
|
2021-04-29 15:26:36 +02:00
|
|
|
|
/* Parse the commonKeyAttributes. */
|
|
|
|
|
where = __LINE__;
|
|
|
|
|
xfree (objid);
|
|
|
|
|
err = parse_common_key_attr (&pp, &nn,
|
|
|
|
|
&objid, &objidlen,
|
|
|
|
|
&usageflags, &accessflags,
|
|
|
|
|
&key_reference, &key_reference_valid);
|
|
|
|
|
if (err)
|
|
|
|
|
goto parse_error;
|
|
|
|
|
log_assert (objid);
|
2004-12-02 07:48:09 +00:00
|
|
|
|
|
2021-04-29 15:26:36 +02:00
|
|
|
|
/* Parse the subClassAttributes. */
|
2005-04-27 12:09:21 +00:00
|
|
|
|
where = __LINE__;
|
|
|
|
|
err = parse_ber_header (&pp, &nn, &class, &tag, &constructed,
|
|
|
|
|
&ndef, &objlen, &hdrlen);
|
|
|
|
|
if (!err && objlen > nn)
|
|
|
|
|
err = gpg_error (GPG_ERR_INV_OBJ);
|
|
|
|
|
if (err)
|
|
|
|
|
goto parse_error;
|
|
|
|
|
if (class == CLASS_CONTEXT && tag == 0)
|
|
|
|
|
{
|
2021-04-29 15:26:36 +02:00
|
|
|
|
/* Skip this CommonPublicKeyAttribute. */
|
2005-04-27 12:09:21 +00:00
|
|
|
|
pp += objlen;
|
|
|
|
|
nn -= objlen;
|
2011-02-04 12:57:53 +01:00
|
|
|
|
|
2005-04-27 12:09:21 +00:00
|
|
|
|
where = __LINE__;
|
|
|
|
|
err = parse_ber_header (&pp, &nn, &class, &tag, &constructed,
|
|
|
|
|
&ndef, &objlen, &hdrlen);
|
|
|
|
|
}
|
2021-04-29 15:26:36 +02:00
|
|
|
|
/* We expect a typeAttribute. */
|
2005-04-27 12:09:21 +00:00
|
|
|
|
if (!err && (objlen > nn || class != CLASS_CONTEXT || tag != 1))
|
|
|
|
|
err = gpg_error (GPG_ERR_INV_OBJ);
|
|
|
|
|
if (err)
|
2021-04-29 15:26:36 +02:00
|
|
|
|
goto parse_error; /* No typeAttribute. */
|
2005-04-27 12:09:21 +00:00
|
|
|
|
nn = objlen;
|
2004-12-02 07:48:09 +00:00
|
|
|
|
|
2005-04-27 12:09:21 +00:00
|
|
|
|
where = __LINE__;
|
|
|
|
|
err = parse_ber_header (&pp, &nn, &class, &tag, &constructed,
|
|
|
|
|
&ndef, &objlen, &hdrlen);
|
2021-04-29 15:26:36 +02:00
|
|
|
|
if (err)
|
|
|
|
|
;
|
|
|
|
|
else if (!err && objlen > nn)
|
|
|
|
|
err = gpg_error (GPG_ERR_INV_OBJ);
|
|
|
|
|
else if (class == CLASS_UNIVERSAL && tag == TAG_SEQUENCE)
|
|
|
|
|
; /* A typeAttribute always starts with a sequence. */
|
|
|
|
|
else
|
2005-04-27 12:09:21 +00:00
|
|
|
|
err = gpg_error (GPG_ERR_INV_OBJ);
|
|
|
|
|
if (err)
|
|
|
|
|
goto parse_error;
|
2004-12-02 07:48:09 +00:00
|
|
|
|
|
2005-04-27 12:09:21 +00:00
|
|
|
|
nn = objlen;
|
2004-12-02 07:48:09 +00:00
|
|
|
|
|
2005-04-27 12:09:21 +00:00
|
|
|
|
/* Check that the reference is a Path object. */
|
|
|
|
|
where = __LINE__;
|
|
|
|
|
err = parse_ber_header (&pp, &nn, &class, &tag, &constructed,
|
|
|
|
|
&ndef, &objlen, &hdrlen);
|
|
|
|
|
if (!err && objlen > nn)
|
|
|
|
|
err = gpg_error (GPG_ERR_INV_OBJ);
|
|
|
|
|
if (err)
|
|
|
|
|
goto parse_error;
|
|
|
|
|
if (class != CLASS_UNIVERSAL || tag != TAG_SEQUENCE)
|
|
|
|
|
{
|
|
|
|
|
errstr = "unsupported reference type";
|
|
|
|
|
goto parse_error;
|
|
|
|
|
}
|
|
|
|
|
nn = objlen;
|
2004-12-02 07:48:09 +00:00
|
|
|
|
|
2005-04-27 12:09:21 +00:00
|
|
|
|
/* Parse the Path object. */
|
|
|
|
|
where = __LINE__;
|
|
|
|
|
err = parse_ber_header (&pp, &nn, &class, &tag, &constructed,
|
|
|
|
|
&ndef, &objlen, &hdrlen);
|
|
|
|
|
if (!err && objlen > nn)
|
|
|
|
|
err = gpg_error (GPG_ERR_INV_OBJ);
|
|
|
|
|
if (err)
|
|
|
|
|
goto parse_error;
|
2004-12-02 07:48:09 +00:00
|
|
|
|
|
2005-04-27 12:09:21 +00:00
|
|
|
|
/* Make sure that the next element is a non zero path and of
|
|
|
|
|
even length (FID are two bytes each). */
|
|
|
|
|
if (class != CLASS_UNIVERSAL || tag != TAG_OCTET_STRING
|
|
|
|
|
|| !objlen || (objlen & 1) )
|
|
|
|
|
{
|
|
|
|
|
errstr = "invalid path reference";
|
|
|
|
|
goto parse_error;
|
|
|
|
|
}
|
2021-04-29 15:26:36 +02:00
|
|
|
|
|
|
|
|
|
/* Create a new PuKDF list item. */
|
|
|
|
|
pukdf = xtrycalloc (1, (sizeof *pukdf
|
2005-04-27 12:09:21 +00:00
|
|
|
|
- sizeof(unsigned short)
|
|
|
|
|
+ objlen/2 * sizeof(unsigned short)));
|
2021-04-29 15:26:36 +02:00
|
|
|
|
if (!pukdf)
|
2005-04-27 12:09:21 +00:00
|
|
|
|
{
|
2006-09-14 16:50:33 +00:00
|
|
|
|
err = gpg_error_from_syserror ();
|
2005-04-27 12:09:21 +00:00
|
|
|
|
goto leave;
|
|
|
|
|
}
|
2021-04-29 15:26:36 +02:00
|
|
|
|
pukdf->objidlen = objidlen;
|
|
|
|
|
pukdf->objid = objid;
|
|
|
|
|
objid = NULL;
|
|
|
|
|
if (authid)
|
2005-04-27 12:09:21 +00:00
|
|
|
|
{
|
2021-04-29 15:26:36 +02:00
|
|
|
|
pukdf->authidlen = authidlen;
|
|
|
|
|
pukdf->authid = authid;
|
|
|
|
|
authid = NULL;
|
2005-04-27 12:09:21 +00:00
|
|
|
|
}
|
2021-04-29 15:26:36 +02:00
|
|
|
|
if (label)
|
2005-04-27 12:09:21 +00:00
|
|
|
|
{
|
2021-04-29 15:26:36 +02:00
|
|
|
|
pukdf->label = label;
|
|
|
|
|
label = NULL;
|
2005-04-27 12:09:21 +00:00
|
|
|
|
}
|
2004-12-02 07:48:09 +00:00
|
|
|
|
|
2021-04-29 15:26:36 +02:00
|
|
|
|
pukdf->pathlen = objlen/2;
|
|
|
|
|
for (i=0; i < pukdf->pathlen; i++, pp += 2, nn -= 2)
|
|
|
|
|
pukdf->path[i] = ((pp[0] << 8) | pp[1]);
|
2004-12-02 07:48:09 +00:00
|
|
|
|
|
2021-04-29 15:26:36 +02:00
|
|
|
|
pukdf->usageflags = usageflags;
|
|
|
|
|
pukdf->accessflags = accessflags;
|
|
|
|
|
pukdf->key_reference = key_reference;
|
|
|
|
|
pukdf->key_reference_valid = key_reference_valid;
|
2004-12-02 07:48:09 +00:00
|
|
|
|
|
2005-04-27 12:09:21 +00:00
|
|
|
|
if (nn)
|
|
|
|
|
{
|
|
|
|
|
/* An index and length follows. */
|
2021-04-29 15:26:36 +02:00
|
|
|
|
pukdf->have_off = 1;
|
2005-04-27 12:09:21 +00:00
|
|
|
|
where = __LINE__;
|
|
|
|
|
err = parse_ber_header (&pp, &nn, &class, &tag, &constructed,
|
|
|
|
|
&ndef, &objlen, &hdrlen);
|
|
|
|
|
if (!err && (objlen > nn
|
|
|
|
|
|| class != CLASS_UNIVERSAL || tag != TAG_INTEGER))
|
|
|
|
|
err = gpg_error (GPG_ERR_INV_OBJ);
|
|
|
|
|
if (err)
|
|
|
|
|
goto parse_error;
|
2011-02-04 12:57:53 +01:00
|
|
|
|
|
2005-04-27 12:09:21 +00:00
|
|
|
|
for (ul=0; objlen; objlen--)
|
|
|
|
|
{
|
|
|
|
|
ul <<= 8;
|
2011-02-04 12:57:53 +01:00
|
|
|
|
ul |= (*pp++) & 0xff;
|
2005-04-27 12:09:21 +00:00
|
|
|
|
nn--;
|
|
|
|
|
}
|
2021-04-29 15:26:36 +02:00
|
|
|
|
pukdf->off = ul;
|
2011-02-04 12:57:53 +01:00
|
|
|
|
|
2005-04-27 12:09:21 +00:00
|
|
|
|
where = __LINE__;
|
|
|
|
|
err = parse_ber_header (&pp, &nn, &class, &tag, &constructed,
|
|
|
|
|
&ndef, &objlen, &hdrlen);
|
|
|
|
|
if (!err && (objlen > nn
|
|
|
|
|
|| class != CLASS_CONTEXT || tag != 0))
|
|
|
|
|
err = gpg_error (GPG_ERR_INV_OBJ);
|
|
|
|
|
if (err)
|
|
|
|
|
goto parse_error;
|
2011-02-04 12:57:53 +01:00
|
|
|
|
|
2005-04-27 12:09:21 +00:00
|
|
|
|
for (ul=0; objlen; objlen--)
|
|
|
|
|
{
|
|
|
|
|
ul <<= 8;
|
2011-02-04 12:57:53 +01:00
|
|
|
|
ul |= (*pp++) & 0xff;
|
2005-04-27 12:09:21 +00:00
|
|
|
|
nn--;
|
|
|
|
|
}
|
2021-04-29 15:26:36 +02:00
|
|
|
|
pukdf->len = ul;
|
2005-04-27 12:09:21 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2020-03-31 10:40:21 +02:00
|
|
|
|
if (opt.verbose)
|
|
|
|
|
{
|
2021-04-29 15:26:36 +02:00
|
|
|
|
log_info ("p15: PuKDF %04hX: id=", fid);
|
|
|
|
|
for (i=0; i < pukdf->objidlen; i++)
|
|
|
|
|
log_printf ("%02X", pukdf->objid[i]);
|
|
|
|
|
if (pukdf->label)
|
|
|
|
|
log_printf (" (%s)", pukdf->label);
|
|
|
|
|
log_info ("p15: path=");
|
|
|
|
|
for (i=0; i < pukdf->pathlen; i++)
|
|
|
|
|
log_printf ("%s%04hX", i?"/":"",pukdf->path[i]);
|
|
|
|
|
if (pukdf->have_off)
|
|
|
|
|
log_printf ("[%lu/%lu]", pukdf->off, pukdf->len);
|
|
|
|
|
if (pukdf->authid)
|
2020-03-31 10:40:21 +02:00
|
|
|
|
{
|
|
|
|
|
log_printf (" authid=");
|
2021-04-29 15:26:36 +02:00
|
|
|
|
for (i=0; i < pukdf->authidlen; i++)
|
|
|
|
|
log_printf ("%02X", pukdf->authid[i]);
|
2020-03-31 10:40:21 +02:00
|
|
|
|
}
|
2021-04-29 15:26:36 +02:00
|
|
|
|
if (pukdf->key_reference_valid)
|
|
|
|
|
log_printf (" keyref=0x%02lX", pukdf->key_reference);
|
|
|
|
|
if (pukdf->accessflags.any)
|
|
|
|
|
dump_keyaccess_flags (pukdf->accessflags);
|
|
|
|
|
dump_keyusage_flags (pukdf->usageflags);
|
2020-03-31 10:40:21 +02:00
|
|
|
|
log_printf ("\n");
|
|
|
|
|
}
|
2011-02-04 12:57:53 +01:00
|
|
|
|
|
2005-04-27 12:09:21 +00:00
|
|
|
|
/* Put it into the list. */
|
2021-04-29 15:26:36 +02:00
|
|
|
|
pukdf->next = pukdflist;
|
|
|
|
|
pukdflist = pukdf;
|
|
|
|
|
pukdf = NULL;
|
|
|
|
|
goto next_record; /* Ready with this record. */
|
2005-04-27 12:09:21 +00:00
|
|
|
|
|
|
|
|
|
parse_error:
|
2021-04-29 15:26:36 +02:00
|
|
|
|
log_error ("p15: error parsing PuKDF record at %d: %s - skipped\n",
|
2005-04-27 12:09:21 +00:00
|
|
|
|
where, errstr? errstr : gpg_strerror (err));
|
2021-04-29 15:26:36 +02:00
|
|
|
|
if (pukdf)
|
2005-04-27 12:09:21 +00:00
|
|
|
|
{
|
2021-04-29 15:26:36 +02:00
|
|
|
|
xfree (pukdf->objid);
|
|
|
|
|
xfree (pukdf->authid);
|
|
|
|
|
xfree (pukdf->label);
|
|
|
|
|
xfree (pukdf);
|
2005-04-27 12:09:21 +00:00
|
|
|
|
}
|
|
|
|
|
err = 0;
|
2021-04-29 15:26:36 +02:00
|
|
|
|
|
|
|
|
|
next_record:
|
|
|
|
|
/* If the card uses a record oriented file structure, read the
|
|
|
|
|
* next record. Otherwise we keep on parsing the current buffer. */
|
|
|
|
|
recno++;
|
|
|
|
|
if (record_mode)
|
|
|
|
|
{
|
|
|
|
|
xfree (buffer); buffer = NULL;
|
|
|
|
|
err = select_and_read_record (app, 0, recno, "PuKDF",
|
|
|
|
|
&buffer, &buflen, NULL);
|
|
|
|
|
if (err) {
|
|
|
|
|
if (gpg_err_code (err) == GPG_ERR_NOT_FOUND)
|
|
|
|
|
err = 0;
|
|
|
|
|
goto leave;
|
|
|
|
|
}
|
|
|
|
|
p = buffer;
|
|
|
|
|
n = buflen;
|
|
|
|
|
}
|
2005-04-27 12:09:21 +00:00
|
|
|
|
} /* End looping over all records. */
|
|
|
|
|
|
|
|
|
|
leave:
|
2021-04-29 15:26:36 +02:00
|
|
|
|
xfree (authid);
|
|
|
|
|
xfree (label);
|
|
|
|
|
xfree (objid);
|
2005-04-27 12:09:21 +00:00
|
|
|
|
xfree (buffer);
|
|
|
|
|
if (err)
|
2021-04-29 15:26:36 +02:00
|
|
|
|
release_pukdflist (pukdflist);
|
2005-04-27 12:09:21 +00:00
|
|
|
|
else
|
2021-04-29 15:26:36 +02:00
|
|
|
|
*result = pukdflist;
|
2005-04-27 12:09:21 +00:00
|
|
|
|
return err;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* Read and parse the Certificate Directory Files identified by FID.
|
|
|
|
|
On success a newlist of CDF object gets stored at RESULT and the
|
|
|
|
|
caller is then responsible of releasing this list. On error a
|
|
|
|
|
error code is returned and RESULT won't get changed. */
|
|
|
|
|
static gpg_error_t
|
2021-04-29 15:26:36 +02:00
|
|
|
|
read_ef_cdf (app_t app, unsigned short fid, int cdftype, cdf_object_t *result)
|
2005-04-27 12:09:21 +00:00
|
|
|
|
{
|
|
|
|
|
gpg_error_t err;
|
2021-04-29 15:26:36 +02:00
|
|
|
|
unsigned char *buffer;
|
2005-04-27 12:09:21 +00:00
|
|
|
|
size_t buflen;
|
|
|
|
|
const unsigned char *p;
|
|
|
|
|
size_t n, objlen, hdrlen;
|
|
|
|
|
int class, tag, constructed, ndef;
|
|
|
|
|
cdf_object_t cdflist = NULL;
|
|
|
|
|
int i;
|
2021-04-29 15:26:36 +02:00
|
|
|
|
int recno = 1;
|
|
|
|
|
unsigned char *authid = NULL;
|
|
|
|
|
size_t authidlen = 0;
|
|
|
|
|
char *label = NULL;
|
|
|
|
|
int record_mode;
|
2011-02-04 12:57:53 +01:00
|
|
|
|
|
2021-04-29 15:26:36 +02:00
|
|
|
|
err = read_first_record (app, fid, "CDF", &buffer, &buflen, &record_mode);
|
2005-04-27 12:09:21 +00:00
|
|
|
|
if (err)
|
|
|
|
|
return err;
|
2011-02-04 12:57:53 +01:00
|
|
|
|
|
2005-04-27 12:09:21 +00:00
|
|
|
|
p = buffer;
|
|
|
|
|
n = buflen;
|
|
|
|
|
|
|
|
|
|
/* Loop over the records. We stop as soon as we detect a new record
|
|
|
|
|
starting with 0x00 or 0xff as these values are commonly used to
|
2021-04-29 15:26:36 +02:00
|
|
|
|
pad data blocks and are no valid ASN.1 encoding. Note the
|
|
|
|
|
special handling for record mode at the end of the loop. */
|
2005-04-27 12:09:21 +00:00
|
|
|
|
while (n && *p && *p != 0xff)
|
|
|
|
|
{
|
|
|
|
|
const unsigned char *pp;
|
|
|
|
|
size_t nn;
|
|
|
|
|
int where;
|
|
|
|
|
const char *errstr = NULL;
|
|
|
|
|
cdf_object_t cdf = NULL;
|
|
|
|
|
unsigned long ul;
|
|
|
|
|
const unsigned char *objid;
|
|
|
|
|
size_t objidlen;
|
2011-02-04 12:57:53 +01:00
|
|
|
|
|
2005-04-27 12:09:21 +00:00
|
|
|
|
err = parse_ber_header (&p, &n, &class, &tag, &constructed,
|
|
|
|
|
&ndef, &objlen, &hdrlen);
|
|
|
|
|
if (!err && (objlen > n || tag != TAG_SEQUENCE))
|
|
|
|
|
err = gpg_error (GPG_ERR_INV_OBJ);
|
|
|
|
|
if (err)
|
|
|
|
|
{
|
2020-03-31 10:40:21 +02:00
|
|
|
|
log_error ("p15: error parsing CDF record: %s\n", gpg_strerror (err));
|
2005-04-27 12:09:21 +00:00
|
|
|
|
goto leave;
|
|
|
|
|
}
|
|
|
|
|
pp = p;
|
|
|
|
|
nn = objlen;
|
|
|
|
|
p += objlen;
|
|
|
|
|
n -= objlen;
|
|
|
|
|
|
2021-04-29 15:26:36 +02:00
|
|
|
|
/* Parse the commonObjectAttributes. */
|
2005-04-27 12:09:21 +00:00
|
|
|
|
where = __LINE__;
|
2021-04-29 15:26:36 +02:00
|
|
|
|
xfree (authid);
|
|
|
|
|
xfree (label);
|
|
|
|
|
err = parse_common_obj_attr (&pp, &nn, &authid, &authidlen, &label);
|
2005-04-27 12:09:21 +00:00
|
|
|
|
if (err)
|
|
|
|
|
goto parse_error;
|
|
|
|
|
|
|
|
|
|
/* Parse the commonCertificateAttributes. */
|
|
|
|
|
where = __LINE__;
|
|
|
|
|
err = parse_ber_header (&pp, &nn, &class, &tag, &constructed,
|
|
|
|
|
&ndef, &objlen, &hdrlen);
|
|
|
|
|
if (!err && (objlen > nn || tag != TAG_SEQUENCE))
|
|
|
|
|
err = gpg_error (GPG_ERR_INV_OBJ);
|
|
|
|
|
if (err)
|
|
|
|
|
goto parse_error;
|
|
|
|
|
{
|
|
|
|
|
const unsigned char *ppp = pp;
|
|
|
|
|
size_t nnn = objlen;
|
|
|
|
|
|
|
|
|
|
pp += objlen;
|
|
|
|
|
nn -= objlen;
|
|
|
|
|
|
|
|
|
|
/* Get the Id. */
|
|
|
|
|
where = __LINE__;
|
|
|
|
|
err = parse_ber_header (&ppp, &nnn, &class, &tag, &constructed,
|
|
|
|
|
&ndef, &objlen, &hdrlen);
|
|
|
|
|
if (!err && (objlen > nnn
|
|
|
|
|
|| class != CLASS_UNIVERSAL || tag != TAG_OCTET_STRING))
|
|
|
|
|
err = gpg_error (GPG_ERR_INV_OBJ);
|
|
|
|
|
if (err)
|
|
|
|
|
goto parse_error;
|
|
|
|
|
objid = ppp;
|
|
|
|
|
objidlen = objlen;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Parse the certAttribute. */
|
|
|
|
|
where = __LINE__;
|
|
|
|
|
err = parse_ber_header (&pp, &nn, &class, &tag, &constructed,
|
|
|
|
|
&ndef, &objlen, &hdrlen);
|
|
|
|
|
if (!err && (objlen > nn || class != CLASS_CONTEXT || tag != 1))
|
|
|
|
|
err = gpg_error (GPG_ERR_INV_OBJ);
|
|
|
|
|
if (err)
|
|
|
|
|
goto parse_error;
|
|
|
|
|
nn = objlen;
|
|
|
|
|
|
|
|
|
|
where = __LINE__;
|
|
|
|
|
err = parse_ber_header (&pp, &nn, &class, &tag, &constructed,
|
|
|
|
|
&ndef, &objlen, &hdrlen);
|
|
|
|
|
if (!err && (objlen > nn
|
|
|
|
|
|| class != CLASS_UNIVERSAL || tag != TAG_SEQUENCE))
|
|
|
|
|
err = gpg_error (GPG_ERR_INV_OBJ);
|
|
|
|
|
if (err)
|
|
|
|
|
goto parse_error;
|
|
|
|
|
nn = objlen;
|
|
|
|
|
|
|
|
|
|
/* Check that the reference is a Path object. */
|
|
|
|
|
where = __LINE__;
|
|
|
|
|
err = parse_ber_header (&pp, &nn, &class, &tag, &constructed,
|
|
|
|
|
&ndef, &objlen, &hdrlen);
|
|
|
|
|
if (!err && objlen > nn)
|
|
|
|
|
err = gpg_error (GPG_ERR_INV_OBJ);
|
|
|
|
|
if (err)
|
|
|
|
|
goto parse_error;
|
|
|
|
|
if (class != CLASS_UNIVERSAL || tag != TAG_SEQUENCE)
|
|
|
|
|
{
|
|
|
|
|
errstr = "unsupported reference type";
|
2017-03-06 14:59:02 +09:00
|
|
|
|
goto parse_error;
|
2005-04-27 12:09:21 +00:00
|
|
|
|
}
|
|
|
|
|
nn = objlen;
|
|
|
|
|
|
|
|
|
|
/* Parse the Path object. */
|
|
|
|
|
where = __LINE__;
|
|
|
|
|
err = parse_ber_header (&pp, &nn, &class, &tag, &constructed,
|
|
|
|
|
&ndef, &objlen, &hdrlen);
|
|
|
|
|
if (!err && objlen > nn)
|
|
|
|
|
err = gpg_error (GPG_ERR_INV_OBJ);
|
|
|
|
|
if (err)
|
|
|
|
|
goto parse_error;
|
|
|
|
|
|
|
|
|
|
/* Make sure that the next element is a non zero path and of
|
|
|
|
|
even length (FID are two bytes each). */
|
|
|
|
|
if (class != CLASS_UNIVERSAL || tag != TAG_OCTET_STRING
|
|
|
|
|
|| !objlen || (objlen & 1) )
|
|
|
|
|
{
|
|
|
|
|
errstr = "invalid path reference";
|
|
|
|
|
goto parse_error;
|
|
|
|
|
}
|
|
|
|
|
/* Create a new CDF list item. */
|
2011-02-04 12:57:53 +01:00
|
|
|
|
cdf = xtrycalloc (1, (sizeof *cdf
|
2005-04-27 12:09:21 +00:00
|
|
|
|
- sizeof(unsigned short)
|
|
|
|
|
+ objlen/2 * sizeof(unsigned short)));
|
|
|
|
|
if (!cdf)
|
|
|
|
|
{
|
2006-09-14 16:50:33 +00:00
|
|
|
|
err = gpg_error_from_syserror ();
|
2005-04-27 12:09:21 +00:00
|
|
|
|
goto leave;
|
|
|
|
|
}
|
2021-04-29 15:26:36 +02:00
|
|
|
|
if (authid)
|
|
|
|
|
{
|
|
|
|
|
cdf->authidlen = authidlen;
|
|
|
|
|
cdf->authid = authid;
|
|
|
|
|
authid = NULL;
|
|
|
|
|
}
|
|
|
|
|
if (label)
|
|
|
|
|
{
|
|
|
|
|
cdf->label = label;
|
|
|
|
|
label = NULL;
|
|
|
|
|
}
|
|
|
|
|
|
2005-04-27 12:09:21 +00:00
|
|
|
|
cdf->objidlen = objidlen;
|
|
|
|
|
cdf->objid = xtrymalloc (objidlen);
|
|
|
|
|
if (!cdf->objid)
|
|
|
|
|
{
|
2006-09-14 16:50:33 +00:00
|
|
|
|
err = gpg_error_from_syserror ();
|
2005-04-27 12:09:21 +00:00
|
|
|
|
xfree (cdf);
|
|
|
|
|
goto leave;
|
|
|
|
|
}
|
|
|
|
|
memcpy (cdf->objid, objid, objidlen);
|
|
|
|
|
|
|
|
|
|
cdf->pathlen = objlen/2;
|
|
|
|
|
for (i=0; i < cdf->pathlen; i++, pp += 2, nn -= 2)
|
|
|
|
|
cdf->path[i] = ((pp[0] << 8) | pp[1]);
|
|
|
|
|
|
|
|
|
|
if (nn)
|
|
|
|
|
{
|
|
|
|
|
/* An index and length follows. */
|
|
|
|
|
cdf->have_off = 1;
|
|
|
|
|
where = __LINE__;
|
|
|
|
|
err = parse_ber_header (&pp, &nn, &class, &tag, &constructed,
|
|
|
|
|
&ndef, &objlen, &hdrlen);
|
|
|
|
|
if (!err && (objlen > nn
|
|
|
|
|
|| class != CLASS_UNIVERSAL || tag != TAG_INTEGER))
|
|
|
|
|
err = gpg_error (GPG_ERR_INV_OBJ);
|
|
|
|
|
if (err)
|
|
|
|
|
goto parse_error;
|
2011-02-04 12:57:53 +01:00
|
|
|
|
|
2005-04-27 12:09:21 +00:00
|
|
|
|
for (ul=0; objlen; objlen--)
|
|
|
|
|
{
|
|
|
|
|
ul <<= 8;
|
2011-02-04 12:57:53 +01:00
|
|
|
|
ul |= (*pp++) & 0xff;
|
2005-04-27 12:09:21 +00:00
|
|
|
|
nn--;
|
|
|
|
|
}
|
|
|
|
|
cdf->off = ul;
|
2011-02-04 12:57:53 +01:00
|
|
|
|
|
2005-04-27 12:09:21 +00:00
|
|
|
|
where = __LINE__;
|
|
|
|
|
err = parse_ber_header (&pp, &nn, &class, &tag, &constructed,
|
|
|
|
|
&ndef, &objlen, &hdrlen);
|
|
|
|
|
if (!err && (objlen > nn
|
|
|
|
|
|| class != CLASS_CONTEXT || tag != 0))
|
|
|
|
|
err = gpg_error (GPG_ERR_INV_OBJ);
|
|
|
|
|
if (err)
|
|
|
|
|
goto parse_error;
|
2011-02-04 12:57:53 +01:00
|
|
|
|
|
2005-04-27 12:09:21 +00:00
|
|
|
|
for (ul=0; objlen; objlen--)
|
|
|
|
|
{
|
|
|
|
|
ul <<= 8;
|
2011-02-04 12:57:53 +01:00
|
|
|
|
ul |= (*pp++) & 0xff;
|
2005-04-27 12:09:21 +00:00
|
|
|
|
nn--;
|
|
|
|
|
}
|
|
|
|
|
cdf->len = ul;
|
|
|
|
|
}
|
|
|
|
|
|
2020-03-31 10:40:21 +02:00
|
|
|
|
if (opt.verbose)
|
|
|
|
|
{
|
2021-04-29 15:26:36 +02:00
|
|
|
|
log_info ("p15: CDF-%c %04hX: id=", cdftype, fid);
|
2020-03-31 10:40:21 +02:00
|
|
|
|
for (i=0; i < cdf->objidlen; i++)
|
|
|
|
|
log_printf ("%02X", cdf->objid[i]);
|
2021-04-29 15:26:36 +02:00
|
|
|
|
if (cdf->label)
|
|
|
|
|
log_printf (" (%s)", cdf->label);
|
|
|
|
|
log_info ("p15: path=");
|
2020-03-31 10:40:21 +02:00
|
|
|
|
for (i=0; i < cdf->pathlen; i++)
|
|
|
|
|
log_printf ("%s%04hX", i?"/":"", cdf->path[i]);
|
|
|
|
|
if (cdf->have_off)
|
|
|
|
|
log_printf ("[%lu/%lu]", cdf->off, cdf->len);
|
2021-04-29 15:26:36 +02:00
|
|
|
|
if (cdf->authid)
|
|
|
|
|
{
|
|
|
|
|
log_printf (" authid=");
|
|
|
|
|
for (i=0; i < cdf->authidlen; i++)
|
|
|
|
|
log_printf ("%02X", cdf->authid[i]);
|
|
|
|
|
}
|
2020-03-31 10:40:21 +02:00
|
|
|
|
log_printf ("\n");
|
|
|
|
|
}
|
2011-02-04 12:57:53 +01:00
|
|
|
|
|
2005-04-27 12:09:21 +00:00
|
|
|
|
/* Put it into the list. */
|
|
|
|
|
cdf->next = cdflist;
|
|
|
|
|
cdflist = cdf;
|
|
|
|
|
cdf = NULL;
|
2021-04-29 15:26:36 +02:00
|
|
|
|
goto next_record; /* Ready with this record. */
|
2005-04-27 12:09:21 +00:00
|
|
|
|
|
|
|
|
|
parse_error:
|
2021-04-29 15:26:36 +02:00
|
|
|
|
log_error ("p15: error parsing CDF record at %d: %s - skipped\n",
|
2005-04-27 12:09:21 +00:00
|
|
|
|
where, errstr? errstr : gpg_strerror (err));
|
|
|
|
|
xfree (cdf);
|
|
|
|
|
err = 0;
|
2021-04-29 15:26:36 +02:00
|
|
|
|
|
|
|
|
|
next_record:
|
|
|
|
|
xfree (authid);
|
|
|
|
|
xfree (label);
|
|
|
|
|
/* If the card uses a record oriented file structure, read the
|
|
|
|
|
* next record. Otherwise we keep on parsing the current buffer. */
|
|
|
|
|
recno++;
|
|
|
|
|
if (record_mode)
|
|
|
|
|
{
|
|
|
|
|
xfree (buffer); buffer = NULL;
|
|
|
|
|
err = select_and_read_record (app, 0, recno, "CDF",
|
|
|
|
|
&buffer, &buflen, NULL);
|
|
|
|
|
if (err) {
|
|
|
|
|
if (gpg_err_code (err) == GPG_ERR_NOT_FOUND)
|
|
|
|
|
err = 0;
|
|
|
|
|
goto leave;
|
|
|
|
|
}
|
|
|
|
|
p = buffer;
|
|
|
|
|
n = buflen;
|
|
|
|
|
}
|
|
|
|
|
} /* End loop over all records. */
|
2005-04-27 12:09:21 +00:00
|
|
|
|
|
|
|
|
|
leave:
|
2021-04-29 15:26:36 +02:00
|
|
|
|
xfree (authid);
|
|
|
|
|
xfree (label);
|
2005-04-27 12:09:21 +00:00
|
|
|
|
xfree (buffer);
|
|
|
|
|
if (err)
|
|
|
|
|
release_cdflist (cdflist);
|
|
|
|
|
else
|
|
|
|
|
*result = cdflist;
|
|
|
|
|
return err;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/*
|
2021-06-22 11:11:46 +02:00
|
|
|
|
* SEQUENCE {
|
|
|
|
|
* SEQUENCE { -- CommonObjectAttributes
|
|
|
|
|
* UTF8String 'specific PIN for DS'
|
|
|
|
|
* BIT STRING 0 unused bits
|
|
|
|
|
* '00000011'B
|
|
|
|
|
* }
|
|
|
|
|
* SEQUENCE { -- CommonAuthenticationObjectAttributes
|
|
|
|
|
* OCTET STRING
|
|
|
|
|
* 07 -- iD
|
|
|
|
|
* }
|
|
|
|
|
*
|
|
|
|
|
* [1] { -- typeAttributes
|
|
|
|
|
* SEQUENCE { -- PinAttributes
|
|
|
|
|
* BIT STRING 0 unused bits
|
|
|
|
|
* '0000100000110010'B -- local,initialized,needs-padding
|
|
|
|
|
* -- exchangeRefData
|
|
|
|
|
* ENUMERATED 1 -- ascii-numeric
|
|
|
|
|
* INTEGER 6 -- minLength
|
|
|
|
|
* INTEGER 6 -- storedLength
|
|
|
|
|
* INTEGER 8 -- maxLength
|
|
|
|
|
* [0]
|
|
|
|
|
* 02 -- pinReference
|
|
|
|
|
* GeneralizedTime 19/04/2002 12:12 GMT -- lastPinChange
|
|
|
|
|
* SEQUENCE {
|
|
|
|
|
* OCTET STRING
|
|
|
|
|
* 3F 00 40 16 -- path to DF of PIN
|
|
|
|
|
* }
|
|
|
|
|
* }
|
|
|
|
|
* }
|
|
|
|
|
* }
|
|
|
|
|
*
|
|
|
|
|
* Or for an authKey:
|
|
|
|
|
*
|
|
|
|
|
* [1] { -- typeAttributes
|
|
|
|
|
* SEQUENCE { -- AuthKeyAttributes
|
|
|
|
|
* BOOLEAN TRUE -- derivedKey
|
|
|
|
|
* OCTET STRING 02 -- authKeyId
|
|
|
|
|
* }
|
|
|
|
|
* }
|
|
|
|
|
* }
|
2005-04-27 12:09:21 +00:00
|
|
|
|
*/
|
|
|
|
|
/* Read and parse an Authentication Object Directory File identified
|
|
|
|
|
by FID. On success a newlist of AODF objects gets stored at RESULT
|
|
|
|
|
and the caller is responsible of releasing this list. On error a
|
|
|
|
|
error code is returned and RESULT won't get changed. */
|
|
|
|
|
static gpg_error_t
|
|
|
|
|
read_ef_aodf (app_t app, unsigned short fid, aodf_object_t *result)
|
|
|
|
|
{
|
|
|
|
|
gpg_error_t err;
|
2021-04-29 15:26:36 +02:00
|
|
|
|
unsigned char *buffer;
|
2005-04-27 12:09:21 +00:00
|
|
|
|
size_t buflen;
|
|
|
|
|
const unsigned char *p;
|
|
|
|
|
size_t n, objlen, hdrlen;
|
|
|
|
|
int class, tag, constructed, ndef;
|
|
|
|
|
aodf_object_t aodflist = NULL;
|
|
|
|
|
int i;
|
2021-04-29 15:26:36 +02:00
|
|
|
|
int recno = 1;
|
|
|
|
|
int record_mode;
|
2011-02-04 12:57:53 +01:00
|
|
|
|
|
2021-04-29 15:26:36 +02:00
|
|
|
|
err = read_first_record (app, fid, "AODF", &buffer, &buflen, &record_mode);
|
2005-04-27 12:09:21 +00:00
|
|
|
|
if (err)
|
|
|
|
|
return err;
|
2011-02-04 12:57:53 +01:00
|
|
|
|
|
2005-04-27 12:09:21 +00:00
|
|
|
|
p = buffer;
|
|
|
|
|
n = buflen;
|
|
|
|
|
|
|
|
|
|
/* Loop over the records. We stop as soon as we detect a new record
|
|
|
|
|
starting with 0x00 or 0xff as these values are commonly used to
|
2021-04-29 15:26:36 +02:00
|
|
|
|
pad data blocks and are no valid ASN.1 encoding. Note the
|
|
|
|
|
special handling for record mode at the end of the loop. */
|
2005-04-27 12:09:21 +00:00
|
|
|
|
while (n && *p && *p != 0xff)
|
|
|
|
|
{
|
|
|
|
|
const unsigned char *pp;
|
|
|
|
|
size_t nn;
|
|
|
|
|
int where;
|
|
|
|
|
const char *errstr = NULL;
|
2021-06-22 11:11:46 +02:00
|
|
|
|
auth_type_t auth_type;
|
2005-04-27 12:09:21 +00:00
|
|
|
|
aodf_object_t aodf = NULL;
|
|
|
|
|
unsigned long ul;
|
|
|
|
|
const char *s;
|
|
|
|
|
|
2021-04-29 15:26:36 +02:00
|
|
|
|
where = __LINE__;
|
2005-04-27 12:09:21 +00:00
|
|
|
|
err = parse_ber_header (&p, &n, &class, &tag, &constructed,
|
|
|
|
|
&ndef, &objlen, &hdrlen);
|
2021-04-29 15:26:36 +02:00
|
|
|
|
if (err)
|
|
|
|
|
;
|
|
|
|
|
else if (objlen > n)
|
2005-04-27 12:09:21 +00:00
|
|
|
|
err = gpg_error (GPG_ERR_INV_OBJ);
|
2021-04-29 15:26:36 +02:00
|
|
|
|
else if (class == CLASS_UNIVERSAL && tag == TAG_SEQUENCE)
|
2021-06-22 11:11:46 +02:00
|
|
|
|
auth_type = AUTH_TYPE_PIN; /* PinAttributes */
|
|
|
|
|
else if (class == CLASS_CONTEXT && tag == 1 )
|
|
|
|
|
auth_type = AUTH_TYPE_AUTHKEY; /* AuthKeyAttributes */
|
2021-04-29 15:26:36 +02:00
|
|
|
|
else if (class == CLASS_CONTEXT)
|
|
|
|
|
{
|
|
|
|
|
switch (tag)
|
|
|
|
|
{
|
|
|
|
|
case 0: errstr = "biometric auth types are not supported"; break;
|
|
|
|
|
case 2: errstr = "external auth type are not supported"; break;
|
|
|
|
|
default: errstr = "unknown privateKeyObject"; break;
|
|
|
|
|
}
|
|
|
|
|
goto parse_error;
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
err = gpg_error (GPG_ERR_INV_OBJ);
|
|
|
|
|
goto parse_error;
|
|
|
|
|
}
|
|
|
|
|
|
2005-04-27 12:09:21 +00:00
|
|
|
|
if (err)
|
|
|
|
|
{
|
2020-03-31 10:40:21 +02:00
|
|
|
|
log_error ("p15: error parsing AODF record: %s\n",
|
|
|
|
|
gpg_strerror (err));
|
2005-04-27 12:09:21 +00:00
|
|
|
|
goto leave;
|
|
|
|
|
}
|
|
|
|
|
pp = p;
|
|
|
|
|
nn = objlen;
|
|
|
|
|
p += objlen;
|
|
|
|
|
n -= objlen;
|
|
|
|
|
|
|
|
|
|
/* Allocate memory for a new AODF list item. */
|
|
|
|
|
aodf = xtrycalloc (1, sizeof *aodf);
|
|
|
|
|
if (!aodf)
|
|
|
|
|
goto no_core;
|
2020-03-31 19:55:15 +02:00
|
|
|
|
aodf->fid = fid;
|
2021-06-22 11:11:46 +02:00
|
|
|
|
aodf->auth_type = auth_type;
|
2005-04-27 12:09:21 +00:00
|
|
|
|
|
|
|
|
|
/* Parse the commonObjectAttributes. */
|
|
|
|
|
where = __LINE__;
|
2021-04-29 15:26:36 +02:00
|
|
|
|
err = parse_common_obj_attr (&pp, &nn, &aodf->authid, &aodf->authidlen,
|
|
|
|
|
&aodf->label);
|
2005-04-27 12:09:21 +00:00
|
|
|
|
if (err)
|
|
|
|
|
goto parse_error;
|
|
|
|
|
|
|
|
|
|
/* Parse the CommonAuthenticationObjectAttributes. */
|
|
|
|
|
where = __LINE__;
|
|
|
|
|
err = parse_ber_header (&pp, &nn, &class, &tag, &constructed,
|
|
|
|
|
&ndef, &objlen, &hdrlen);
|
|
|
|
|
if (!err && (objlen > nn || tag != TAG_SEQUENCE))
|
|
|
|
|
err = gpg_error (GPG_ERR_INV_OBJ);
|
|
|
|
|
if (err)
|
|
|
|
|
goto parse_error;
|
|
|
|
|
{
|
|
|
|
|
const unsigned char *ppp = pp;
|
|
|
|
|
size_t nnn = objlen;
|
|
|
|
|
|
|
|
|
|
pp += objlen;
|
|
|
|
|
nn -= objlen;
|
|
|
|
|
|
|
|
|
|
/* Get the Id. */
|
|
|
|
|
where = __LINE__;
|
|
|
|
|
err = parse_ber_header (&ppp, &nnn, &class, &tag, &constructed,
|
2021-04-29 15:26:36 +02:00
|
|
|
|
&ndef, &objlen, &hdrlen);
|
2005-04-27 12:09:21 +00:00
|
|
|
|
if (!err && (objlen > nnn
|
|
|
|
|
|| class != CLASS_UNIVERSAL || tag != TAG_OCTET_STRING))
|
|
|
|
|
err = gpg_error (GPG_ERR_INV_OBJ);
|
|
|
|
|
if (err)
|
|
|
|
|
goto parse_error;
|
|
|
|
|
|
|
|
|
|
aodf->objidlen = objlen;
|
|
|
|
|
aodf->objid = xtrymalloc (objlen);
|
|
|
|
|
if (!aodf->objid)
|
|
|
|
|
goto no_core;
|
|
|
|
|
memcpy (aodf->objid, ppp, objlen);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Parse the typeAttributes. */
|
|
|
|
|
where = __LINE__;
|
|
|
|
|
err = parse_ber_header (&pp, &nn, &class, &tag, &constructed,
|
|
|
|
|
&ndef, &objlen, &hdrlen);
|
|
|
|
|
if (!err && (objlen > nn || class != CLASS_CONTEXT || tag != 1))
|
|
|
|
|
err = gpg_error (GPG_ERR_INV_OBJ);
|
|
|
|
|
if (err)
|
|
|
|
|
goto parse_error;
|
|
|
|
|
nn = objlen;
|
|
|
|
|
|
|
|
|
|
where = __LINE__;
|
|
|
|
|
err = parse_ber_header (&pp, &nn, &class, &tag, &constructed,
|
|
|
|
|
&ndef, &objlen, &hdrlen);
|
2021-04-29 15:26:36 +02:00
|
|
|
|
if (err)
|
|
|
|
|
;
|
|
|
|
|
else if (!err && objlen > nn)
|
|
|
|
|
err = gpg_error (GPG_ERR_INV_OBJ);
|
|
|
|
|
else if (class == CLASS_UNIVERSAL && tag == TAG_SEQUENCE)
|
2021-06-22 11:11:46 +02:00
|
|
|
|
; /* Okay */
|
2021-04-29 15:26:36 +02:00
|
|
|
|
else
|
2005-04-27 12:09:21 +00:00
|
|
|
|
err = gpg_error (GPG_ERR_INV_OBJ);
|
|
|
|
|
if (err)
|
|
|
|
|
goto parse_error;
|
|
|
|
|
|
|
|
|
|
nn = objlen;
|
|
|
|
|
|
2021-06-22 11:11:46 +02:00
|
|
|
|
if (auth_type == AUTH_TYPE_PIN)
|
|
|
|
|
{
|
|
|
|
|
/* PinFlags */
|
|
|
|
|
where = __LINE__;
|
|
|
|
|
err = parse_ber_header (&pp, &nn, &class, &tag, &constructed,
|
|
|
|
|
&ndef, &objlen, &hdrlen);
|
|
|
|
|
if (!err && (objlen > nn || !objlen
|
|
|
|
|
|| class != CLASS_UNIVERSAL || tag != TAG_BIT_STRING))
|
|
|
|
|
err = gpg_error (GPG_ERR_INV_OBJ);
|
|
|
|
|
if (err)
|
2005-04-27 12:09:21 +00:00
|
|
|
|
goto parse_error;
|
2021-06-22 11:11:46 +02:00
|
|
|
|
|
2005-04-27 12:09:21 +00:00
|
|
|
|
{
|
2021-06-22 11:11:46 +02:00
|
|
|
|
unsigned int bits, mask;
|
|
|
|
|
int unused, full;
|
|
|
|
|
|
|
|
|
|
unused = *pp++; nn--; objlen--;
|
|
|
|
|
if ((!objlen && unused) || unused/8 > objlen)
|
2005-04-27 12:09:21 +00:00
|
|
|
|
{
|
2021-06-22 11:11:46 +02:00
|
|
|
|
err = gpg_error (GPG_ERR_ENCODING_PROBLEM);
|
|
|
|
|
goto parse_error;
|
2005-04-27 12:09:21 +00:00
|
|
|
|
}
|
2021-06-22 11:11:46 +02:00
|
|
|
|
full = objlen - (unused+7)/8;
|
|
|
|
|
unused %= 8;
|
|
|
|
|
mask = 0;
|
|
|
|
|
for (i=1; unused; i <<= 1, unused--)
|
|
|
|
|
mask |= i;
|
|
|
|
|
|
|
|
|
|
/* The first octet */
|
|
|
|
|
bits = 0;
|
|
|
|
|
if (objlen)
|
2005-04-27 12:09:21 +00:00
|
|
|
|
{
|
2021-06-22 11:11:46 +02:00
|
|
|
|
bits = *pp++; nn--; objlen--;
|
|
|
|
|
if (full)
|
|
|
|
|
full--;
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
bits &= ~mask;
|
|
|
|
|
mask = 0;
|
|
|
|
|
}
|
2005-04-27 12:09:21 +00:00
|
|
|
|
}
|
2021-06-22 11:11:46 +02:00
|
|
|
|
if ((bits & 0x80)) /* ASN.1 bit 0. */
|
|
|
|
|
aodf->pinflags.case_sensitive = 1;
|
|
|
|
|
if ((bits & 0x40)) /* ASN.1 bit 1. */
|
|
|
|
|
aodf->pinflags.local = 1;
|
|
|
|
|
if ((bits & 0x20))
|
|
|
|
|
aodf->pinflags.change_disabled = 1;
|
|
|
|
|
if ((bits & 0x10))
|
|
|
|
|
aodf->pinflags.unblock_disabled = 1;
|
|
|
|
|
if ((bits & 0x08))
|
|
|
|
|
aodf->pinflags.initialized = 1;
|
|
|
|
|
if ((bits & 0x04))
|
|
|
|
|
aodf->pinflags.needs_padding = 1;
|
|
|
|
|
if ((bits & 0x02))
|
|
|
|
|
aodf->pinflags.unblocking_pin = 1;
|
|
|
|
|
if ((bits & 0x01))
|
|
|
|
|
aodf->pinflags.so_pin = 1;
|
|
|
|
|
/* The second octet. */
|
|
|
|
|
bits = 0;
|
|
|
|
|
if (objlen)
|
|
|
|
|
{
|
|
|
|
|
bits = *pp++; nn--; objlen--;
|
|
|
|
|
if (full)
|
|
|
|
|
full--;
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
bits &= ~mask;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
if ((bits & 0x80))
|
|
|
|
|
aodf->pinflags.disable_allowed = 1;
|
|
|
|
|
if ((bits & 0x40))
|
|
|
|
|
aodf->pinflags.integrity_protected = 1;
|
|
|
|
|
if ((bits & 0x20))
|
|
|
|
|
aodf->pinflags.confidentiality_protected = 1;
|
|
|
|
|
if ((bits & 0x10))
|
|
|
|
|
aodf->pinflags.exchange_ref_data = 1;
|
|
|
|
|
/* Skip remaining bits. */
|
|
|
|
|
pp += objlen;
|
|
|
|
|
nn -= objlen;
|
2005-04-27 12:09:21 +00:00
|
|
|
|
}
|
2011-02-04 12:57:53 +01:00
|
|
|
|
|
2021-06-22 11:11:46 +02:00
|
|
|
|
/* PinType */
|
2005-04-27 12:09:21 +00:00
|
|
|
|
where = __LINE__;
|
|
|
|
|
err = parse_ber_header (&pp, &nn, &class, &tag, &constructed,
|
|
|
|
|
&ndef, &objlen, &hdrlen);
|
2021-06-22 11:11:46 +02:00
|
|
|
|
if (!err && (objlen > nn
|
|
|
|
|
|| class != CLASS_UNIVERSAL || tag != TAG_ENUMERATED))
|
2005-04-27 12:09:21 +00:00
|
|
|
|
err = gpg_error (GPG_ERR_INV_OBJ);
|
2021-06-22 11:11:46 +02:00
|
|
|
|
if (!err && objlen > sizeof (ul))
|
|
|
|
|
err = gpg_error (GPG_ERR_UNSUPPORTED_ENCODING);
|
2005-04-27 12:09:21 +00:00
|
|
|
|
if (err)
|
|
|
|
|
goto parse_error;
|
|
|
|
|
|
|
|
|
|
for (ul=0; objlen; objlen--)
|
|
|
|
|
{
|
|
|
|
|
ul <<= 8;
|
2011-02-04 12:57:53 +01:00
|
|
|
|
ul |= (*pp++) & 0xff;
|
2005-04-27 12:09:21 +00:00
|
|
|
|
nn--;
|
|
|
|
|
}
|
2021-06-22 11:11:46 +02:00
|
|
|
|
aodf->pintype = ul;
|
2011-02-04 12:57:53 +01:00
|
|
|
|
|
2021-06-22 11:11:46 +02:00
|
|
|
|
/* minLength */
|
2005-04-27 12:09:21 +00:00
|
|
|
|
where = __LINE__;
|
|
|
|
|
err = parse_ber_header (&pp, &nn, &class, &tag, &constructed,
|
|
|
|
|
&ndef, &objlen, &hdrlen);
|
2021-06-22 11:11:46 +02:00
|
|
|
|
if (!err && (objlen > nn
|
|
|
|
|
|| class != CLASS_UNIVERSAL || tag != TAG_INTEGER))
|
2005-04-27 12:09:21 +00:00
|
|
|
|
err = gpg_error (GPG_ERR_INV_OBJ);
|
2021-06-22 11:11:46 +02:00
|
|
|
|
if (!err && objlen > sizeof (ul))
|
|
|
|
|
err = gpg_error (GPG_ERR_UNSUPPORTED_ENCODING);
|
2005-04-27 12:09:21 +00:00
|
|
|
|
if (err)
|
|
|
|
|
goto parse_error;
|
2021-06-22 11:11:46 +02:00
|
|
|
|
for (ul=0; objlen; objlen--)
|
2005-04-27 12:09:21 +00:00
|
|
|
|
{
|
2021-06-22 11:11:46 +02:00
|
|
|
|
ul <<= 8;
|
|
|
|
|
ul |= (*pp++) & 0xff;
|
|
|
|
|
nn--;
|
2005-04-27 12:09:21 +00:00
|
|
|
|
}
|
2021-06-22 11:11:46 +02:00
|
|
|
|
aodf->min_length = ul;
|
2011-02-04 12:57:53 +01:00
|
|
|
|
|
2021-06-22 11:11:46 +02:00
|
|
|
|
/* storedLength */
|
2005-04-27 12:09:21 +00:00
|
|
|
|
where = __LINE__;
|
|
|
|
|
err = parse_ber_header (&pp, &nn, &class, &tag, &constructed,
|
|
|
|
|
&ndef, &objlen, &hdrlen);
|
2021-06-22 11:11:46 +02:00
|
|
|
|
if (!err && (objlen > nn
|
|
|
|
|
|| class != CLASS_UNIVERSAL || tag != TAG_INTEGER))
|
2005-04-27 12:09:21 +00:00
|
|
|
|
err = gpg_error (GPG_ERR_INV_OBJ);
|
2021-06-22 11:11:46 +02:00
|
|
|
|
if (!err && objlen > sizeof (ul))
|
|
|
|
|
err = gpg_error (GPG_ERR_UNSUPPORTED_ENCODING);
|
2005-04-27 12:09:21 +00:00
|
|
|
|
if (err)
|
|
|
|
|
goto parse_error;
|
2021-06-22 11:11:46 +02:00
|
|
|
|
for (ul=0; objlen; objlen--)
|
|
|
|
|
{
|
|
|
|
|
ul <<= 8;
|
|
|
|
|
ul |= (*pp++) & 0xff;
|
|
|
|
|
nn--;
|
|
|
|
|
}
|
|
|
|
|
aodf->stored_length = ul;
|
2011-02-04 12:57:53 +01:00
|
|
|
|
|
2021-06-22 11:11:46 +02:00
|
|
|
|
/* optional maxLength */
|
2005-04-27 12:09:21 +00:00
|
|
|
|
where = __LINE__;
|
|
|
|
|
err = parse_ber_header (&pp, &nn, &class, &tag, &constructed,
|
|
|
|
|
&ndef, &objlen, &hdrlen);
|
|
|
|
|
if (gpg_err_code (err) == GPG_ERR_EOF)
|
|
|
|
|
goto ready;
|
|
|
|
|
if (!err && objlen > nn)
|
|
|
|
|
err = gpg_error (GPG_ERR_INV_OBJ);
|
|
|
|
|
if (err)
|
|
|
|
|
goto parse_error;
|
2021-06-22 11:11:46 +02:00
|
|
|
|
if (class == CLASS_UNIVERSAL && tag == TAG_INTEGER)
|
|
|
|
|
{
|
|
|
|
|
if (objlen > sizeof (ul))
|
|
|
|
|
{
|
|
|
|
|
err = gpg_error (GPG_ERR_UNSUPPORTED_ENCODING);
|
|
|
|
|
goto parse_error;
|
|
|
|
|
}
|
|
|
|
|
for (ul=0; objlen; objlen--)
|
|
|
|
|
{
|
|
|
|
|
ul <<= 8;
|
|
|
|
|
ul |= (*pp++) & 0xff;
|
|
|
|
|
nn--;
|
|
|
|
|
}
|
|
|
|
|
aodf->max_length = ul;
|
|
|
|
|
aodf->max_length_valid = 1;
|
2005-04-27 12:09:21 +00:00
|
|
|
|
|
2021-06-22 11:11:46 +02:00
|
|
|
|
where = __LINE__;
|
|
|
|
|
err = parse_ber_header (&pp, &nn, &class, &tag, &constructed,
|
|
|
|
|
&ndef, &objlen, &hdrlen);
|
|
|
|
|
if (gpg_err_code (err) == GPG_ERR_EOF)
|
|
|
|
|
goto ready;
|
|
|
|
|
if (!err && objlen > nn)
|
|
|
|
|
err = gpg_error (GPG_ERR_INV_OBJ);
|
|
|
|
|
if (err)
|
|
|
|
|
goto parse_error;
|
|
|
|
|
}
|
2011-02-04 12:57:53 +01:00
|
|
|
|
|
2021-06-22 11:11:46 +02:00
|
|
|
|
/* Optional pinReference. */
|
|
|
|
|
if (class == CLASS_CONTEXT && tag == 0)
|
|
|
|
|
{
|
|
|
|
|
if (objlen > sizeof (ul))
|
|
|
|
|
{
|
|
|
|
|
err = gpg_error (GPG_ERR_UNSUPPORTED_ENCODING);
|
|
|
|
|
goto parse_error;
|
|
|
|
|
}
|
|
|
|
|
for (ul=0; objlen; objlen--)
|
|
|
|
|
{
|
|
|
|
|
ul <<= 8;
|
|
|
|
|
ul |= (*pp++) & 0xff;
|
|
|
|
|
nn--;
|
|
|
|
|
}
|
|
|
|
|
aodf->pin_reference = ul;
|
|
|
|
|
aodf->pin_reference_valid = 1;
|
2005-04-27 12:09:21 +00:00
|
|
|
|
|
2021-06-22 11:11:46 +02:00
|
|
|
|
where = __LINE__;
|
|
|
|
|
err = parse_ber_header (&pp, &nn, &class, &tag, &constructed,
|
|
|
|
|
&ndef, &objlen, &hdrlen);
|
|
|
|
|
if (gpg_err_code (err) == GPG_ERR_EOF)
|
|
|
|
|
goto ready;
|
|
|
|
|
if (!err && objlen > nn)
|
|
|
|
|
err = gpg_error (GPG_ERR_INV_OBJ);
|
|
|
|
|
if (err)
|
|
|
|
|
goto parse_error;
|
|
|
|
|
}
|
2005-04-27 12:09:21 +00:00
|
|
|
|
|
2021-06-22 11:11:46 +02:00
|
|
|
|
/* Optional padChar. */
|
|
|
|
|
if (class == CLASS_UNIVERSAL && tag == TAG_OCTET_STRING)
|
2005-04-27 12:09:21 +00:00
|
|
|
|
{
|
2021-06-22 11:11:46 +02:00
|
|
|
|
if (objlen != 1)
|
|
|
|
|
{
|
|
|
|
|
errstr = "padChar is not of size(1)";
|
|
|
|
|
goto parse_error;
|
|
|
|
|
}
|
|
|
|
|
aodf->pad_char = *pp++; nn--;
|
|
|
|
|
aodf->pad_char_valid = 1;
|
2005-04-27 12:09:21 +00:00
|
|
|
|
|
2021-06-22 11:11:46 +02:00
|
|
|
|
where = __LINE__;
|
|
|
|
|
err = parse_ber_header (&pp, &nn, &class, &tag, &constructed,
|
|
|
|
|
&ndef, &objlen, &hdrlen);
|
|
|
|
|
if (gpg_err_code (err) == GPG_ERR_EOF)
|
|
|
|
|
goto ready;
|
|
|
|
|
if (!err && objlen > nn)
|
|
|
|
|
err = gpg_error (GPG_ERR_INV_OBJ);
|
|
|
|
|
if (err)
|
|
|
|
|
goto parse_error;
|
|
|
|
|
}
|
2011-02-04 12:57:53 +01:00
|
|
|
|
|
2021-06-22 11:11:46 +02:00
|
|
|
|
/* Skip optional lastPinChange. */
|
|
|
|
|
if (class == CLASS_UNIVERSAL && tag == TAG_GENERALIZED_TIME)
|
2005-04-27 12:09:21 +00:00
|
|
|
|
{
|
2021-06-22 11:11:46 +02:00
|
|
|
|
pp += objlen;
|
|
|
|
|
nn -= objlen;
|
|
|
|
|
|
2005-04-27 12:09:21 +00:00
|
|
|
|
where = __LINE__;
|
2021-06-22 11:11:46 +02:00
|
|
|
|
err = parse_ber_header (&pp, &nn, &class, &tag, &constructed,
|
2005-04-27 12:09:21 +00:00
|
|
|
|
&ndef, &objlen, &hdrlen);
|
2021-06-22 11:11:46 +02:00
|
|
|
|
if (gpg_err_code (err) == GPG_ERR_EOF)
|
|
|
|
|
goto ready;
|
|
|
|
|
if (!err && objlen > nn)
|
2005-04-27 12:09:21 +00:00
|
|
|
|
err = gpg_error (GPG_ERR_INV_OBJ);
|
|
|
|
|
if (err)
|
|
|
|
|
goto parse_error;
|
2021-06-22 11:11:46 +02:00
|
|
|
|
}
|
2011-02-04 12:57:53 +01:00
|
|
|
|
|
2021-06-22 11:11:46 +02:00
|
|
|
|
/* Optional Path object. */
|
|
|
|
|
if (class == CLASS_UNIVERSAL || tag == TAG_SEQUENCE)
|
|
|
|
|
{
|
|
|
|
|
const unsigned char *ppp = pp;
|
|
|
|
|
size_t nnn = objlen;
|
|
|
|
|
|
|
|
|
|
pp += objlen;
|
|
|
|
|
nn -= objlen;
|
2011-02-04 12:57:53 +01:00
|
|
|
|
|
2005-04-27 12:09:21 +00:00
|
|
|
|
where = __LINE__;
|
|
|
|
|
err = parse_ber_header (&ppp, &nnn, &class, &tag, &constructed,
|
|
|
|
|
&ndef, &objlen, &hdrlen);
|
2021-06-22 11:11:46 +02:00
|
|
|
|
if (!err && objlen > nnn)
|
2005-04-27 12:09:21 +00:00
|
|
|
|
err = gpg_error (GPG_ERR_INV_OBJ);
|
|
|
|
|
if (err)
|
|
|
|
|
goto parse_error;
|
2011-02-04 12:57:53 +01:00
|
|
|
|
|
2021-06-22 11:11:46 +02:00
|
|
|
|
/* Make sure that the next element has a path of even
|
|
|
|
|
* length (FIDs are two bytes each). */
|
|
|
|
|
if (class != CLASS_UNIVERSAL || tag != TAG_OCTET_STRING
|
|
|
|
|
|| (objlen & 1) )
|
2005-04-27 12:09:21 +00:00
|
|
|
|
{
|
2021-06-22 11:11:46 +02:00
|
|
|
|
errstr = "invalid path reference";
|
|
|
|
|
goto parse_error;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
aodf->pathlen = objlen/2;
|
|
|
|
|
aodf->path = xtrycalloc (aodf->pathlen, sizeof *aodf->path);
|
|
|
|
|
if (!aodf->path)
|
|
|
|
|
goto no_core;
|
|
|
|
|
for (i=0; i < aodf->pathlen; i++, ppp += 2, nnn -= 2)
|
|
|
|
|
aodf->path[i] = ((ppp[0] << 8) | ppp[1]);
|
|
|
|
|
|
|
|
|
|
if (nnn)
|
|
|
|
|
{
|
|
|
|
|
/* An index and length follows. */
|
|
|
|
|
aodf->have_off = 1;
|
|
|
|
|
where = __LINE__;
|
|
|
|
|
err = parse_ber_header (&ppp, &nnn, &class, &tag,
|
|
|
|
|
&constructed,
|
|
|
|
|
&ndef, &objlen, &hdrlen);
|
|
|
|
|
if (!err && (objlen > nnn
|
|
|
|
|
|| class != CLASS_UNIVERSAL
|
|
|
|
|
|| tag != TAG_INTEGER))
|
|
|
|
|
err = gpg_error (GPG_ERR_INV_OBJ);
|
|
|
|
|
if (err)
|
|
|
|
|
goto parse_error;
|
|
|
|
|
|
|
|
|
|
for (ul=0; objlen; objlen--)
|
|
|
|
|
{
|
|
|
|
|
ul <<= 8;
|
|
|
|
|
ul |= (*ppp++) & 0xff;
|
|
|
|
|
nnn--;
|
|
|
|
|
}
|
|
|
|
|
aodf->off = ul;
|
|
|
|
|
|
|
|
|
|
where = __LINE__;
|
|
|
|
|
err = parse_ber_header (&ppp, &nnn, &class, &tag,
|
|
|
|
|
&constructed,
|
|
|
|
|
&ndef, &objlen, &hdrlen);
|
|
|
|
|
if (!err && (objlen > nnn
|
|
|
|
|
|| class != CLASS_CONTEXT || tag != 0))
|
|
|
|
|
err = gpg_error (GPG_ERR_INV_OBJ);
|
|
|
|
|
if (err)
|
|
|
|
|
goto parse_error;
|
|
|
|
|
|
|
|
|
|
for (ul=0; objlen; objlen--)
|
|
|
|
|
{
|
|
|
|
|
ul <<= 8;
|
|
|
|
|
ul |= (*ppp++) & 0xff;
|
|
|
|
|
nnn--;
|
|
|
|
|
}
|
|
|
|
|
aodf->len = ul;
|
2005-04-27 12:09:21 +00:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
2021-06-22 11:11:46 +02:00
|
|
|
|
else if (auth_type == AUTH_TYPE_AUTHKEY)
|
|
|
|
|
{
|
|
|
|
|
|
|
|
|
|
}
|
2011-02-04 12:57:53 +01:00
|
|
|
|
|
2021-04-29 15:26:36 +02:00
|
|
|
|
/* Ignore further objects which might be there due to future
|
2005-04-27 12:09:21 +00:00
|
|
|
|
extensions of pkcs#15. */
|
|
|
|
|
|
|
|
|
|
ready:
|
2021-04-29 15:26:36 +02:00
|
|
|
|
if (gpg_err_code (err) == GPG_ERR_EOF)
|
|
|
|
|
err = 0;
|
2020-03-31 10:40:21 +02:00
|
|
|
|
if (opt.verbose)
|
|
|
|
|
{
|
2021-04-29 15:26:36 +02:00
|
|
|
|
log_info ("p15: AODF %04hX: id=", fid);
|
2020-03-31 10:40:21 +02:00
|
|
|
|
for (i=0; i < aodf->objidlen; i++)
|
|
|
|
|
log_printf ("%02X", aodf->objid[i]);
|
2021-04-29 15:26:36 +02:00
|
|
|
|
if (aodf->label)
|
|
|
|
|
log_printf (" (%s)", aodf->label);
|
|
|
|
|
log_info ("p15: ");
|
2021-06-22 11:11:46 +02:00
|
|
|
|
log_printf (" %s",
|
|
|
|
|
aodf->auth_type == AUTH_TYPE_PIN? "pin" :
|
|
|
|
|
aodf->auth_type == AUTH_TYPE_AUTHKEY? "authkey" : "?");
|
2020-03-31 10:40:21 +02:00
|
|
|
|
if (aodf->pathlen)
|
|
|
|
|
{
|
|
|
|
|
log_printf (" path=");
|
|
|
|
|
for (i=0; i < aodf->pathlen; i++)
|
|
|
|
|
log_printf ("%s%04hX", i?"/":"",aodf->path[i]);
|
|
|
|
|
if (aodf->have_off)
|
|
|
|
|
log_printf ("[%lu/%lu]", aodf->off, aodf->len);
|
|
|
|
|
}
|
2021-04-29 15:26:36 +02:00
|
|
|
|
if (aodf->authid)
|
|
|
|
|
{
|
|
|
|
|
log_printf (" authid=");
|
|
|
|
|
for (i=0; i < aodf->authidlen; i++)
|
|
|
|
|
log_printf ("%02X", aodf->authid[i]);
|
|
|
|
|
}
|
2021-06-22 11:11:46 +02:00
|
|
|
|
if (aodf->auth_type == AUTH_TYPE_PIN)
|
|
|
|
|
{
|
|
|
|
|
if (aodf->pin_reference_valid)
|
|
|
|
|
log_printf (" pinref=0x%02lX", aodf->pin_reference);
|
|
|
|
|
log_printf (" min=%lu", aodf->min_length);
|
|
|
|
|
log_printf (" stored=%lu", aodf->stored_length);
|
|
|
|
|
if (aodf->max_length_valid)
|
|
|
|
|
log_printf (" max=%lu", aodf->max_length);
|
|
|
|
|
if (aodf->pad_char_valid)
|
|
|
|
|
log_printf (" pad=0x%02x", aodf->pad_char);
|
|
|
|
|
|
|
|
|
|
log_info ("p15: flags=");
|
|
|
|
|
s = "";
|
|
|
|
|
if (aodf->pinflags.case_sensitive)
|
|
|
|
|
log_printf ("%scase_sensitive", s), s = ",";
|
|
|
|
|
if (aodf->pinflags.local)
|
|
|
|
|
log_printf ("%slocal", s), s = ",";
|
|
|
|
|
if (aodf->pinflags.change_disabled)
|
|
|
|
|
log_printf ("%schange_disabled", s), s = ",";
|
|
|
|
|
if (aodf->pinflags.unblock_disabled)
|
|
|
|
|
log_printf ("%sunblock_disabled", s), s = ",";
|
|
|
|
|
if (aodf->pinflags.initialized)
|
|
|
|
|
log_printf ("%sinitialized", s), s = ",";
|
|
|
|
|
if (aodf->pinflags.needs_padding)
|
|
|
|
|
log_printf ("%sneeds_padding", s), s = ",";
|
|
|
|
|
if (aodf->pinflags.unblocking_pin)
|
|
|
|
|
log_printf ("%sunblocking_pin", s), s = ",";
|
|
|
|
|
if (aodf->pinflags.so_pin)
|
|
|
|
|
log_printf ("%sso_pin", s), s = ",";
|
|
|
|
|
if (aodf->pinflags.disable_allowed)
|
|
|
|
|
log_printf ("%sdisable_allowed", s), s = ",";
|
|
|
|
|
if (aodf->pinflags.integrity_protected)
|
|
|
|
|
log_printf ("%sintegrity_protected", s), s = ",";
|
|
|
|
|
if (aodf->pinflags.confidentiality_protected)
|
|
|
|
|
log_printf ("%sconfidentiality_protected", s), s = ",";
|
|
|
|
|
if (aodf->pinflags.exchange_ref_data)
|
|
|
|
|
log_printf ("%sexchange_ref_data", s), s = ",";
|
2020-03-31 10:40:21 +02:00
|
|
|
|
{
|
2021-06-22 11:11:46 +02:00
|
|
|
|
char numbuf[50];
|
|
|
|
|
const char *s2;
|
|
|
|
|
|
|
|
|
|
switch (aodf->pintype)
|
|
|
|
|
{
|
|
|
|
|
case PIN_TYPE_BCD: s2 = "bcd"; break;
|
|
|
|
|
case PIN_TYPE_ASCII_NUMERIC: s2 = "ascii-numeric"; break;
|
|
|
|
|
case PIN_TYPE_UTF8: s2 = "utf8"; break;
|
|
|
|
|
case PIN_TYPE_HALF_NIBBLE_BCD: s2 = "half-nibble-bcd"; break;
|
|
|
|
|
case PIN_TYPE_ISO9564_1: s2 = "iso9564-1"; break;
|
|
|
|
|
default:
|
|
|
|
|
sprintf (numbuf, "%lu", (unsigned long)aodf->pintype);
|
|
|
|
|
s2 = numbuf;
|
|
|
|
|
}
|
|
|
|
|
log_printf ("%stype=%s", s, s2); s = ",";
|
2020-03-31 10:40:21 +02:00
|
|
|
|
}
|
2021-06-22 11:11:46 +02:00
|
|
|
|
}
|
|
|
|
|
else if (aodf->auth_type == AUTH_TYPE_AUTHKEY)
|
|
|
|
|
{
|
|
|
|
|
}
|
2020-03-31 10:40:21 +02:00
|
|
|
|
log_printf ("\n");
|
2005-04-27 12:09:21 +00:00
|
|
|
|
}
|
2011-02-04 12:57:53 +01:00
|
|
|
|
|
2005-04-27 12:09:21 +00:00
|
|
|
|
/* Put it into the list. */
|
|
|
|
|
aodf->next = aodflist;
|
|
|
|
|
aodflist = aodf;
|
|
|
|
|
aodf = NULL;
|
2021-04-29 15:26:36 +02:00
|
|
|
|
goto next_record; /* Ready with this record. */
|
2005-04-27 12:09:21 +00:00
|
|
|
|
|
|
|
|
|
no_core:
|
2006-09-14 16:50:33 +00:00
|
|
|
|
err = gpg_error_from_syserror ();
|
2005-04-27 12:09:21 +00:00
|
|
|
|
release_aodf_object (aodf);
|
|
|
|
|
goto leave;
|
|
|
|
|
|
|
|
|
|
parse_error:
|
2021-04-29 15:26:36 +02:00
|
|
|
|
log_error ("p15: error parsing AODF record at %d: %s - skipped\n",
|
2005-04-27 12:09:21 +00:00
|
|
|
|
where, errstr? errstr : gpg_strerror (err));
|
|
|
|
|
err = 0;
|
|
|
|
|
release_aodf_object (aodf);
|
2021-04-29 15:26:36 +02:00
|
|
|
|
|
|
|
|
|
next_record:
|
|
|
|
|
/* If the card uses a record oriented file structure, read the
|
|
|
|
|
* next record. Otherwise we keep on parsing the current buffer. */
|
|
|
|
|
recno++;
|
|
|
|
|
if (record_mode)
|
|
|
|
|
{
|
|
|
|
|
xfree (buffer); buffer = NULL;
|
|
|
|
|
err = select_and_read_record (app, 0, recno, "AODF",
|
|
|
|
|
&buffer, &buflen, NULL);
|
|
|
|
|
if (err) {
|
|
|
|
|
if (gpg_err_code (err) == GPG_ERR_NOT_FOUND)
|
|
|
|
|
err = 0;
|
|
|
|
|
goto leave;
|
|
|
|
|
}
|
|
|
|
|
p = buffer;
|
|
|
|
|
n = buflen;
|
|
|
|
|
}
|
2005-04-27 12:09:21 +00:00
|
|
|
|
} /* End looping over all records. */
|
|
|
|
|
|
|
|
|
|
leave:
|
|
|
|
|
xfree (buffer);
|
|
|
|
|
if (err)
|
|
|
|
|
release_aodflist (aodflist);
|
|
|
|
|
else
|
|
|
|
|
*result = aodflist;
|
|
|
|
|
return err;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2020-03-30 21:18:12 +02:00
|
|
|
|
/* Print the BIT STRING with the tokenflags from the TokenInfo. */
|
|
|
|
|
static void
|
|
|
|
|
print_tokeninfo_tokenflags (const unsigned char *der, size_t derlen)
|
|
|
|
|
{
|
|
|
|
|
unsigned int bits, mask;
|
|
|
|
|
int i, unused, full;
|
|
|
|
|
int other = 0;
|
|
|
|
|
|
|
|
|
|
if (!derlen)
|
|
|
|
|
{
|
|
|
|
|
log_printf (" [invalid object]");
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
unused = *der++; derlen--;
|
|
|
|
|
if ((!derlen && unused) || unused/8 > derlen)
|
|
|
|
|
{
|
|
|
|
|
log_printf (" [wrong encoding]");
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
full = derlen - (unused+7)/8;
|
|
|
|
|
unused %= 8;
|
|
|
|
|
mask = 0;
|
|
|
|
|
for (i=1; unused; i <<= 1, unused--)
|
|
|
|
|
mask |= i;
|
|
|
|
|
|
|
|
|
|
/* First octet */
|
|
|
|
|
if (derlen)
|
|
|
|
|
{
|
|
|
|
|
bits = *der++; derlen--;
|
|
|
|
|
if (full)
|
|
|
|
|
full--;
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
bits &= ~mask;
|
|
|
|
|
mask = 0;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
bits = 0;
|
|
|
|
|
if ((bits & 0x80)) log_printf (" readonly");
|
|
|
|
|
if ((bits & 0x40)) log_printf (" loginRequired");
|
|
|
|
|
if ((bits & 0x20)) log_printf (" prnGeneration");
|
|
|
|
|
if ((bits & 0x10)) log_printf (" eidCompliant");
|
|
|
|
|
if ((bits & 0x08)) other = 1;
|
|
|
|
|
if ((bits & 0x04)) other = 1;
|
|
|
|
|
if ((bits & 0x02)) other = 1;
|
|
|
|
|
if ((bits & 0x01)) other = 1;
|
|
|
|
|
|
|
|
|
|
/* Next octet. */
|
|
|
|
|
if (derlen)
|
|
|
|
|
other = 1;
|
|
|
|
|
|
|
|
|
|
if (other)
|
|
|
|
|
log_printf (" [unknown]");
|
|
|
|
|
}
|
2005-04-27 12:09:21 +00:00
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
2011-02-04 12:57:53 +01:00
|
|
|
|
/* Read and parse the EF(TokenInfo).
|
2021-04-29 15:26:36 +02:00
|
|
|
|
*
|
|
|
|
|
* TokenInfo ::= SEQUENCE {
|
|
|
|
|
* version INTEGER {v1(0)} (v1,...),
|
|
|
|
|
* serialNumber OCTET STRING,
|
|
|
|
|
* manufacturerID Label OPTIONAL,
|
|
|
|
|
* label [0] Label OPTIONAL,
|
|
|
|
|
* tokenflags TokenFlags,
|
|
|
|
|
* seInfo SEQUENCE OF SecurityEnvironmentInfo OPTIONAL,
|
|
|
|
|
* recordInfo [1] RecordInfo OPTIONAL,
|
|
|
|
|
* supportedAlgorithms [2] SEQUENCE OF AlgorithmInfo OPTIONAL,
|
|
|
|
|
* ...,
|
|
|
|
|
* issuerId [3] Label OPTIONAL,
|
|
|
|
|
* holderId [4] Label OPTIONAL,
|
|
|
|
|
* lastUpdate [5] LastUpdate OPTIONAL,
|
|
|
|
|
* preferredLanguage PrintableString OPTIONAL -- In accordance with
|
|
|
|
|
* -- IETF RFC 1766
|
|
|
|
|
* } (CONSTRAINED BY { -- Each AlgorithmInfo.reference value must be unique --})
|
|
|
|
|
*
|
|
|
|
|
* TokenFlags ::= BIT STRING {
|
|
|
|
|
* readOnly (0),
|
|
|
|
|
* loginRequired (1),
|
|
|
|
|
* prnGeneration (2),
|
|
|
|
|
* eidCompliant (3)
|
|
|
|
|
* }
|
|
|
|
|
*
|
|
|
|
|
*
|
|
|
|
|
* Sample EF 5032:
|
|
|
|
|
* 30 31 02 01 00 04 04 05 45 36 9F 0C 0C 44 2D 54 01......E6...D-T
|
|
|
|
|
* 72 75 73 74 20 47 6D 62 48 80 14 4F 66 66 69 63 rust GmbH..Offic
|
|
|
|
|
* 65 20 69 64 65 6E 74 69 74 79 20 63 61 72 64 03 e identity card.
|
|
|
|
|
* 02 00 40 20 63 61 72 64 03 02 00 40 00 00 00 00 ..@ card...@....
|
|
|
|
|
* 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 ................
|
|
|
|
|
*
|
|
|
|
|
* 0 49: SEQUENCE {
|
|
|
|
|
* 2 1: INTEGER 0
|
|
|
|
|
* 5 4: OCTET STRING 05 45 36 9F
|
|
|
|
|
* 11 12: UTF8String 'D-Trust GmbH'
|
|
|
|
|
* 25 20: [0] 'Office identity card'
|
|
|
|
|
* 47 2: BIT STRING
|
|
|
|
|
* : '00000010'B (bit 1)
|
|
|
|
|
* : Error: Spurious zero bits in bitstring.
|
|
|
|
|
* : }
|
2004-12-02 07:48:09 +00:00
|
|
|
|
*/
|
2005-09-05 14:36:36 +00:00
|
|
|
|
static gpg_error_t
|
|
|
|
|
read_ef_tokeninfo (app_t app)
|
|
|
|
|
{
|
|
|
|
|
gpg_error_t err;
|
|
|
|
|
unsigned char *buffer = NULL;
|
|
|
|
|
size_t buflen;
|
|
|
|
|
const unsigned char *p;
|
|
|
|
|
size_t n, objlen, hdrlen;
|
|
|
|
|
int class, tag, constructed, ndef;
|
|
|
|
|
unsigned long ul;
|
2011-02-04 12:57:53 +01:00
|
|
|
|
|
2021-04-29 15:26:36 +02:00
|
|
|
|
release_tokeninfo (app);
|
2020-04-07 16:00:11 +02:00
|
|
|
|
app->app_local->card_product = CARD_PRODUCT_UNKNOWN;
|
2020-04-03 11:21:32 +02:00
|
|
|
|
|
2021-04-29 15:26:36 +02:00
|
|
|
|
err = select_and_read_binary (app, 0x5032, "TokenInfo", &buffer, &buflen);
|
2005-09-05 14:36:36 +00:00
|
|
|
|
if (err)
|
|
|
|
|
return err;
|
2011-02-04 12:57:53 +01:00
|
|
|
|
|
2005-09-05 14:36:36 +00:00
|
|
|
|
p = buffer;
|
|
|
|
|
n = buflen;
|
|
|
|
|
|
|
|
|
|
err = parse_ber_header (&p, &n, &class, &tag, &constructed,
|
|
|
|
|
&ndef, &objlen, &hdrlen);
|
|
|
|
|
if (!err && (objlen > n || tag != TAG_SEQUENCE))
|
|
|
|
|
err = gpg_error (GPG_ERR_INV_OBJ);
|
|
|
|
|
if (err)
|
|
|
|
|
{
|
2020-03-31 10:40:21 +02:00
|
|
|
|
log_error ("p15: error parsing TokenInfo: %s\n", gpg_strerror (err));
|
2005-09-05 14:36:36 +00:00
|
|
|
|
goto leave;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
n = objlen;
|
|
|
|
|
|
|
|
|
|
/* Version. */
|
|
|
|
|
err = parse_ber_header (&p, &n, &class, &tag, &constructed,
|
|
|
|
|
&ndef, &objlen, &hdrlen);
|
|
|
|
|
if (!err && (objlen > n || tag != TAG_INTEGER))
|
|
|
|
|
err = gpg_error (GPG_ERR_INV_OBJ);
|
|
|
|
|
if (err)
|
|
|
|
|
goto leave;
|
|
|
|
|
|
|
|
|
|
for (ul=0; objlen; objlen--)
|
|
|
|
|
{
|
|
|
|
|
ul <<= 8;
|
2011-02-04 12:57:53 +01:00
|
|
|
|
ul |= (*p++) & 0xff;
|
2005-09-05 14:36:36 +00:00
|
|
|
|
n--;
|
|
|
|
|
}
|
|
|
|
|
if (ul)
|
|
|
|
|
{
|
2020-03-31 10:40:21 +02:00
|
|
|
|
log_error ("p15: invalid version %lu in TokenInfo\n", ul);
|
2005-09-05 14:36:36 +00:00
|
|
|
|
err = gpg_error (GPG_ERR_INV_OBJ);
|
|
|
|
|
goto leave;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* serialNumber. */
|
|
|
|
|
err = parse_ber_header (&p, &n, &class, &tag, &constructed,
|
|
|
|
|
&ndef, &objlen, &hdrlen);
|
|
|
|
|
if (!err && (objlen > n || tag != TAG_OCTET_STRING || !objlen))
|
|
|
|
|
err = gpg_error (GPG_ERR_INV_OBJ);
|
|
|
|
|
if (err)
|
|
|
|
|
goto leave;
|
2011-02-04 12:57:53 +01:00
|
|
|
|
|
2005-09-05 14:36:36 +00:00
|
|
|
|
xfree (app->app_local->serialno);
|
|
|
|
|
app->app_local->serialno = xtrymalloc (objlen);
|
|
|
|
|
if (!app->app_local->serialno)
|
|
|
|
|
{
|
2006-09-14 16:50:33 +00:00
|
|
|
|
err = gpg_error_from_syserror ();
|
2005-09-05 14:36:36 +00:00
|
|
|
|
goto leave;
|
|
|
|
|
}
|
|
|
|
|
memcpy (app->app_local->serialno, p, objlen);
|
|
|
|
|
app->app_local->serialnolen = objlen;
|
2020-03-30 21:18:12 +02:00
|
|
|
|
p += objlen;
|
|
|
|
|
n -= objlen;
|
|
|
|
|
|
|
|
|
|
/* Is there an optional manufacturerID? */
|
|
|
|
|
err = parse_ber_header (&p, &n, &class, &tag, &constructed,
|
|
|
|
|
&ndef, &objlen, &hdrlen);
|
|
|
|
|
if (!err && (objlen > n || !objlen))
|
|
|
|
|
err = gpg_error (GPG_ERR_INV_OBJ);
|
|
|
|
|
if (err)
|
|
|
|
|
goto leave;
|
|
|
|
|
if (class == CLASS_UNIVERSAL && tag == TAG_UTF8_STRING)
|
|
|
|
|
{
|
2020-04-03 11:21:32 +02:00
|
|
|
|
app->app_local->manufacturer_id = percent_data_escape (0, NULL,
|
|
|
|
|
p, objlen);
|
2020-03-30 21:18:12 +02:00
|
|
|
|
p += objlen;
|
|
|
|
|
n -= objlen;
|
|
|
|
|
/* Get next TLV. */
|
|
|
|
|
err = parse_ber_header (&p, &n, &class, &tag, &constructed,
|
|
|
|
|
&ndef, &objlen, &hdrlen);
|
|
|
|
|
if (!err && (objlen > n || !objlen))
|
|
|
|
|
err = gpg_error (GPG_ERR_INV_OBJ);
|
|
|
|
|
if (err)
|
|
|
|
|
goto leave;
|
|
|
|
|
}
|
|
|
|
|
if (class == CLASS_CONTEXT && tag == 0)
|
|
|
|
|
{
|
2021-04-29 15:26:36 +02:00
|
|
|
|
app->app_local->token_label = percent_data_escape (0, NULL, p, objlen);
|
2020-04-07 16:00:11 +02:00
|
|
|
|
|
2020-03-30 21:18:12 +02:00
|
|
|
|
p += objlen;
|
|
|
|
|
n -= objlen;
|
|
|
|
|
/* Get next TLV. */
|
|
|
|
|
err = parse_ber_header (&p, &n, &class, &tag, &constructed,
|
|
|
|
|
&ndef, &objlen, &hdrlen);
|
|
|
|
|
if (!err && (objlen > n || !objlen))
|
|
|
|
|
err = gpg_error (GPG_ERR_INV_OBJ);
|
|
|
|
|
if (err)
|
|
|
|
|
goto leave;
|
|
|
|
|
}
|
|
|
|
|
/* The next is the mandatory tokenflags object. */
|
|
|
|
|
if (class == CLASS_UNIVERSAL && tag == TAG_BIT_STRING)
|
|
|
|
|
{
|
2021-04-29 15:26:36 +02:00
|
|
|
|
app->app_local->tokenflagslen = objlen;
|
|
|
|
|
app->app_local->tokenflags = xtrymalloc (objlen);
|
|
|
|
|
if (!app->app_local->tokenflags)
|
2020-03-31 10:40:21 +02:00
|
|
|
|
{
|
2021-04-29 15:26:36 +02:00
|
|
|
|
err = gpg_error_from_syserror ();
|
|
|
|
|
goto leave;
|
2020-03-31 10:40:21 +02:00
|
|
|
|
}
|
2021-04-29 15:26:36 +02:00
|
|
|
|
memcpy (app->app_local->tokenflags, p, objlen);
|
2020-03-30 21:18:12 +02:00
|
|
|
|
p += objlen;
|
|
|
|
|
n -= objlen;
|
|
|
|
|
}
|
2005-09-05 14:36:36 +00:00
|
|
|
|
|
2021-04-29 15:26:36 +02:00
|
|
|
|
|
2005-09-05 14:36:36 +00:00
|
|
|
|
leave:
|
|
|
|
|
xfree (buffer);
|
|
|
|
|
return err;
|
|
|
|
|
}
|
2005-04-27 12:09:21 +00:00
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* Get all the basic information from the pkcs#15 card, check the
|
|
|
|
|
structure and initialize our local context. This is used once at
|
|
|
|
|
application initialization. */
|
2004-12-02 07:48:09 +00:00
|
|
|
|
static gpg_error_t
|
2005-04-27 12:09:21 +00:00
|
|
|
|
read_p15_info (app_t app)
|
2004-12-02 07:48:09 +00:00
|
|
|
|
{
|
2005-04-27 12:09:21 +00:00
|
|
|
|
gpg_error_t err;
|
2021-04-29 15:26:36 +02:00
|
|
|
|
prkdf_object_t prkdf;
|
|
|
|
|
unsigned int flag;
|
2005-04-27 12:09:21 +00:00
|
|
|
|
|
2021-04-29 15:26:36 +02:00
|
|
|
|
err = read_ef_tokeninfo (app);
|
|
|
|
|
if (err)
|
|
|
|
|
return err;
|
|
|
|
|
/* If we don't have a serial number yet but the TokenInfo provides
|
|
|
|
|
* one, use that. */
|
|
|
|
|
if (!APP_CARD(app)->serialno && app->app_local->serialno)
|
2005-09-05 14:36:36 +00:00
|
|
|
|
{
|
2021-04-29 15:26:36 +02:00
|
|
|
|
APP_CARD(app)->serialno = app->app_local->serialno;
|
|
|
|
|
APP_CARD(app)->serialnolen = app->app_local->serialnolen;
|
|
|
|
|
app->app_local->serialno = NULL;
|
|
|
|
|
app->app_local->serialnolen = 0;
|
|
|
|
|
err = app_munge_serialno (APP_CARD(app));
|
|
|
|
|
if (err)
|
|
|
|
|
return err;
|
2005-09-05 14:36:36 +00:00
|
|
|
|
}
|
2011-02-04 12:57:53 +01:00
|
|
|
|
|
2021-04-29 15:26:36 +02:00
|
|
|
|
release_lists (app);
|
|
|
|
|
|
2022-04-11 17:48:45 +02:00
|
|
|
|
if (IS_CARDOS_5 (app)
|
|
|
|
|
&& app->app_local->manufacturer_id
|
|
|
|
|
&& !ascii_strcasecmp (app->app_local->manufacturer_id, "GeNUA mbH")
|
|
|
|
|
&& !app->app_local->no_extended_mode)
|
|
|
|
|
{
|
2022-04-13 13:06:27 +02:00
|
|
|
|
if (!app->app_local->card_product)
|
|
|
|
|
app->app_local->card_product = CARD_PRODUCT_GENUA;
|
|
|
|
|
if (opt.verbose)
|
|
|
|
|
log_info ("p15: disabling extended mode based on TokenInfo\n");
|
2022-04-11 17:48:45 +02:00
|
|
|
|
app->app_local->no_extended_mode = 1;
|
|
|
|
|
}
|
|
|
|
|
|
2005-04-27 12:09:21 +00:00
|
|
|
|
/* Read the ODF so that we know the location of all directory
|
|
|
|
|
files. */
|
2005-09-05 14:36:36 +00:00
|
|
|
|
/* Fixme: We might need to get a non-standard ODF FID from TokenInfo. */
|
2005-04-27 12:09:21 +00:00
|
|
|
|
err = read_ef_odf (app, 0x5031);
|
|
|
|
|
if (err)
|
|
|
|
|
return err;
|
|
|
|
|
|
|
|
|
|
/* Read certificate information. */
|
2021-04-29 15:26:36 +02:00
|
|
|
|
log_assert (!app->app_local->certificate_info);
|
|
|
|
|
log_assert (!app->app_local->trusted_certificate_info);
|
|
|
|
|
log_assert (!app->app_local->useful_certificate_info);
|
|
|
|
|
err = read_ef_cdf (app, app->app_local->odf.certificates, 'c',
|
2005-04-27 12:09:21 +00:00
|
|
|
|
&app->app_local->certificate_info);
|
|
|
|
|
if (!err || gpg_err_code (err) == GPG_ERR_NO_DATA)
|
2021-04-29 15:26:36 +02:00
|
|
|
|
err = read_ef_cdf (app, app->app_local->odf.trusted_certificates, 't',
|
2005-04-27 12:09:21 +00:00
|
|
|
|
&app->app_local->trusted_certificate_info);
|
|
|
|
|
if (!err || gpg_err_code (err) == GPG_ERR_NO_DATA)
|
2021-04-29 15:26:36 +02:00
|
|
|
|
err = read_ef_cdf (app, app->app_local->odf.useful_certificates, 'u',
|
2005-04-27 12:09:21 +00:00
|
|
|
|
&app->app_local->useful_certificate_info);
|
|
|
|
|
if (gpg_err_code (err) == GPG_ERR_NO_DATA)
|
|
|
|
|
err = 0;
|
|
|
|
|
if (err)
|
|
|
|
|
return err;
|
2004-12-02 07:48:09 +00:00
|
|
|
|
|
2021-04-29 15:26:36 +02:00
|
|
|
|
/* Read information about public keys. */
|
|
|
|
|
log_assert (!app->app_local->public_key_info);
|
|
|
|
|
err = read_ef_pukdf (app, app->app_local->odf.public_keys,
|
|
|
|
|
&app->app_local->public_key_info);
|
|
|
|
|
if (!err || gpg_err_code (err) == GPG_ERR_NO_DATA)
|
|
|
|
|
err = read_ef_pukdf (app, app->app_local->odf.trusted_public_keys,
|
|
|
|
|
&app->app_local->public_key_info);
|
|
|
|
|
if (gpg_err_code (err) == GPG_ERR_NO_DATA)
|
|
|
|
|
err = 0;
|
|
|
|
|
if (err)
|
|
|
|
|
return err;
|
|
|
|
|
|
2005-04-27 12:09:21 +00:00
|
|
|
|
/* Read information about private keys. */
|
2021-04-29 15:26:36 +02:00
|
|
|
|
log_assert (!app->app_local->private_key_info);
|
2005-04-27 12:09:21 +00:00
|
|
|
|
err = read_ef_prkdf (app, app->app_local->odf.private_keys,
|
|
|
|
|
&app->app_local->private_key_info);
|
|
|
|
|
if (gpg_err_code (err) == GPG_ERR_NO_DATA)
|
|
|
|
|
err = 0;
|
|
|
|
|
if (err)
|
|
|
|
|
return err;
|
|
|
|
|
|
|
|
|
|
/* Read information about authentication objects. */
|
2021-04-29 15:26:36 +02:00
|
|
|
|
log_assert (!app->app_local->auth_object_info);
|
2005-04-27 12:09:21 +00:00
|
|
|
|
err = read_ef_aodf (app, app->app_local->odf.auth_objects,
|
|
|
|
|
&app->app_local->auth_object_info);
|
|
|
|
|
if (gpg_err_code (err) == GPG_ERR_NO_DATA)
|
|
|
|
|
err = 0;
|
|
|
|
|
|
|
|
|
|
|
2021-04-29 15:26:36 +02:00
|
|
|
|
/* See whether we can extend the private key information using
|
|
|
|
|
* information from certificates. We use only the first matching
|
|
|
|
|
* certificate; if we want to change this strategy we should walk
|
|
|
|
|
* over the certificates and then find the corresponsing private key
|
|
|
|
|
* objects. */
|
|
|
|
|
app->app_local->any_gpgusage = 0;
|
|
|
|
|
for (prkdf = app->app_local->private_key_info; prkdf; prkdf = prkdf->next)
|
|
|
|
|
{
|
|
|
|
|
cdf_object_t cdf;
|
|
|
|
|
char *extusage;
|
|
|
|
|
char *p, *pend;
|
|
|
|
|
int seen, i;
|
|
|
|
|
|
|
|
|
|
if (opt.debug)
|
|
|
|
|
log_printhex (prkdf->objid, prkdf->objidlen, "p15: prkdf id=");
|
|
|
|
|
if (cdf_object_from_objid (app, prkdf->objidlen, prkdf->objid, &cdf)
|
|
|
|
|
&& cdf_object_from_label (app, prkdf->label, &cdf))
|
|
|
|
|
continue; /* No matching certificate. */
|
|
|
|
|
if (!cdf->cert) /* Read and parse the certificate. */
|
|
|
|
|
readcert_by_cdf (app, cdf, NULL, NULL);
|
|
|
|
|
if (!cdf->cert)
|
|
|
|
|
continue; /* Unsupported or broken certificate. */
|
|
|
|
|
|
|
|
|
|
if (prkdf->is_ecc)
|
|
|
|
|
{
|
|
|
|
|
const char *oid;
|
|
|
|
|
const unsigned char *der;
|
|
|
|
|
size_t off, derlen, objlen, hdrlen;
|
|
|
|
|
int class, tag, constructed, ndef;
|
|
|
|
|
|
|
|
|
|
for (i=0; !(err = ksba_cert_get_extension
|
|
|
|
|
(cdf->cert, i, &oid, NULL, &off, &derlen)); i++)
|
|
|
|
|
if (!strcmp (oid, "1.3.6.1.4.1.11591.2.2.10") )
|
|
|
|
|
break;
|
|
|
|
|
if (!err && (der = ksba_cert_get_image (cdf->cert, NULL)))
|
|
|
|
|
{
|
|
|
|
|
der += off;
|
|
|
|
|
err = parse_ber_header (&der, &derlen, &class, &tag, &constructed,
|
|
|
|
|
&ndef, &objlen, &hdrlen);
|
|
|
|
|
if (!err && (objlen > derlen || tag != TAG_OCTET_STRING || ndef))
|
|
|
|
|
err = gpg_error (GPG_ERR_INV_OBJ);
|
|
|
|
|
if (!err)
|
|
|
|
|
{
|
|
|
|
|
derlen = objlen;
|
|
|
|
|
if (opt.debug)
|
|
|
|
|
log_printhex (der, derlen, "p15: OpenPGP KDF parms:");
|
|
|
|
|
/* Store them if they match the known OpenPGP format. */
|
|
|
|
|
if (derlen == 4 && der[0] == 3 && der[1] == 1)
|
|
|
|
|
memcpy (prkdf->ecdh_kdf, der, 4);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
err = 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (ksba_cert_get_ext_key_usages (cdf->cert, &extusage))
|
|
|
|
|
continue; /* No extended key usage attribute. */
|
|
|
|
|
|
|
|
|
|
if (opt.debug)
|
|
|
|
|
log_debug ("p15: ExtKeyUsages: %s\n", extusage);
|
|
|
|
|
p = extusage;
|
|
|
|
|
while (p && (pend=strchr (p, ':')))
|
|
|
|
|
{
|
|
|
|
|
*pend++ = 0;
|
|
|
|
|
if ( *pend == 'C' ) /* Look only at critical usages. */
|
|
|
|
|
{
|
|
|
|
|
prkdf->extusage.valid = 1;
|
|
|
|
|
seen = 1;
|
|
|
|
|
if (!strcmp (p, oid_kp_codeSigning)
|
|
|
|
|
|| !strcmp (p, oid_kp_timeStamping)
|
|
|
|
|
|| !strcmp (p, oid_kp_ocspSigning)
|
|
|
|
|
|| !strcmp (p, oid_kp_ms_documentSigning)
|
|
|
|
|
|| !strcmp (p, oid_kp_ms_old_documentSigning))
|
|
|
|
|
prkdf->extusage.sign = 1;
|
|
|
|
|
else if (!strcmp (p, oid_kp_emailProtection))
|
|
|
|
|
prkdf->extusage.encr = 1;
|
|
|
|
|
else if (!strcmp (p, oid_kp_serverAuth)
|
|
|
|
|
|| !strcmp (p, oid_kp_clientAuth)
|
|
|
|
|
|| !strcmp (p, oid_kp_ms_smartcardLogon))
|
|
|
|
|
prkdf->extusage.auth = 1;
|
|
|
|
|
else if (!strcmp (p, oid_kp_anyExtendedKeyUsage))
|
|
|
|
|
{
|
|
|
|
|
prkdf->extusage.sign = 1;
|
|
|
|
|
prkdf->extusage.encr = 1;
|
|
|
|
|
prkdf->extusage.auth = 1;
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
seen = 0;
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
seen = 0;
|
|
|
|
|
|
|
|
|
|
/* Now check the gpg Usage. Here we don't care about
|
|
|
|
|
* critical or non-critical here. */
|
|
|
|
|
if (seen)
|
|
|
|
|
; /* No more need to look for other caps. */
|
|
|
|
|
else if (!strcmp (p, oid_kp_gpgUsageCert))
|
|
|
|
|
{
|
|
|
|
|
prkdf->gpgusage.cert = 1;
|
|
|
|
|
prkdf->gpgusage.any = 1;
|
|
|
|
|
app->app_local->any_gpgusage = 1;
|
|
|
|
|
}
|
|
|
|
|
else if (!strcmp (p, oid_kp_gpgUsageSign))
|
|
|
|
|
{
|
|
|
|
|
prkdf->gpgusage.sign = 1;
|
|
|
|
|
prkdf->gpgusage.any = 1;
|
|
|
|
|
app->app_local->any_gpgusage = 1;
|
|
|
|
|
}
|
|
|
|
|
else if (!strcmp (p, oid_kp_gpgUsageEncr))
|
|
|
|
|
{
|
|
|
|
|
prkdf->gpgusage.encr = 1;
|
|
|
|
|
prkdf->gpgusage.any = 1;
|
|
|
|
|
app->app_local->any_gpgusage = 1;
|
|
|
|
|
}
|
|
|
|
|
else if (!strcmp (p, oid_kp_gpgUsageAuth))
|
|
|
|
|
{
|
|
|
|
|
prkdf->gpgusage.auth = 1;
|
|
|
|
|
prkdf->gpgusage.any = 1;
|
|
|
|
|
app->app_local->any_gpgusage = 1;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Skip to next item. */
|
|
|
|
|
if ((p = strchr (pend, '\n')))
|
|
|
|
|
p++;
|
|
|
|
|
}
|
|
|
|
|
xfree (extusage);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* See whether we can figure out something about the card. */
|
|
|
|
|
if (!app->app_local->card_product
|
|
|
|
|
&& app->app_local->manufacturer_id
|
|
|
|
|
&& !strcmp (app->app_local->manufacturer_id, "www.atos.net/cardos")
|
|
|
|
|
&& IS_CARDOS_5 (app))
|
|
|
|
|
{
|
|
|
|
|
/* This is a modern CARDOS card. */
|
|
|
|
|
flag = 0;
|
|
|
|
|
for (prkdf = app->app_local->private_key_info; prkdf; prkdf = prkdf->next)
|
|
|
|
|
{
|
|
|
|
|
if (prkdf->label && !strcmp (prkdf->label, "IdentityKey")
|
|
|
|
|
&& prkdf->key_reference_valid && prkdf->key_reference == 1
|
|
|
|
|
&& !prkdf->authid)
|
|
|
|
|
flag |= 1;
|
|
|
|
|
else if (prkdf->label && !strcmp (prkdf->label, "TransportKey")
|
|
|
|
|
&& prkdf->key_reference_valid && prkdf->key_reference==2
|
|
|
|
|
&& prkdf->authid)
|
|
|
|
|
flag |= 2;
|
|
|
|
|
}
|
|
|
|
|
if (flag == 3)
|
|
|
|
|
app->app_local->card_product = CARD_PRODUCT_RSCS;
|
|
|
|
|
|
|
|
|
|
}
|
|
|
|
|
if (!app->app_local->card_product
|
|
|
|
|
&& app->app_local->token_label
|
|
|
|
|
&& !strncmp (app->app_local->token_label, "D-TRUST Card V3", 15)
|
|
|
|
|
&& app->app_local->card_type == CARD_TYPE_CARDOS_50)
|
|
|
|
|
{
|
|
|
|
|
app->app_local->card_product = CARD_PRODUCT_DTRUST;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* Now print the info about the PrKDF. */
|
|
|
|
|
if (opt.verbose)
|
|
|
|
|
{
|
|
|
|
|
int i;
|
|
|
|
|
unsigned char *atr;
|
|
|
|
|
size_t atrlen;
|
|
|
|
|
const char *cardstr;
|
|
|
|
|
|
|
|
|
|
for (prkdf = app->app_local->private_key_info; prkdf; prkdf = prkdf->next)
|
|
|
|
|
{
|
|
|
|
|
log_info ("p15: PrKDF %04hX: id=", app->app_local->odf.private_keys);
|
|
|
|
|
for (i=0; i < prkdf->objidlen; i++)
|
|
|
|
|
log_printf ("%02X", prkdf->objid[i]);
|
|
|
|
|
if (prkdf->label)
|
|
|
|
|
log_printf (" (%s)", prkdf->label);
|
|
|
|
|
log_info ("p15: path=");
|
|
|
|
|
for (i=0; i < prkdf->pathlen; i++)
|
|
|
|
|
log_printf ("%s%04hX", i?"/":"",prkdf->path[i]);
|
|
|
|
|
if (prkdf->have_off)
|
|
|
|
|
log_printf ("[%lu/%lu]", prkdf->off, prkdf->len);
|
|
|
|
|
if (prkdf->authid)
|
|
|
|
|
{
|
|
|
|
|
log_printf (" authid=");
|
|
|
|
|
for (i=0; i < prkdf->authidlen; i++)
|
|
|
|
|
log_printf ("%02X", prkdf->authid[i]);
|
|
|
|
|
}
|
|
|
|
|
if (prkdf->key_reference_valid)
|
|
|
|
|
log_printf (" keyref=0x%02lX", prkdf->key_reference);
|
|
|
|
|
log_printf (" type=%s", prkdf->is_ecc? "ecc":"rsa");
|
|
|
|
|
if (prkdf->accessflags.any)
|
|
|
|
|
dump_keyaccess_flags (prkdf->accessflags);
|
|
|
|
|
dump_keyusage_flags (prkdf->usageflags);
|
|
|
|
|
if (prkdf->extusage.valid)
|
|
|
|
|
log_info ("p15: extusage=%s%s%s%s%s",
|
|
|
|
|
prkdf->extusage.sign? "sign":"",
|
|
|
|
|
(prkdf->extusage.sign
|
|
|
|
|
&& prkdf->extusage.encr)?",":"",
|
|
|
|
|
prkdf->extusage.encr? "encr":"",
|
|
|
|
|
((prkdf->extusage.sign || prkdf->extusage.encr)
|
|
|
|
|
&& prkdf->extusage.auth)?",":"",
|
|
|
|
|
prkdf->extusage.auth? "auth":"");
|
|
|
|
|
if (prkdf->gpgusage.any)
|
|
|
|
|
dump_gpgusage_flags (prkdf->gpgusage);
|
|
|
|
|
|
|
|
|
|
log_printf ("\n");
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
log_info ("p15: TokenInfo:\n");
|
|
|
|
|
if (app->app_local->serialno)
|
|
|
|
|
{
|
|
|
|
|
log_info ("p15: serialNumber .: ");
|
|
|
|
|
log_printhex (app->app_local->serialno, app->app_local->serialnolen,
|
|
|
|
|
"");
|
|
|
|
|
}
|
|
|
|
|
else if (APP_CARD(app)->serialno)
|
|
|
|
|
{
|
|
|
|
|
log_info ("p15: serialNumber .: ");
|
|
|
|
|
log_printhex (APP_CARD(app)->serialno, APP_CARD(app)->serialnolen,
|
|
|
|
|
"");
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (app->app_local->manufacturer_id)
|
|
|
|
|
log_info ("p15: manufacturerID: %s\n",
|
|
|
|
|
app->app_local->manufacturer_id);
|
|
|
|
|
if (app->app_local->card_product)
|
|
|
|
|
{
|
|
|
|
|
cardstr = cardproduct2str (app->app_local->card_product);
|
|
|
|
|
log_info ("p15: product ......: %d%s%s%s\n",
|
|
|
|
|
app->app_local->card_product,
|
|
|
|
|
*cardstr? " (":"", cardstr, *cardstr? ")":"");
|
|
|
|
|
}
|
|
|
|
|
if (app->app_local->token_label)
|
|
|
|
|
log_info ("p15: label ........: %s\n", app->app_local->token_label);
|
|
|
|
|
if (app->app_local->tokenflags)
|
|
|
|
|
{
|
|
|
|
|
log_info ("p15: tokenflags ...:");
|
|
|
|
|
print_tokeninfo_tokenflags (app->app_local->tokenflags,
|
|
|
|
|
app->app_local->tokenflagslen);
|
|
|
|
|
log_printf ("\n");
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
log_info ("p15: atr ..........: ");
|
|
|
|
|
atr = apdu_get_atr (app_get_slot (app), &atrlen);
|
|
|
|
|
if (!atr)
|
|
|
|
|
log_printf ("[error]\n");
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
log_printhex (atr, atrlen, "");
|
|
|
|
|
xfree (atr);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
cardstr = cardtype2str (app->app_local->card_type);
|
|
|
|
|
log_info ("p15: cardtype .....: %d%s%s%s\n",
|
|
|
|
|
app->app_local->card_type,
|
|
|
|
|
*cardstr? " (":"", cardstr, *cardstr? ")":"");
|
|
|
|
|
}
|
|
|
|
|
|
2005-04-27 12:09:21 +00:00
|
|
|
|
return err;
|
2004-12-02 07:48:09 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2005-04-27 12:09:21 +00:00
|
|
|
|
/* Helper to do_learn_status: Send information about all certificates
|
|
|
|
|
listed in CERTINFO back. Use CERTTYPE as type of the
|
|
|
|
|
certificate. */
|
2004-12-02 07:48:09 +00:00
|
|
|
|
static gpg_error_t
|
2005-04-27 12:09:21 +00:00
|
|
|
|
send_certinfo (app_t app, ctrl_t ctrl, const char *certtype,
|
|
|
|
|
cdf_object_t certinfo)
|
2004-12-02 07:48:09 +00:00
|
|
|
|
{
|
2005-04-27 12:09:21 +00:00
|
|
|
|
for (; certinfo; certinfo = certinfo->next)
|
|
|
|
|
{
|
|
|
|
|
char *buf, *p;
|
2021-04-29 15:26:36 +02:00
|
|
|
|
const char *label;
|
|
|
|
|
char *labelbuf;
|
2005-04-27 12:09:21 +00:00
|
|
|
|
|
|
|
|
|
buf = xtrymalloc (9 + certinfo->objidlen*2 + 1);
|
|
|
|
|
if (!buf)
|
2006-09-14 16:50:33 +00:00
|
|
|
|
return gpg_error_from_syserror ();
|
2005-04-27 12:09:21 +00:00
|
|
|
|
p = stpcpy (buf, "P15");
|
2021-04-29 15:26:36 +02:00
|
|
|
|
if (app->app_local->home_df != DEFAULT_HOME_DF)
|
2005-04-27 12:09:21 +00:00
|
|
|
|
{
|
2014-09-18 15:39:50 +02:00
|
|
|
|
snprintf (p, 6, "-%04X",
|
|
|
|
|
(unsigned int)(app->app_local->home_df & 0xffff));
|
2005-04-27 12:09:21 +00:00
|
|
|
|
p += 5;
|
|
|
|
|
}
|
|
|
|
|
p = stpcpy (p, ".");
|
2008-11-03 10:54:18 +00:00
|
|
|
|
bin2hex (certinfo->objid, certinfo->objidlen, p);
|
2005-04-27 12:09:21 +00:00
|
|
|
|
|
2021-04-29 15:26:36 +02:00
|
|
|
|
label = (certinfo->label && *certinfo->label)? certinfo->label : "-";
|
|
|
|
|
labelbuf = percent_data_escape (0, NULL, label, strlen (label));
|
|
|
|
|
if (!labelbuf)
|
|
|
|
|
{
|
|
|
|
|
xfree (buf);
|
|
|
|
|
return gpg_error_from_syserror ();
|
|
|
|
|
}
|
|
|
|
|
|
2005-04-27 12:09:21 +00:00
|
|
|
|
send_status_info (ctrl, "CERTINFO",
|
|
|
|
|
certtype, strlen (certtype),
|
2011-02-04 12:57:53 +01:00
|
|
|
|
buf, strlen (buf),
|
2021-04-29 15:26:36 +02:00
|
|
|
|
labelbuf, strlen (labelbuf),
|
2005-04-27 12:09:21 +00:00
|
|
|
|
NULL, (size_t)0);
|
|
|
|
|
xfree (buf);
|
2021-04-13 14:02:18 +02:00
|
|
|
|
xfree (labelbuf);
|
2005-04-27 12:09:21 +00:00
|
|
|
|
}
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
2004-12-02 07:48:09 +00:00
|
|
|
|
|
2005-04-27 12:09:21 +00:00
|
|
|
|
|
|
|
|
|
/* Get the keygrip of the private key object PRKDF. On success the
|
2020-03-31 19:55:15 +02:00
|
|
|
|
* keygrip, the algo and the length are stored in the KEYGRIP,
|
2020-04-07 20:38:15 +02:00
|
|
|
|
* KEYALGO, and KEYNBITS fields of the PRKDF object. */
|
2005-04-27 12:09:21 +00:00
|
|
|
|
static gpg_error_t
|
2020-03-31 19:55:15 +02:00
|
|
|
|
keygrip_from_prkdf (app_t app, prkdf_object_t prkdf)
|
2005-04-27 12:09:21 +00:00
|
|
|
|
{
|
|
|
|
|
gpg_error_t err;
|
|
|
|
|
cdf_object_t cdf;
|
|
|
|
|
unsigned char *der;
|
|
|
|
|
size_t derlen;
|
|
|
|
|
ksba_cert_t cert;
|
2020-03-31 19:55:15 +02:00
|
|
|
|
gcry_sexp_t s_pkey = NULL;
|
|
|
|
|
|
|
|
|
|
/* Easy if we got a cached version. */
|
|
|
|
|
if (prkdf->keygrip_valid)
|
|
|
|
|
return 0;
|
2005-04-27 12:09:21 +00:00
|
|
|
|
|
2020-04-07 20:38:15 +02:00
|
|
|
|
xfree (prkdf->common_name);
|
|
|
|
|
prkdf->common_name = NULL;
|
|
|
|
|
xfree (prkdf->serial_number);
|
|
|
|
|
prkdf->serial_number = NULL;
|
|
|
|
|
|
2021-04-29 15:26:36 +02:00
|
|
|
|
/* We could have also checked whether a public key directory file
|
|
|
|
|
* and a matching public key for PRKDF is available. This would
|
|
|
|
|
* make extraction of the key faster. However, this way we don't
|
|
|
|
|
* have a way to look at extended key attributes to check gpgusage.
|
|
|
|
|
* FIXME: Add public key lookup if no certificate was found. */
|
|
|
|
|
|
|
|
|
|
/* Look for a matching certificate. A certificate matches if the id
|
|
|
|
|
* matches the one of the private key info. If none was found we
|
|
|
|
|
* also try to match on the label. */
|
|
|
|
|
err = cdf_object_from_objid (app, prkdf->objidlen, prkdf->objid, &cdf);
|
|
|
|
|
if (gpg_err_code (err) == GPG_ERR_NOT_FOUND)
|
|
|
|
|
err = cdf_object_from_label (app, prkdf->label, &cdf);
|
|
|
|
|
if (!err && !cdf)
|
|
|
|
|
err = gpg_error (GPG_ERR_NOT_FOUND);
|
|
|
|
|
if (err)
|
|
|
|
|
goto leave;
|
2005-04-27 12:09:21 +00:00
|
|
|
|
|
|
|
|
|
err = readcert_by_cdf (app, cdf, &der, &derlen);
|
2004-12-02 07:48:09 +00:00
|
|
|
|
if (err)
|
2020-03-31 19:55:15 +02:00
|
|
|
|
goto leave;
|
2004-12-02 07:48:09 +00:00
|
|
|
|
|
2005-04-27 12:09:21 +00:00
|
|
|
|
err = ksba_cert_new (&cert);
|
|
|
|
|
if (!err)
|
|
|
|
|
err = ksba_cert_init_from_mem (cert, der, derlen);
|
|
|
|
|
xfree (der);
|
|
|
|
|
if (!err)
|
2020-05-05 08:07:11 +02:00
|
|
|
|
err = app_help_get_keygrip_string (cert, prkdf->keygrip, &s_pkey, NULL);
|
2021-04-29 15:26:36 +02:00
|
|
|
|
if (!err && !prkdf->gpgusage.any)
|
2020-04-07 20:38:15 +02:00
|
|
|
|
{
|
|
|
|
|
/* Try to get the CN and the SerialNumber from the certificate;
|
|
|
|
|
* we use a very simple approach here which should work in many
|
|
|
|
|
* cases. Eventually we should add a rfc-2253 parser into
|
|
|
|
|
* libksba to make it easier to parse such a string.
|
2021-04-29 15:26:36 +02:00
|
|
|
|
* We don't do this if this is marked as gpg key and thus
|
|
|
|
|
* has only a dummy certificate.
|
2020-04-07 20:38:15 +02:00
|
|
|
|
*
|
|
|
|
|
* First example string:
|
|
|
|
|
* "CN=Otto Schily,O=Miniluv,C=DE"
|
|
|
|
|
* Second example string:
|
|
|
|
|
* "2.5.4.5=#445452323030303236333531,2.5.4.4=#4B6F6368,"
|
|
|
|
|
* "2.5.4.42=#5765726E6572,CN=Werner Koch,OU=For testing"
|
|
|
|
|
* " purposes only!,O=Testorganisation,C=DE"
|
|
|
|
|
*/
|
|
|
|
|
char *dn = ksba_cert_get_subject (cert, 0);
|
|
|
|
|
if (dn)
|
|
|
|
|
{
|
|
|
|
|
char *p, *pend, *buf;
|
|
|
|
|
|
|
|
|
|
p = strstr (dn, "CN=");
|
|
|
|
|
if (p && (p==dn || p[-1] == ','))
|
|
|
|
|
{
|
|
|
|
|
p += 3;
|
|
|
|
|
if (!(pend = strchr (p, ',')))
|
|
|
|
|
pend = p + strlen (p);
|
|
|
|
|
if (pend && pend > p
|
|
|
|
|
&& (prkdf->common_name = xtrymalloc ((pend - p) + 1)))
|
|
|
|
|
{
|
|
|
|
|
memcpy (prkdf->common_name, p, pend-p);
|
|
|
|
|
prkdf->common_name[pend-p] = 0;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
p = strstr (dn, "2.5.4.5=#"); /* OID of the SerialNumber */
|
|
|
|
|
if (p && (p==dn || p[-1] == ','))
|
|
|
|
|
{
|
|
|
|
|
p += 9;
|
|
|
|
|
if (!(pend = strchr (p, ',')))
|
|
|
|
|
pend = p + strlen (p);
|
|
|
|
|
if (pend && pend > p
|
|
|
|
|
&& (buf = xtrymalloc ((pend - p) + 1)))
|
|
|
|
|
{
|
|
|
|
|
memcpy (buf, p, pend-p);
|
|
|
|
|
buf[pend-p] = 0;
|
|
|
|
|
if (!hex2str (buf, buf, strlen (buf)+1, NULL))
|
|
|
|
|
xfree (buf); /* Invalid hex encoding. */
|
|
|
|
|
else
|
|
|
|
|
prkdf->serial_number = buf;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
ksba_free (dn);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2021-04-29 15:26:36 +02:00
|
|
|
|
if (!err && !prkdf->keytime)
|
|
|
|
|
{
|
|
|
|
|
ksba_isotime_t isot;
|
|
|
|
|
time_t t;
|
|
|
|
|
|
|
|
|
|
ksba_cert_get_validity (cert, 0, isot);
|
|
|
|
|
t = isotime2epoch (isot);
|
|
|
|
|
prkdf->keytime = (t == (time_t)(-1))? 0 : (u32)t;
|
|
|
|
|
prkdf->have_keytime = 1;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (!err && !prkdf->keyalgostr)
|
|
|
|
|
prkdf->keyalgostr = pubkey_algo_string (s_pkey, NULL);
|
|
|
|
|
|
2005-04-27 12:09:21 +00:00
|
|
|
|
ksba_cert_release (cert);
|
2020-03-31 19:55:15 +02:00
|
|
|
|
if (err)
|
|
|
|
|
goto leave;
|
|
|
|
|
|
|
|
|
|
prkdf->keyalgo = get_pk_algo_from_key (s_pkey);
|
|
|
|
|
if (!prkdf->keyalgo)
|
|
|
|
|
{
|
|
|
|
|
err = gpg_error (GPG_ERR_PUBKEY_ALGO);
|
|
|
|
|
goto leave;
|
|
|
|
|
}
|
2005-04-27 12:09:21 +00:00
|
|
|
|
|
2020-03-31 19:55:15 +02:00
|
|
|
|
prkdf->keynbits = gcry_pk_get_nbits (s_pkey);
|
|
|
|
|
if (!prkdf->keynbits)
|
|
|
|
|
{
|
|
|
|
|
err = gpg_error (GPG_ERR_PUBKEY_ALGO);
|
|
|
|
|
goto leave;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
prkdf->keygrip_valid = 1; /* Yeah, got everything. */
|
|
|
|
|
|
|
|
|
|
leave:
|
|
|
|
|
gcry_sexp_release (s_pkey);
|
2005-04-27 12:09:21 +00:00
|
|
|
|
return err;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2020-04-02 12:39:56 +02:00
|
|
|
|
/* Return a malloced keyref string for PRKDF. Returns NULL on
|
2020-03-31 19:55:15 +02:00
|
|
|
|
* malloc failure. */
|
|
|
|
|
static char *
|
2020-04-02 12:39:56 +02:00
|
|
|
|
keyref_from_prkdf (app_t app, prkdf_object_t prkdf)
|
2020-03-31 19:55:15 +02:00
|
|
|
|
{
|
|
|
|
|
char *buf, *p;
|
|
|
|
|
|
2020-04-02 12:39:56 +02:00
|
|
|
|
buf = xtrymalloc (4 + 5 + prkdf->objidlen*2 + 1);
|
2020-03-31 19:55:15 +02:00
|
|
|
|
if (!buf)
|
|
|
|
|
return NULL;
|
|
|
|
|
p = stpcpy (buf, "P15");
|
2021-04-29 15:26:36 +02:00
|
|
|
|
if (app->app_local->home_df != DEFAULT_HOME_DF)
|
2020-03-31 19:55:15 +02:00
|
|
|
|
{
|
|
|
|
|
snprintf (p, 6, "-%04X",
|
|
|
|
|
(unsigned int)(app->app_local->home_df & 0xffff));
|
|
|
|
|
p += 5;
|
|
|
|
|
}
|
|
|
|
|
p = stpcpy (p, ".");
|
2020-04-02 12:39:56 +02:00
|
|
|
|
bin2hex (prkdf->objid, prkdf->objidlen, p);
|
2020-03-31 19:55:15 +02:00
|
|
|
|
return buf;
|
|
|
|
|
}
|
2005-04-27 12:09:21 +00:00
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* Helper to do_learn_status: Send information about all known
|
|
|
|
|
keypairs back. FIXME: much code duplication from
|
2014-07-24 16:16:53 +02:00
|
|
|
|
send_certinfo(). */
|
2005-04-27 12:09:21 +00:00
|
|
|
|
static gpg_error_t
|
2020-04-02 12:39:56 +02:00
|
|
|
|
send_keypairinfo (app_t app, ctrl_t ctrl, prkdf_object_t prkdf)
|
2005-04-27 12:09:21 +00:00
|
|
|
|
{
|
|
|
|
|
gpg_error_t err;
|
|
|
|
|
|
2020-04-02 12:39:56 +02:00
|
|
|
|
for (; prkdf; prkdf = prkdf->next)
|
2005-04-27 12:09:21 +00:00
|
|
|
|
{
|
2020-03-31 19:55:15 +02:00
|
|
|
|
char *buf;
|
2008-11-03 10:54:18 +00:00
|
|
|
|
int j;
|
2005-04-27 12:09:21 +00:00
|
|
|
|
|
2020-04-02 12:39:56 +02:00
|
|
|
|
buf = keyref_from_prkdf (app, prkdf);
|
2005-04-27 12:09:21 +00:00
|
|
|
|
if (!buf)
|
2006-09-14 16:50:33 +00:00
|
|
|
|
return gpg_error_from_syserror ();
|
2005-04-27 12:09:21 +00:00
|
|
|
|
|
2020-04-02 12:39:56 +02:00
|
|
|
|
err = keygrip_from_prkdf (app, prkdf);
|
2005-04-27 12:09:21 +00:00
|
|
|
|
if (err)
|
|
|
|
|
{
|
2020-03-31 10:40:21 +02:00
|
|
|
|
log_error ("p15: error getting keygrip from ");
|
2020-04-02 12:39:56 +02:00
|
|
|
|
for (j=0; j < prkdf->pathlen; j++)
|
|
|
|
|
log_printf ("%s%04hX", j?"/":"", prkdf->path[j]);
|
2005-04-27 12:09:21 +00:00
|
|
|
|
log_printf (": %s\n", gpg_strerror (err));
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
2020-04-03 11:21:32 +02:00
|
|
|
|
char usage[5];
|
2021-04-29 15:26:36 +02:00
|
|
|
|
char keytime[20];
|
|
|
|
|
const char *algostr;
|
2020-04-03 11:21:32 +02:00
|
|
|
|
size_t usagelen = 0;
|
|
|
|
|
|
2021-04-29 15:26:36 +02:00
|
|
|
|
if (prkdf->gpgusage.any)
|
|
|
|
|
{
|
|
|
|
|
if (prkdf->gpgusage.sign)
|
|
|
|
|
usage[usagelen++] = 's';
|
|
|
|
|
if (prkdf->gpgusage.cert)
|
|
|
|
|
usage[usagelen++] = 'c';
|
|
|
|
|
if (prkdf->gpgusage.encr)
|
|
|
|
|
usage[usagelen++] = 'e';
|
|
|
|
|
if (prkdf->gpgusage.auth)
|
|
|
|
|
usage[usagelen++] = 'a';
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
if ((prkdf->usageflags.sign
|
|
|
|
|
|| prkdf->usageflags.sign_recover
|
|
|
|
|
|| prkdf->usageflags.non_repudiation)
|
|
|
|
|
&& (!prkdf->extusage.valid
|
|
|
|
|
|| prkdf->extusage.sign))
|
|
|
|
|
usage[usagelen++] = 's';
|
|
|
|
|
if ((prkdf->usageflags.sign
|
|
|
|
|
|| prkdf->usageflags.sign_recover)
|
|
|
|
|
&& (!prkdf->extusage.valid || prkdf->extusage.sign))
|
|
|
|
|
usage[usagelen++] = 'c';
|
|
|
|
|
if ((prkdf->usageflags.decrypt
|
|
|
|
|
|| prkdf->usageflags.unwrap)
|
|
|
|
|
&& (!prkdf->extusage.valid || prkdf->extusage.encr))
|
|
|
|
|
usage[usagelen++] = 'e';
|
|
|
|
|
if ((prkdf->usageflags.sign
|
|
|
|
|
|| prkdf->usageflags.sign_recover)
|
|
|
|
|
&& (!prkdf->extusage.valid || prkdf->extusage.auth))
|
|
|
|
|
usage[usagelen++] = 'a';
|
|
|
|
|
}
|
2020-04-03 11:21:32 +02:00
|
|
|
|
|
2020-04-02 12:39:56 +02:00
|
|
|
|
log_assert (strlen (prkdf->keygrip) == 40);
|
2021-04-29 15:26:36 +02:00
|
|
|
|
if (prkdf->keytime && prkdf->have_keytime)
|
|
|
|
|
snprintf (keytime, sizeof keytime, "%lu",
|
|
|
|
|
(unsigned long)prkdf->keytime);
|
|
|
|
|
else
|
|
|
|
|
strcpy (keytime, "-");
|
|
|
|
|
|
|
|
|
|
algostr = prkdf->keyalgostr;
|
|
|
|
|
|
2005-04-27 12:09:21 +00:00
|
|
|
|
send_status_info (ctrl, "KEYPAIRINFO",
|
2020-04-02 12:39:56 +02:00
|
|
|
|
prkdf->keygrip, 2*KEYGRIP_LEN,
|
2011-02-04 12:57:53 +01:00
|
|
|
|
buf, strlen (buf),
|
2020-04-03 11:21:32 +02:00
|
|
|
|
usage, usagelen,
|
2021-04-29 15:26:36 +02:00
|
|
|
|
keytime, strlen (keytime),
|
|
|
|
|
algostr, strlen (algostr?algostr:""),
|
2005-04-27 12:09:21 +00:00
|
|
|
|
NULL, (size_t)0);
|
|
|
|
|
}
|
|
|
|
|
xfree (buf);
|
|
|
|
|
}
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
2021-04-29 15:26:36 +02:00
|
|
|
|
/* This is the handler for the LEARN command. Note that if
|
|
|
|
|
* APP_LEARN_FLAG_REREAD is set and this function returns an error,
|
|
|
|
|
* the caller must deinitialize this application. */
|
2011-02-04 12:57:53 +01:00
|
|
|
|
static gpg_error_t
|
2009-03-18 11:18:56 +00:00
|
|
|
|
do_learn_status (app_t app, ctrl_t ctrl, unsigned int flags)
|
2005-04-27 12:09:21 +00:00
|
|
|
|
{
|
|
|
|
|
gpg_error_t err;
|
|
|
|
|
|
2021-04-29 15:26:36 +02:00
|
|
|
|
if (flags & APP_LEARN_FLAG_REREAD)
|
|
|
|
|
{
|
|
|
|
|
err = read_p15_info (app);
|
|
|
|
|
if (err)
|
|
|
|
|
return err;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if ((flags & APP_LEARN_FLAG_KEYPAIRINFO))
|
2009-03-18 11:18:56 +00:00
|
|
|
|
err = 0;
|
|
|
|
|
else
|
|
|
|
|
{
|
2020-04-03 11:21:32 +02:00
|
|
|
|
err = do_getattr (app, ctrl, "MANUFACTURER");
|
|
|
|
|
if (!err)
|
|
|
|
|
err = send_certinfo (app, ctrl, "100",
|
|
|
|
|
app->app_local->certificate_info);
|
2009-03-18 11:18:56 +00:00
|
|
|
|
if (!err)
|
|
|
|
|
err = send_certinfo (app, ctrl, "101",
|
|
|
|
|
app->app_local->trusted_certificate_info);
|
|
|
|
|
if (!err)
|
|
|
|
|
err = send_certinfo (app, ctrl, "102",
|
|
|
|
|
app->app_local->useful_certificate_info);
|
|
|
|
|
}
|
|
|
|
|
|
2005-04-27 12:09:21 +00:00
|
|
|
|
if (!err)
|
|
|
|
|
err = send_keypairinfo (app, ctrl, app->app_local->private_key_info);
|
|
|
|
|
|
2021-04-29 15:26:36 +02:00
|
|
|
|
if (!err)
|
|
|
|
|
err = do_getattr (app, ctrl, "CHV-STATUS");
|
|
|
|
|
if (!err)
|
|
|
|
|
err = do_getattr (app, ctrl, "CHV-LABEL");
|
|
|
|
|
|
|
|
|
|
|
2005-04-27 12:09:21 +00:00
|
|
|
|
return err;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2021-04-29 15:26:36 +02:00
|
|
|
|
/* Read a certificate using the information in CDF and return the
|
|
|
|
|
* certificate in a newly malloced buffer R_CERT and its length
|
|
|
|
|
* R_CERTLEN. Also parses the certificate. R_CERT and R_CERTLEN may
|
|
|
|
|
* be NULL to do just the caching. */
|
2005-04-27 12:09:21 +00:00
|
|
|
|
static gpg_error_t
|
|
|
|
|
readcert_by_cdf (app_t app, cdf_object_t cdf,
|
|
|
|
|
unsigned char **r_cert, size_t *r_certlen)
|
|
|
|
|
{
|
|
|
|
|
gpg_error_t err;
|
|
|
|
|
unsigned char *buffer = NULL;
|
|
|
|
|
const unsigned char *p, *save_p;
|
|
|
|
|
size_t buflen, n;
|
|
|
|
|
int class, tag, constructed, ndef;
|
|
|
|
|
size_t totobjlen, objlen, hdrlen;
|
|
|
|
|
int rootca;
|
|
|
|
|
int i;
|
|
|
|
|
|
2021-04-29 15:26:36 +02:00
|
|
|
|
if (r_cert)
|
|
|
|
|
*r_cert = NULL;
|
|
|
|
|
if (r_certlen)
|
|
|
|
|
*r_certlen = 0;
|
2005-04-27 12:09:21 +00:00
|
|
|
|
|
|
|
|
|
/* First check whether it has been cached. */
|
2021-04-29 15:26:36 +02:00
|
|
|
|
if (cdf->cert)
|
2005-04-27 12:09:21 +00:00
|
|
|
|
{
|
2021-04-29 15:26:36 +02:00
|
|
|
|
const unsigned char *image;
|
|
|
|
|
size_t imagelen;
|
|
|
|
|
|
|
|
|
|
if (!r_cert || !r_certlen)
|
|
|
|
|
return 0; /* Caller does not actually want the result. */
|
|
|
|
|
|
|
|
|
|
image = ksba_cert_get_image (cdf->cert, &imagelen);
|
|
|
|
|
if (!image)
|
|
|
|
|
{
|
|
|
|
|
log_error ("p15: ksba_cert_get_image failed\n");
|
|
|
|
|
return gpg_error (GPG_ERR_INTERNAL);
|
|
|
|
|
}
|
|
|
|
|
*r_cert = xtrymalloc (imagelen);
|
2005-04-27 12:09:21 +00:00
|
|
|
|
if (!*r_cert)
|
2006-09-14 16:50:33 +00:00
|
|
|
|
return gpg_error_from_syserror ();
|
2021-04-29 15:26:36 +02:00
|
|
|
|
memcpy (*r_cert, image, imagelen);
|
|
|
|
|
*r_certlen = imagelen;
|
2005-04-27 12:09:21 +00:00
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Read the entire file. fixme: This could be optimized by first
|
|
|
|
|
reading the header to figure out how long the certificate
|
|
|
|
|
actually is. */
|
|
|
|
|
err = select_ef_by_path (app, cdf->path, cdf->pathlen);
|
|
|
|
|
if (err)
|
|
|
|
|
goto leave;
|
|
|
|
|
|
2021-06-18 17:32:01 +02:00
|
|
|
|
if (app->app_local->no_extended_mode)
|
|
|
|
|
err = iso7816_read_binary_ext (app_get_slot (app), 0, cdf->off, 0,
|
|
|
|
|
&buffer, &buflen, NULL);
|
|
|
|
|
else
|
|
|
|
|
err = iso7816_read_binary_ext (app_get_slot (app), 1, cdf->off, cdf->len,
|
|
|
|
|
&buffer, &buflen, NULL);
|
2005-04-27 12:09:21 +00:00
|
|
|
|
if (!err && (!buflen || *buffer == 0xff))
|
|
|
|
|
err = gpg_error (GPG_ERR_NOT_FOUND);
|
|
|
|
|
if (err)
|
|
|
|
|
{
|
2020-03-31 10:40:21 +02:00
|
|
|
|
log_error ("p15: error reading certificate id=");
|
2005-04-27 12:09:21 +00:00
|
|
|
|
for (i=0; i < cdf->objidlen; i++)
|
|
|
|
|
log_printf ("%02X", cdf->objid[i]);
|
2020-03-31 10:40:21 +02:00
|
|
|
|
log_printf (" at ");
|
|
|
|
|
for (i=0; i < cdf->pathlen; i++)
|
|
|
|
|
log_printf ("%s%04hX", i? "/":"", cdf->path[i]);
|
2005-04-27 12:09:21 +00:00
|
|
|
|
log_printf (": %s\n", gpg_strerror (err));
|
|
|
|
|
goto leave;
|
|
|
|
|
}
|
2011-02-04 12:57:53 +01:00
|
|
|
|
|
2005-04-27 12:09:21 +00:00
|
|
|
|
/* Check whether this is really a certificate. */
|
|
|
|
|
p = buffer;
|
|
|
|
|
n = buflen;
|
|
|
|
|
err = parse_ber_header (&p, &n, &class, &tag, &constructed,
|
|
|
|
|
&ndef, &objlen, &hdrlen);
|
|
|
|
|
if (err)
|
|
|
|
|
goto leave;
|
|
|
|
|
|
|
|
|
|
if (class == CLASS_UNIVERSAL && tag == TAG_SEQUENCE && constructed)
|
|
|
|
|
rootca = 0;
|
|
|
|
|
else if ( class == CLASS_UNIVERSAL && tag == TAG_SET && constructed )
|
|
|
|
|
rootca = 1;
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
err = gpg_error (GPG_ERR_INV_OBJ);
|
|
|
|
|
goto leave;
|
|
|
|
|
}
|
|
|
|
|
totobjlen = objlen + hdrlen;
|
2021-04-29 15:26:36 +02:00
|
|
|
|
log_assert (totobjlen <= buflen);
|
2005-04-27 12:09:21 +00:00
|
|
|
|
|
|
|
|
|
err = parse_ber_header (&p, &n, &class, &tag, &constructed,
|
|
|
|
|
&ndef, &objlen, &hdrlen);
|
|
|
|
|
if (err)
|
|
|
|
|
goto leave;
|
2011-02-04 12:57:53 +01:00
|
|
|
|
|
2005-04-27 12:09:21 +00:00
|
|
|
|
if (!rootca
|
|
|
|
|
&& class == CLASS_UNIVERSAL && tag == TAG_OBJECT_ID && !constructed)
|
|
|
|
|
{
|
|
|
|
|
/* The certificate seems to be contained in a userCertificate
|
|
|
|
|
container. Skip this and assume the following sequence is
|
|
|
|
|
the certificate. */
|
|
|
|
|
if (n < objlen)
|
|
|
|
|
{
|
|
|
|
|
err = gpg_error (GPG_ERR_INV_OBJ);
|
|
|
|
|
goto leave;
|
|
|
|
|
}
|
|
|
|
|
p += objlen;
|
|
|
|
|
n -= objlen;
|
|
|
|
|
save_p = p;
|
|
|
|
|
err = parse_ber_header (&p, &n, &class, &tag, &constructed,
|
|
|
|
|
&ndef, &objlen, &hdrlen);
|
2011-02-04 12:57:53 +01:00
|
|
|
|
if (err)
|
2005-04-27 12:09:21 +00:00
|
|
|
|
goto leave;
|
|
|
|
|
if ( !(class == CLASS_UNIVERSAL && tag == TAG_SEQUENCE && constructed) )
|
|
|
|
|
{
|
|
|
|
|
err = gpg_error (GPG_ERR_INV_OBJ);
|
|
|
|
|
goto leave;
|
|
|
|
|
}
|
|
|
|
|
totobjlen = objlen + hdrlen;
|
2021-04-29 15:26:36 +02:00
|
|
|
|
log_assert (save_p + totobjlen <= buffer + buflen);
|
2005-04-27 12:09:21 +00:00
|
|
|
|
memmove (buffer, save_p, totobjlen);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2021-04-29 15:26:36 +02:00
|
|
|
|
/* Try to parse and cache the certificate. */
|
|
|
|
|
err = ksba_cert_new (&cdf->cert);
|
|
|
|
|
if (!err)
|
2005-04-27 12:09:21 +00:00
|
|
|
|
{
|
2021-04-29 15:26:36 +02:00
|
|
|
|
err = ksba_cert_init_from_mem (cdf->cert, buffer, totobjlen);
|
|
|
|
|
if (!err) /* Call us to use the just cached cert object. */
|
|
|
|
|
err = readcert_by_cdf (app, cdf, r_cert, r_certlen);
|
|
|
|
|
if (err)
|
|
|
|
|
{
|
|
|
|
|
ksba_cert_release (cdf->cert);
|
|
|
|
|
cdf->cert = NULL;
|
|
|
|
|
}
|
2005-04-27 12:09:21 +00:00
|
|
|
|
|
2021-04-29 15:26:36 +02:00
|
|
|
|
}
|
|
|
|
|
if (err)
|
|
|
|
|
{
|
|
|
|
|
log_error ("p15: caching certificate failed: %s\n",
|
|
|
|
|
gpg_strerror (err));
|
|
|
|
|
/* We return the certificate anyway so that the caller has a
|
|
|
|
|
* chance to get an even unsupported or broken certificate. */
|
|
|
|
|
if (r_cert && r_certlen)
|
|
|
|
|
{
|
|
|
|
|
*r_cert = buffer;
|
|
|
|
|
buffer = NULL;
|
|
|
|
|
*r_certlen = totobjlen;
|
|
|
|
|
}
|
|
|
|
|
}
|
2005-04-27 12:09:21 +00:00
|
|
|
|
|
|
|
|
|
leave:
|
|
|
|
|
xfree (buffer);
|
|
|
|
|
return err;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* Handler for the READCERT command.
|
|
|
|
|
|
|
|
|
|
Read the certificate with id CERTID (as returned by learn_status in
|
|
|
|
|
the CERTINFO status lines) and return it in the freshly allocated
|
|
|
|
|
buffer to be stored at R_CERT and its length at R_CERTLEN. A error
|
|
|
|
|
code will be returned on failure and R_CERT and R_CERTLEN will be
|
2014-07-24 16:16:53 +02:00
|
|
|
|
set to (NULL,0). */
|
2011-02-04 12:57:53 +01:00
|
|
|
|
static gpg_error_t
|
2005-04-27 12:09:21 +00:00
|
|
|
|
do_readcert (app_t app, const char *certid,
|
|
|
|
|
unsigned char **r_cert, size_t *r_certlen)
|
|
|
|
|
{
|
|
|
|
|
gpg_error_t err;
|
|
|
|
|
cdf_object_t cdf;
|
|
|
|
|
|
|
|
|
|
*r_cert = NULL;
|
|
|
|
|
*r_certlen = 0;
|
|
|
|
|
err = cdf_object_from_certid (app, certid, &cdf);
|
|
|
|
|
if (!err)
|
2014-07-24 16:16:53 +02:00
|
|
|
|
err = readcert_by_cdf (app, cdf, r_cert, r_certlen);
|
2005-04-27 12:09:21 +00:00
|
|
|
|
return err;
|
2004-12-02 07:48:09 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2021-04-29 15:26:36 +02:00
|
|
|
|
/* Sort helper for an array of authentication objects. */
|
|
|
|
|
static int
|
|
|
|
|
compare_aodf_objid (const void *arg_a, const void *arg_b)
|
|
|
|
|
{
|
|
|
|
|
const aodf_object_t a = *(const aodf_object_t *)arg_a;
|
|
|
|
|
const aodf_object_t b = *(const aodf_object_t *)arg_b;
|
|
|
|
|
int rc;
|
|
|
|
|
|
|
|
|
|
rc = memcmp (a->objid, b->objid,
|
|
|
|
|
a->objidlen < b->objidlen? a->objidlen : b->objidlen);
|
|
|
|
|
if (!rc)
|
|
|
|
|
{
|
|
|
|
|
if (a->objidlen < b->objidlen)
|
|
|
|
|
rc = -1;
|
|
|
|
|
else if (a->objidlen > b->objidlen)
|
|
|
|
|
rc = 1;
|
|
|
|
|
}
|
|
|
|
|
return rc;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
static void
|
|
|
|
|
send_key_fpr_line (ctrl_t ctrl, int number, const unsigned char *fpr)
|
|
|
|
|
{
|
|
|
|
|
char buf[41];
|
|
|
|
|
char numbuf[25];
|
|
|
|
|
|
|
|
|
|
bin2hex (fpr, 20, buf);
|
|
|
|
|
if (number == -1)
|
|
|
|
|
*numbuf = 0; /* Don't print the key number */
|
|
|
|
|
else
|
|
|
|
|
snprintf (numbuf, sizeof numbuf, "%d", number);
|
|
|
|
|
send_status_info (ctrl, "KEY-FPR",
|
|
|
|
|
numbuf, (size_t)strlen(numbuf),
|
|
|
|
|
buf, (size_t)strlen (buf),
|
|
|
|
|
NULL, 0);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* If possible Emit a FPR-KEY status line for the private key object
|
|
|
|
|
* PRKDF using NUMBER as index. */
|
|
|
|
|
static void
|
|
|
|
|
send_key_fpr (app_t app, ctrl_t ctrl, prkdf_object_t prkdf, int number)
|
|
|
|
|
{
|
|
|
|
|
gpg_error_t err;
|
|
|
|
|
cdf_object_t cdf;
|
|
|
|
|
unsigned char *pk, *fixed_pk;
|
|
|
|
|
size_t pklen, fixed_pklen;
|
|
|
|
|
const unsigned char *m, *e, *q;
|
|
|
|
|
size_t mlen, elen, qlen;
|
|
|
|
|
unsigned char fpr20[20];
|
|
|
|
|
|
|
|
|
|
if (cdf_object_from_objid (app, prkdf->objidlen, prkdf->objid, &cdf)
|
|
|
|
|
&& cdf_object_from_label (app, prkdf->label, &cdf))
|
|
|
|
|
return;
|
|
|
|
|
if (!cdf->cert)
|
|
|
|
|
readcert_by_cdf (app, cdf, NULL, NULL);
|
|
|
|
|
if (!cdf->cert)
|
|
|
|
|
return;
|
|
|
|
|
if (!prkdf->have_keytime)
|
|
|
|
|
return;
|
|
|
|
|
pk = ksba_cert_get_public_key (cdf->cert);
|
|
|
|
|
if (!pk)
|
|
|
|
|
return;
|
|
|
|
|
pklen = gcry_sexp_canon_len (pk, 0, NULL, &err);
|
|
|
|
|
|
|
|
|
|
if (uncompress_ecc_q_in_canon_sexp (pk, pklen, &fixed_pk, &fixed_pklen))
|
|
|
|
|
{
|
|
|
|
|
xfree (pk);
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
if (fixed_pk)
|
|
|
|
|
{
|
|
|
|
|
xfree (pk); pk = NULL;
|
|
|
|
|
pk = fixed_pk;
|
|
|
|
|
pklen = fixed_pklen;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
switch (prkdf->keyalgo)
|
|
|
|
|
{
|
|
|
|
|
case GCRY_PK_RSA:
|
|
|
|
|
if (!get_rsa_pk_from_canon_sexp (pk, pklen,
|
|
|
|
|
&m, &mlen, &e, &elen)
|
|
|
|
|
&& !compute_openpgp_fpr_rsa (4,
|
|
|
|
|
prkdf->keytime,
|
|
|
|
|
m, mlen, e, elen,
|
|
|
|
|
fpr20, NULL))
|
|
|
|
|
send_key_fpr_line (ctrl, number, fpr20);
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case GCRY_PK_ECC:
|
|
|
|
|
case GCRY_PK_ECDSA:
|
|
|
|
|
case GCRY_PK_ECDH:
|
|
|
|
|
case GCRY_PK_EDDSA:
|
|
|
|
|
/* Note that NUMBER 2 indicates the encryption key. */
|
|
|
|
|
if (!get_ecc_q_from_canon_sexp (pk, pklen, &q, &qlen)
|
|
|
|
|
&& !compute_openpgp_fpr_ecc (4,
|
|
|
|
|
prkdf->keytime,
|
|
|
|
|
prkdf->keyalgostr,
|
|
|
|
|
number == 2,
|
|
|
|
|
q, qlen,
|
|
|
|
|
prkdf->ecdh_kdf, 4,
|
|
|
|
|
fpr20, NULL))
|
|
|
|
|
send_key_fpr_line (ctrl, number, fpr20);
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
default: /* No Fingerprint for an unknown algo. */
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
}
|
|
|
|
|
xfree (pk);
|
|
|
|
|
}
|
|
|
|
|
|
2005-09-09 11:18:08 +00:00
|
|
|
|
|
|
|
|
|
/* Implement the GETATTR command. This is similar to the LEARN
|
|
|
|
|
command but returns just one value via the status interface. */
|
2011-02-04 12:57:53 +01:00
|
|
|
|
static gpg_error_t
|
2005-09-09 11:18:08 +00:00
|
|
|
|
do_getattr (app_t app, ctrl_t ctrl, const char *name)
|
|
|
|
|
{
|
|
|
|
|
gpg_error_t err;
|
2020-04-08 07:54:01 +02:00
|
|
|
|
prkdf_object_t prkdf;
|
2005-09-09 11:18:08 +00:00
|
|
|
|
|
2020-04-03 11:21:32 +02:00
|
|
|
|
if (!strcmp (name, "$AUTHKEYID")
|
|
|
|
|
|| !strcmp (name, "$ENCRKEYID")
|
|
|
|
|
|| !strcmp (name, "$SIGNKEYID"))
|
2005-09-09 11:18:08 +00:00
|
|
|
|
{
|
2020-04-03 11:21:32 +02:00
|
|
|
|
char *buf;
|
2005-09-09 11:18:08 +00:00
|
|
|
|
|
2020-04-03 11:21:32 +02:00
|
|
|
|
/* We return the ID of the first private key capable of the
|
2021-04-29 15:26:36 +02:00
|
|
|
|
* requested action. If any gpgusage flag has been set for the
|
|
|
|
|
* card we consult the gpgusage flags and not the regualr usage
|
|
|
|
|
* flags.
|
|
|
|
|
*/
|
|
|
|
|
/* FIXME: This changed: Note that we do not yet return
|
2020-04-03 11:21:32 +02:00
|
|
|
|
* non_repudiation keys for $SIGNKEYID because our D-Trust
|
|
|
|
|
* testcard uses rsaPSS, which is not supported by gpgsm and not
|
|
|
|
|
* covered by the VS-NfD approval. */
|
2005-09-09 11:18:08 +00:00
|
|
|
|
for (prkdf = app->app_local->private_key_info; prkdf;
|
|
|
|
|
prkdf = prkdf->next)
|
2020-04-03 11:21:32 +02:00
|
|
|
|
{
|
2021-04-29 15:26:36 +02:00
|
|
|
|
if (app->app_local->any_gpgusage)
|
|
|
|
|
{
|
|
|
|
|
if ((name[1] == 'A' && prkdf->gpgusage.auth)
|
|
|
|
|
|| (name[1] == 'E' && prkdf->gpgusage.encr)
|
|
|
|
|
|| (name[1] == 'S' && prkdf->gpgusage.sign))
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
if ((name[1] == 'A' && (prkdf->usageflags.sign
|
2020-04-03 11:21:32 +02:00
|
|
|
|
|| prkdf->usageflags.sign_recover))
|
2021-04-29 15:26:36 +02:00
|
|
|
|
|| (name[1] == 'E' && (prkdf->usageflags.decrypt
|
|
|
|
|
|| prkdf->usageflags.unwrap))
|
|
|
|
|
|| (name[1] == 'S' && (prkdf->usageflags.sign
|
|
|
|
|
|| prkdf->usageflags.sign_recover)))
|
|
|
|
|
break;
|
|
|
|
|
}
|
2020-04-03 11:21:32 +02:00
|
|
|
|
}
|
2005-09-09 11:18:08 +00:00
|
|
|
|
if (prkdf)
|
|
|
|
|
{
|
2020-04-03 11:21:32 +02:00
|
|
|
|
buf = keyref_from_prkdf (app, prkdf);
|
2005-09-09 11:18:08 +00:00
|
|
|
|
if (!buf)
|
2006-09-14 16:50:33 +00:00
|
|
|
|
return gpg_error_from_syserror ();
|
2005-09-09 11:18:08 +00:00
|
|
|
|
|
|
|
|
|
send_status_info (ctrl, name, buf, strlen (buf), NULL, 0);
|
|
|
|
|
xfree (buf);
|
|
|
|
|
}
|
2020-04-03 11:21:32 +02:00
|
|
|
|
return 0;
|
2005-09-09 11:18:08 +00:00
|
|
|
|
}
|
|
|
|
|
else if (!strcmp (name, "$DISPSERIALNO"))
|
|
|
|
|
{
|
|
|
|
|
/* For certain cards we return special IDs. There is no
|
|
|
|
|
general rule for it so we need to decide case by case. */
|
|
|
|
|
if (app->app_local->card_type == CARD_TYPE_BELPIC)
|
|
|
|
|
{
|
2006-10-23 14:02:13 +00:00
|
|
|
|
/* The eID card has a card number printed on the front matter
|
2005-09-09 11:18:08 +00:00
|
|
|
|
which seems to be a good indication. */
|
|
|
|
|
unsigned char *buffer;
|
|
|
|
|
const unsigned char *p;
|
|
|
|
|
size_t buflen, n;
|
|
|
|
|
unsigned short path[] = { 0x3F00, 0xDF01, 0x4031 };
|
|
|
|
|
|
|
|
|
|
err = select_ef_by_path (app, path, DIM(path) );
|
|
|
|
|
if (!err)
|
2021-04-29 15:26:36 +02:00
|
|
|
|
err = iso7816_read_binary (app_get_slot (app), 0, 0,
|
|
|
|
|
&buffer, &buflen);
|
2005-09-09 11:18:08 +00:00
|
|
|
|
if (err)
|
|
|
|
|
{
|
2020-03-31 10:40:21 +02:00
|
|
|
|
log_error ("p15: error accessing EF(ID): %s\n",
|
|
|
|
|
gpg_strerror (err));
|
2005-09-09 11:18:08 +00:00
|
|
|
|
return err;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
p = find_tlv (buffer, buflen, 1, &n);
|
|
|
|
|
if (p && n == 12)
|
|
|
|
|
{
|
|
|
|
|
char tmp[12+2+1];
|
|
|
|
|
memcpy (tmp, p, 3);
|
|
|
|
|
tmp[3] = '-';
|
|
|
|
|
memcpy (tmp+4, p+3, 7);
|
|
|
|
|
tmp[11] = '-';
|
|
|
|
|
memcpy (tmp+12, p+10, 2);
|
|
|
|
|
tmp[14] = 0;
|
|
|
|
|
send_status_info (ctrl, name, tmp, strlen (tmp), NULL, 0);
|
|
|
|
|
xfree (buffer);
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
xfree (buffer);
|
|
|
|
|
}
|
2020-04-08 07:54:01 +02:00
|
|
|
|
else
|
|
|
|
|
{
|
2021-04-29 15:26:36 +02:00
|
|
|
|
char *sn;
|
|
|
|
|
|
2020-04-08 07:54:01 +02:00
|
|
|
|
/* We use the first private key object which has a serial
|
|
|
|
|
* number set. If none was found, we parse the first
|
|
|
|
|
* object and see whether this has then a serial number. */
|
|
|
|
|
for (prkdf = app->app_local->private_key_info; prkdf;
|
|
|
|
|
prkdf = prkdf->next)
|
|
|
|
|
if (prkdf->serial_number)
|
|
|
|
|
break;
|
|
|
|
|
if (!prkdf && app->app_local->private_key_info)
|
|
|
|
|
{
|
|
|
|
|
prkdf = app->app_local->private_key_info;
|
|
|
|
|
keygrip_from_prkdf (app, prkdf);
|
|
|
|
|
if (!prkdf->serial_number)
|
|
|
|
|
prkdf = NULL;
|
|
|
|
|
}
|
2021-04-29 15:26:36 +02:00
|
|
|
|
sn = get_dispserialno (app, prkdf);
|
|
|
|
|
/* Unless there is a bogus S/N in the cert, or the product
|
|
|
|
|
* has a different strategy for the display-s/n, we should
|
|
|
|
|
* have a suitable one from the cert now. */
|
|
|
|
|
if (sn)
|
2020-04-08 07:54:01 +02:00
|
|
|
|
{
|
|
|
|
|
err = send_status_printf (ctrl, name, "%s", sn);
|
|
|
|
|
xfree (sn);
|
|
|
|
|
return err;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
/* No abbreviated serial number. */
|
2005-09-09 11:18:08 +00:00
|
|
|
|
}
|
2020-04-03 11:21:32 +02:00
|
|
|
|
else if (!strcmp (name, "MANUFACTURER"))
|
|
|
|
|
{
|
2021-04-29 15:26:36 +02:00
|
|
|
|
if (app->app_local->manufacturer_id
|
2021-05-18 09:44:53 +02:00
|
|
|
|
&& !strchr (app->app_local->manufacturer_id, '[')
|
|
|
|
|
&& app->app_local->card_product)
|
2021-04-29 15:26:36 +02:00
|
|
|
|
return send_status_printf (ctrl, "MANUFACTURER", "0 %s [%s]",
|
|
|
|
|
app->app_local->manufacturer_id,
|
|
|
|
|
cardproduct2str (app->app_local->card_product));
|
|
|
|
|
else if (app->app_local->manufacturer_id)
|
2020-04-03 11:21:32 +02:00
|
|
|
|
return send_status_printf (ctrl, "MANUFACTURER", "0 %s",
|
|
|
|
|
app->app_local->manufacturer_id);
|
|
|
|
|
else
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
2021-04-29 15:26:36 +02:00
|
|
|
|
else if (!strcmp (name, "CHV-STATUS") || !strcmp (name, "CHV-LABEL"))
|
|
|
|
|
{
|
|
|
|
|
int is_label = (name[4] == 'L');
|
|
|
|
|
aodf_object_t aodf;
|
|
|
|
|
aodf_object_t aodfarray[16];
|
|
|
|
|
int naodf = 0;
|
|
|
|
|
membuf_t mb;
|
|
|
|
|
char *p;
|
|
|
|
|
int i;
|
|
|
|
|
|
|
|
|
|
/* Put the AODFs into an array for easier sorting. Note that we
|
|
|
|
|
* handle onl the first 16 encountrer which should be more than
|
|
|
|
|
* enough. */
|
|
|
|
|
for (aodf = app->app_local->auth_object_info;
|
|
|
|
|
aodf && naodf < DIM(aodfarray); aodf = aodf->next)
|
|
|
|
|
if (aodf->objidlen && aodf->pin_reference_valid)
|
|
|
|
|
aodfarray[naodf++] = aodf;
|
|
|
|
|
qsort (aodfarray, naodf, sizeof *aodfarray, compare_aodf_objid);
|
|
|
|
|
|
|
|
|
|
init_membuf (&mb, 256);
|
|
|
|
|
for (i = 0; i < naodf; i++)
|
|
|
|
|
{
|
|
|
|
|
/* int j; */
|
|
|
|
|
/* log_debug ("p15: AODF[%d] pinref=%lu id=", */
|
|
|
|
|
/* i, aodfarray[i]->pin_reference); */
|
|
|
|
|
/* for (j=0; j < aodfarray[i]->objidlen; j++) */
|
|
|
|
|
/* log_printf ("%02X", aodfarray[i]->objid[j]); */
|
|
|
|
|
/* Note that there is no need to percent escape the label
|
|
|
|
|
* because all white space have been replaced by '_'. */
|
|
|
|
|
if (is_label)
|
|
|
|
|
put_membuf_printf (&mb, "%s%s", i? " ":"",
|
|
|
|
|
(aodfarray[i]->label
|
|
|
|
|
&& *aodfarray[i]->label)?
|
|
|
|
|
aodfarray[i]->label:"X");
|
|
|
|
|
else
|
|
|
|
|
put_membuf_printf
|
|
|
|
|
(&mb, "%s%d", i? " ":"",
|
|
|
|
|
iso7816_verify_status (app_get_slot (app),
|
|
|
|
|
aodfarray[i]->pin_reference));
|
|
|
|
|
}
|
|
|
|
|
put_membuf( &mb, "", 1);
|
|
|
|
|
p = get_membuf (&mb, NULL);
|
|
|
|
|
if (!p)
|
|
|
|
|
return gpg_error_from_syserror ();
|
|
|
|
|
err = send_status_direct (ctrl, is_label? "CHV-LABEL":"CHV-STATUS", p);
|
|
|
|
|
xfree (p);
|
|
|
|
|
return err;
|
|
|
|
|
}
|
|
|
|
|
else if (!strcmp (name, "KEY-LABEL"))
|
|
|
|
|
{
|
|
|
|
|
/* Send KEY-LABEL lines for all private key objects. */
|
|
|
|
|
const char *label;
|
|
|
|
|
char *idbuf, *labelbuf;
|
|
|
|
|
|
|
|
|
|
for (prkdf = app->app_local->private_key_info; prkdf;
|
|
|
|
|
prkdf = prkdf->next)
|
|
|
|
|
{
|
|
|
|
|
idbuf = keyref_from_prkdf (app, prkdf);
|
|
|
|
|
if (!idbuf)
|
|
|
|
|
return gpg_error_from_syserror ();
|
|
|
|
|
|
|
|
|
|
label = (prkdf->label && *prkdf->label)? prkdf->label : "-";
|
|
|
|
|
labelbuf = percent_data_escape (0, NULL, label, strlen (label));
|
|
|
|
|
if (!labelbuf)
|
|
|
|
|
{
|
|
|
|
|
xfree (idbuf);
|
|
|
|
|
return gpg_error_from_syserror ();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
send_status_info (ctrl, name,
|
|
|
|
|
idbuf, strlen (idbuf),
|
|
|
|
|
labelbuf, strlen(labelbuf),
|
|
|
|
|
NULL, 0);
|
|
|
|
|
xfree (idbuf);
|
|
|
|
|
xfree (labelbuf);
|
|
|
|
|
}
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
else if (!strcmp (name, "KEY-FPR"))
|
|
|
|
|
{
|
|
|
|
|
/* Send KEY-FPR for the two openpgp keys. */
|
|
|
|
|
for (prkdf = app->app_local->private_key_info; prkdf;
|
|
|
|
|
prkdf = prkdf->next)
|
|
|
|
|
{
|
|
|
|
|
if (app->app_local->any_gpgusage)
|
|
|
|
|
{
|
|
|
|
|
if (prkdf->gpgusage.sign)
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
if (prkdf->usageflags.sign || prkdf->usageflags.sign_recover)
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
if (prkdf)
|
|
|
|
|
send_key_fpr (app, ctrl, prkdf, 1);
|
|
|
|
|
for (prkdf = app->app_local->private_key_info; prkdf;
|
|
|
|
|
prkdf = prkdf->next)
|
|
|
|
|
{
|
|
|
|
|
if (app->app_local->any_gpgusage)
|
|
|
|
|
{
|
|
|
|
|
if (prkdf->gpgusage.encr)
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
if (prkdf->usageflags.decrypt || prkdf->usageflags.unwrap)
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
if (prkdf)
|
|
|
|
|
send_key_fpr (app, ctrl, prkdf, 2);
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
2011-02-04 12:57:53 +01:00
|
|
|
|
return gpg_error (GPG_ERR_INV_NAME);
|
2005-09-09 11:18:08 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
2005-04-27 19:47:53 +00:00
|
|
|
|
/* Micardo cards require special treatment. This is a helper for the
|
|
|
|
|
crypto functions to manage the security environment. We expect that
|
|
|
|
|
the key file has already been selected. FID is the one of the
|
|
|
|
|
selected key. */
|
|
|
|
|
static gpg_error_t
|
|
|
|
|
micardo_mse (app_t app, unsigned short fid)
|
|
|
|
|
{
|
|
|
|
|
gpg_error_t err;
|
|
|
|
|
int recno;
|
|
|
|
|
unsigned short refdata = 0;
|
|
|
|
|
int se_num;
|
|
|
|
|
unsigned char msebuf[10];
|
|
|
|
|
|
|
|
|
|
/* Read the KeyD file containing extra information on keys. */
|
2021-04-29 15:26:36 +02:00
|
|
|
|
err = iso7816_select_file (app_get_slot (app), 0x0013, 0);
|
2005-04-27 19:47:53 +00:00
|
|
|
|
if (err)
|
|
|
|
|
{
|
2020-03-31 10:40:21 +02:00
|
|
|
|
log_error ("p15: error reading EF_keyD: %s\n", gpg_strerror (err));
|
2005-04-27 19:47:53 +00:00
|
|
|
|
return err;
|
|
|
|
|
}
|
2011-02-04 12:57:53 +01:00
|
|
|
|
|
2005-04-27 19:47:53 +00:00
|
|
|
|
for (recno = 1, se_num = -1; ; recno++)
|
|
|
|
|
{
|
|
|
|
|
unsigned char *buffer;
|
|
|
|
|
size_t buflen;
|
|
|
|
|
size_t n, nn;
|
|
|
|
|
const unsigned char *p, *pp;
|
2011-02-04 12:57:53 +01:00
|
|
|
|
|
2021-04-29 15:26:36 +02:00
|
|
|
|
err = iso7816_read_record (app_get_slot (app), recno, 1, 0,
|
|
|
|
|
&buffer, &buflen);
|
2005-04-27 19:47:53 +00:00
|
|
|
|
if (gpg_err_code (err) == GPG_ERR_NOT_FOUND)
|
|
|
|
|
break; /* ready */
|
|
|
|
|
if (err)
|
|
|
|
|
{
|
2020-03-31 10:40:21 +02:00
|
|
|
|
log_error ("p15: error reading EF_keyD record: %s\n",
|
2005-04-27 19:47:53 +00:00
|
|
|
|
gpg_strerror (err));
|
|
|
|
|
return err;
|
|
|
|
|
}
|
2020-03-31 10:40:21 +02:00
|
|
|
|
if (opt.verbose)
|
|
|
|
|
{
|
|
|
|
|
log_info (buffer, buflen, "p15: keyD record: ");
|
2020-05-12 18:51:47 +02:00
|
|
|
|
log_printhex (buffer, buflen, "");
|
2020-03-31 10:40:21 +02:00
|
|
|
|
}
|
2005-04-27 19:47:53 +00:00
|
|
|
|
p = find_tlv (buffer, buflen, 0x83, &n);
|
|
|
|
|
if (p && n == 4 && ((p[2]<<8)|p[3]) == fid)
|
|
|
|
|
{
|
|
|
|
|
refdata = ((p[0]<<8)|p[1]);
|
|
|
|
|
/* Locate the SE DO and the there included sec env number. */
|
|
|
|
|
p = find_tlv (buffer, buflen, 0x7b, &n);
|
|
|
|
|
if (p && n)
|
|
|
|
|
{
|
|
|
|
|
pp = find_tlv (p, n, 0x80, &nn);
|
|
|
|
|
if (pp && nn == 1)
|
|
|
|
|
{
|
|
|
|
|
se_num = *pp;
|
|
|
|
|
xfree (buffer);
|
|
|
|
|
break; /* found. */
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
xfree (buffer);
|
|
|
|
|
}
|
|
|
|
|
if (se_num == -1)
|
|
|
|
|
{
|
2020-03-31 10:40:21 +02:00
|
|
|
|
log_error ("p15: CRT for keyfile %04hX not found\n", fid);
|
2005-04-27 19:47:53 +00:00
|
|
|
|
return gpg_error (GPG_ERR_NOT_FOUND);
|
|
|
|
|
}
|
2011-02-04 12:57:53 +01:00
|
|
|
|
|
|
|
|
|
|
2005-04-27 19:47:53 +00:00
|
|
|
|
/* Restore the security environment to SE_NUM if needed */
|
|
|
|
|
if (se_num)
|
|
|
|
|
{
|
2021-04-29 15:26:36 +02:00
|
|
|
|
err = iso7816_manage_security_env (app_get_slot (app),
|
|
|
|
|
0xf3, se_num, NULL, 0);
|
2005-04-27 19:47:53 +00:00
|
|
|
|
if (err)
|
|
|
|
|
{
|
2020-03-31 10:40:21 +02:00
|
|
|
|
log_error ("p15: restoring SE to %d failed: %s\n",
|
2005-04-27 19:47:53 +00:00
|
|
|
|
se_num, gpg_strerror (err));
|
|
|
|
|
return err;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Set the DST reference data. */
|
|
|
|
|
msebuf[0] = 0x83;
|
|
|
|
|
msebuf[1] = 0x03;
|
|
|
|
|
msebuf[2] = 0x80;
|
|
|
|
|
msebuf[3] = (refdata >> 8);
|
|
|
|
|
msebuf[4] = refdata;
|
2021-04-29 15:26:36 +02:00
|
|
|
|
err = iso7816_manage_security_env (app_get_slot (app), 0x41, 0xb6, msebuf, 5);
|
2005-04-27 19:47:53 +00:00
|
|
|
|
if (err)
|
|
|
|
|
{
|
2020-03-31 10:40:21 +02:00
|
|
|
|
log_error ("p15: setting SE to reference file %04hX failed: %s\n",
|
2005-04-27 19:47:53 +00:00
|
|
|
|
refdata, gpg_strerror (err));
|
|
|
|
|
return err;
|
|
|
|
|
}
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2005-04-27 12:09:21 +00:00
|
|
|
|
|
2020-04-01 11:58:08 +02:00
|
|
|
|
/* Prepare the verification of the PIN for the key PRKDF by checking
|
|
|
|
|
* the AODF and selecting the key file. KEYREF is used for error
|
2021-04-29 15:26:36 +02:00
|
|
|
|
* messages. AODF may be NULL if no verification needs to be done. */
|
2020-04-01 11:58:08 +02:00
|
|
|
|
static gpg_error_t
|
|
|
|
|
prepare_verify_pin (app_t app, const char *keyref,
|
|
|
|
|
prkdf_object_t prkdf, aodf_object_t aodf)
|
|
|
|
|
{
|
|
|
|
|
gpg_error_t err;
|
|
|
|
|
int i;
|
|
|
|
|
|
2021-04-29 15:26:36 +02:00
|
|
|
|
if (aodf)
|
2020-04-01 11:58:08 +02:00
|
|
|
|
{
|
2021-04-29 15:26:36 +02:00
|
|
|
|
if (opt.verbose)
|
|
|
|
|
{
|
|
|
|
|
log_info ("p15: using AODF %04hX id=", aodf->fid);
|
|
|
|
|
for (i=0; i < aodf->objidlen; i++)
|
|
|
|
|
log_printf ("%02X", aodf->objid[i]);
|
|
|
|
|
log_printf ("\n");
|
|
|
|
|
}
|
2020-04-01 11:58:08 +02:00
|
|
|
|
|
2021-04-29 15:26:36 +02:00
|
|
|
|
if (aodf->authid && opt.verbose)
|
|
|
|
|
log_info ("p15: PIN is controlled by another authentication token\n");
|
2020-04-01 11:58:08 +02:00
|
|
|
|
|
2021-04-29 15:26:36 +02:00
|
|
|
|
if (aodf->pinflags.integrity_protected
|
|
|
|
|
|| aodf->pinflags.confidentiality_protected)
|
|
|
|
|
{
|
|
|
|
|
log_error ("p15: PIN verification requires"
|
|
|
|
|
" unsupported protection method\n");
|
|
|
|
|
return gpg_error (GPG_ERR_BAD_PIN_METHOD);
|
|
|
|
|
}
|
|
|
|
|
if (!aodf->stored_length && aodf->pinflags.needs_padding)
|
|
|
|
|
{
|
|
|
|
|
log_error ("p15: PIN verification requires"
|
|
|
|
|
" padding but no length known\n");
|
|
|
|
|
return gpg_error (GPG_ERR_INV_CARD);
|
|
|
|
|
}
|
2020-04-01 11:58:08 +02:00
|
|
|
|
}
|
|
|
|
|
|
2020-04-07 16:00:11 +02:00
|
|
|
|
|
|
|
|
|
if (app->app_local->card_product == CARD_PRODUCT_DTRUST)
|
|
|
|
|
{
|
|
|
|
|
/* According to our protocol analysis we need to select a
|
|
|
|
|
* special AID here. Before that the master file needs to be
|
|
|
|
|
* selected. (RID A000000167 is assigned to IBM) */
|
|
|
|
|
static char const dtrust_aid[] =
|
|
|
|
|
{ 0xA0, 0x00, 0x00, 0x01, 0x67, 0x45, 0x53, 0x49, 0x47, 0x4E };
|
|
|
|
|
|
|
|
|
|
err = iso7816_select_mf (app_get_slot (app));
|
|
|
|
|
if (!err)
|
|
|
|
|
err = iso7816_select_application (app_get_slot (app),
|
|
|
|
|
dtrust_aid, sizeof dtrust_aid, 0);
|
|
|
|
|
if (err)
|
|
|
|
|
log_error ("p15: error selecting D-TRUST's AID for key %s: %s\n",
|
|
|
|
|
keyref, gpg_strerror (err));
|
|
|
|
|
}
|
2021-04-29 15:26:36 +02:00
|
|
|
|
else if (prkdf)
|
2020-04-07 16:00:11 +02:00
|
|
|
|
{
|
|
|
|
|
/* Standard case: Select the key file. Note that this may
|
|
|
|
|
* change the security environment thus we need to do it before
|
|
|
|
|
* PIN verification. */
|
|
|
|
|
err = select_ef_by_path (app, prkdf->path, prkdf->pathlen);
|
|
|
|
|
if (err)
|
|
|
|
|
log_error ("p15: error selecting file for key %s: %s\n",
|
|
|
|
|
keyref, gpg_strerror (err));
|
|
|
|
|
}
|
2021-04-29 15:26:36 +02:00
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
log_info ("p15: skipping EF selection for auth object '%s'\n", keyref);
|
|
|
|
|
err = 0;
|
|
|
|
|
}
|
2020-04-01 11:58:08 +02:00
|
|
|
|
|
|
|
|
|
return err;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2020-04-07 20:38:15 +02:00
|
|
|
|
static int
|
|
|
|
|
any_control_or_space (const char *string)
|
|
|
|
|
{
|
|
|
|
|
const unsigned char *s;
|
|
|
|
|
|
|
|
|
|
for (s = string; *string; string++)
|
|
|
|
|
if (*s <= 0x20 || *s >= 0x7f)
|
|
|
|
|
return 1;
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2020-04-08 07:54:01 +02:00
|
|
|
|
/* Return a malloced serial number to be shown to the user. PRKDF is
|
|
|
|
|
* used to get it from a certificate; PRKDF may be NULL. */
|
2020-04-07 20:38:15 +02:00
|
|
|
|
static char *
|
2020-04-08 07:54:01 +02:00
|
|
|
|
get_dispserialno (app_t app, prkdf_object_t prkdf)
|
2020-04-07 20:38:15 +02:00
|
|
|
|
{
|
2020-04-08 07:54:01 +02:00
|
|
|
|
char *serial;
|
2021-04-29 15:26:36 +02:00
|
|
|
|
size_t n;
|
2020-04-07 20:38:15 +02:00
|
|
|
|
|
|
|
|
|
/* We prefer the SerialNumber RDN from the Subject-DN but we don't
|
|
|
|
|
* use it if it features a percent sign (special character in pin
|
|
|
|
|
* prompts) or has any control character. */
|
2021-04-29 15:26:36 +02:00
|
|
|
|
if (app->app_local->card_product == CARD_PRODUCT_RSCS)
|
|
|
|
|
{
|
|
|
|
|
/* We use only the right 8 hex digits. */
|
|
|
|
|
serial = app_get_serialno (app);
|
|
|
|
|
if (serial && (n=strlen (serial)) > 8)
|
|
|
|
|
memmove (serial, serial + n - 8, 9);
|
|
|
|
|
}
|
|
|
|
|
else if (prkdf && prkdf->serial_number && *prkdf->serial_number
|
2020-04-08 07:54:01 +02:00
|
|
|
|
&& !strchr (prkdf->serial_number, '%')
|
|
|
|
|
&& !any_control_or_space (prkdf->serial_number))
|
2020-04-07 20:38:15 +02:00
|
|
|
|
{
|
2020-04-08 07:54:01 +02:00
|
|
|
|
serial = xtrystrdup (prkdf->serial_number);
|
2020-04-07 20:38:15 +02:00
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
serial = app_get_serialno (app);
|
|
|
|
|
}
|
2020-04-08 07:54:01 +02:00
|
|
|
|
return serial;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2021-04-29 15:26:36 +02:00
|
|
|
|
/* Return an allocated string to be used as prompt. PRKDF may be
|
|
|
|
|
* NULL. Returns NULL on malloc error. */
|
2020-04-08 07:54:01 +02:00
|
|
|
|
static char *
|
|
|
|
|
make_pin_prompt (app_t app, int remaining, const char *firstline,
|
|
|
|
|
prkdf_object_t prkdf)
|
|
|
|
|
{
|
|
|
|
|
char *serial, *tmpbuf, *result;
|
2022-04-13 13:06:27 +02:00
|
|
|
|
const char *holder = NULL;
|
2020-04-08 07:54:01 +02:00
|
|
|
|
|
|
|
|
|
serial = get_dispserialno (app, prkdf);
|
2020-04-07 20:38:15 +02:00
|
|
|
|
|
2022-04-13 13:06:27 +02:00
|
|
|
|
if (app->app_local->card_product == CARD_PRODUCT_GENUA)
|
|
|
|
|
{
|
|
|
|
|
/* The label of the first non SO-PIN is used for the holder. */
|
|
|
|
|
aodf_object_t aodf;
|
|
|
|
|
|
|
|
|
|
for (aodf = app->app_local->auth_object_info; aodf; aodf = aodf->next)
|
|
|
|
|
if (aodf->auth_type == AUTH_TYPE_PIN
|
|
|
|
|
&& !aodf->pinflags.so_pin
|
|
|
|
|
&& aodf->label)
|
|
|
|
|
{
|
|
|
|
|
holder = aodf->label;
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (holder)
|
|
|
|
|
;
|
|
|
|
|
else if (prkdf && prkdf->common_name)
|
2021-04-29 15:26:36 +02:00
|
|
|
|
holder = prkdf->common_name;
|
|
|
|
|
else if (app->app_local->token_label)
|
|
|
|
|
holder = app->app_local->token_label;
|
|
|
|
|
else
|
|
|
|
|
holder = "";
|
|
|
|
|
|
2020-04-07 20:38:15 +02:00
|
|
|
|
/* TRANSLATORS: Put a \x1f right before a colon. This can be
|
|
|
|
|
* used by pinentry to nicely align the names and values. Keep
|
|
|
|
|
* the %s at the start and end of the string. */
|
|
|
|
|
result = xtryasprintf (_("%s"
|
|
|
|
|
"Number\x1f: %s%%0A"
|
|
|
|
|
"Holder\x1f: %s"
|
|
|
|
|
"%s"),
|
|
|
|
|
"\x1e",
|
2020-04-08 07:54:01 +02:00
|
|
|
|
serial,
|
2021-04-29 15:26:36 +02:00
|
|
|
|
holder,
|
2020-04-07 20:38:15 +02:00
|
|
|
|
"");
|
2020-04-08 07:54:01 +02:00
|
|
|
|
xfree (serial);
|
2020-04-07 20:38:15 +02:00
|
|
|
|
if (!result)
|
|
|
|
|
return NULL; /* Out of core. */
|
|
|
|
|
|
|
|
|
|
/* Append a "remaining attempts" info if needed. */
|
|
|
|
|
if (remaining != -1 && remaining < 3)
|
|
|
|
|
{
|
|
|
|
|
char *rembuf;
|
|
|
|
|
|
|
|
|
|
/* TRANSLATORS: This is the number of remaining attempts to
|
|
|
|
|
* enter a PIN. Use %%0A (double-percent,0A) for a linefeed. */
|
|
|
|
|
rembuf = xtryasprintf (_("Remaining attempts: %d"), remaining);
|
|
|
|
|
if (rembuf)
|
|
|
|
|
{
|
|
|
|
|
tmpbuf = strconcat (firstline, "%0A%0A", result,
|
|
|
|
|
"%0A%0A", rembuf, NULL);
|
|
|
|
|
xfree (rembuf);
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
tmpbuf = NULL;
|
|
|
|
|
xfree (result);
|
|
|
|
|
result = tmpbuf;
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
tmpbuf = strconcat (firstline, "%0A%0A", result, NULL);
|
|
|
|
|
xfree (result);
|
|
|
|
|
result = tmpbuf;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return result;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2020-04-01 11:58:08 +02:00
|
|
|
|
/* Given the private key object PRKDF and its authentication object
|
2021-04-29 15:26:36 +02:00
|
|
|
|
* AODF ask for the PIN and verify that PIN. If AODF is NULL, no
|
|
|
|
|
* authentication is done. */
|
2020-04-01 11:58:08 +02:00
|
|
|
|
static gpg_error_t
|
|
|
|
|
verify_pin (app_t app,
|
|
|
|
|
gpg_error_t (*pincb)(void*, const char *, char **), void *pincb_arg,
|
|
|
|
|
prkdf_object_t prkdf, aodf_object_t aodf)
|
|
|
|
|
{
|
|
|
|
|
gpg_error_t err;
|
|
|
|
|
char *pinvalue;
|
|
|
|
|
size_t pinvaluelen;
|
2020-04-07 20:38:15 +02:00
|
|
|
|
const char *label;
|
2020-04-01 11:58:08 +02:00
|
|
|
|
const char *errstr;
|
|
|
|
|
const char *s;
|
2020-04-07 20:38:15 +02:00
|
|
|
|
int remaining;
|
|
|
|
|
int pin_reference;
|
2021-04-29 15:26:36 +02:00
|
|
|
|
int verified = 0;
|
2020-04-01 11:58:08 +02:00
|
|
|
|
int i;
|
|
|
|
|
|
2020-04-07 20:38:15 +02:00
|
|
|
|
if (!aodf)
|
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
|
|
pin_reference = aodf->pin_reference_valid? aodf->pin_reference : 0;
|
|
|
|
|
|
2021-04-29 15:26:36 +02:00
|
|
|
|
if (IS_CARDOS_5 (app))
|
2020-04-07 20:38:15 +02:00
|
|
|
|
{
|
|
|
|
|
/* We know that this card supports a verify status check. Note
|
|
|
|
|
* that in contrast to PIV cards ISO7816_VERIFY_NOT_NEEDED is
|
2021-04-29 15:26:36 +02:00
|
|
|
|
* not supported. Noet that we don't use the pin_verified cache
|
|
|
|
|
* status because that is not as reliable than to ask the card
|
|
|
|
|
* about its state. */
|
|
|
|
|
if (prkdf) /* Clear the cache which we don't use. */
|
|
|
|
|
prkdf->pin_verified = 0;
|
|
|
|
|
|
2020-04-07 20:38:15 +02:00
|
|
|
|
remaining = iso7816_verify_status (app_get_slot (app), pin_reference);
|
2021-04-29 15:26:36 +02:00
|
|
|
|
if (remaining == ISO7816_VERIFY_NOT_NEEDED)
|
2020-04-07 20:38:15 +02:00
|
|
|
|
{
|
2021-04-29 15:26:36 +02:00
|
|
|
|
verified = 1;
|
|
|
|
|
remaining = -1;
|
2020-04-07 20:38:15 +02:00
|
|
|
|
}
|
2021-04-29 15:26:36 +02:00
|
|
|
|
else if (remaining < 0)
|
|
|
|
|
remaining = -1; /* We don't care about the concrete error. */
|
|
|
|
|
else if (remaining < 3)
|
|
|
|
|
log_info ("p15: PIN has %d attempts left\n", remaining);
|
2020-04-07 20:38:15 +02:00
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
remaining = -1; /* Unknown. */
|
|
|
|
|
|
|
|
|
|
/* Check whether we already verified it. */
|
2021-04-29 15:26:36 +02:00
|
|
|
|
if (prkdf && (prkdf->pin_verified || verified))
|
2020-04-01 20:31:21 +02:00
|
|
|
|
return 0; /* Already done. */
|
|
|
|
|
|
2021-04-29 15:26:36 +02:00
|
|
|
|
if (prkdf
|
|
|
|
|
&& prkdf->usageflags.non_repudiation
|
2020-04-07 20:38:15 +02:00
|
|
|
|
&& (app->app_local->card_type == CARD_TYPE_BELPIC
|
|
|
|
|
|| app->app_local->card_product == CARD_PRODUCT_DTRUST))
|
|
|
|
|
label = _("||Please enter the PIN for the key to create "
|
|
|
|
|
"qualified signatures.");
|
2021-04-29 15:26:36 +02:00
|
|
|
|
else if (aodf->pinflags.so_pin)
|
|
|
|
|
label = _("|A|Please enter the Admin PIN");
|
|
|
|
|
else if (aodf->pinflags.unblocking_pin)
|
|
|
|
|
label = _("|P|Please enter the PIN Unblocking Code (PUK) "
|
|
|
|
|
"for the standard keys.");
|
2020-04-01 11:58:08 +02:00
|
|
|
|
else
|
2020-04-07 20:38:15 +02:00
|
|
|
|
label = _("||Please enter the PIN for the standard keys.");
|
|
|
|
|
|
|
|
|
|
{
|
2020-04-08 07:54:01 +02:00
|
|
|
|
char *prompt = make_pin_prompt (app, remaining, label, prkdf);
|
2020-04-07 20:38:15 +02:00
|
|
|
|
if (!prompt)
|
|
|
|
|
err = gpg_error_from_syserror ();
|
|
|
|
|
else
|
|
|
|
|
err = pincb (pincb_arg, prompt, &pinvalue);
|
|
|
|
|
xfree (prompt);
|
|
|
|
|
}
|
2020-04-01 11:58:08 +02:00
|
|
|
|
if (err)
|
|
|
|
|
{
|
2020-04-07 20:38:15 +02:00
|
|
|
|
log_info ("p15: PIN callback returned error: %s\n", gpg_strerror (err));
|
2020-04-01 11:58:08 +02:00
|
|
|
|
return err;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* We might need to cope with UTF8 things here. Not sure how
|
|
|
|
|
min_length etc. are exactly defined, for now we take them as
|
|
|
|
|
a plain octet count. */
|
|
|
|
|
if (strlen (pinvalue) < aodf->min_length)
|
|
|
|
|
{
|
|
|
|
|
log_error ("p15: PIN is too short; minimum length is %lu\n",
|
|
|
|
|
aodf->min_length);
|
|
|
|
|
err = gpg_error (GPG_ERR_BAD_PIN);
|
|
|
|
|
}
|
|
|
|
|
else if (aodf->stored_length && strlen (pinvalue) > aodf->stored_length)
|
|
|
|
|
{
|
|
|
|
|
/* This would otherwise truncate the PIN silently. */
|
|
|
|
|
log_error ("p15: PIN is too large; maximum length is %lu\n",
|
|
|
|
|
aodf->stored_length);
|
|
|
|
|
err = gpg_error (GPG_ERR_BAD_PIN);
|
|
|
|
|
}
|
|
|
|
|
else if (aodf->max_length_valid && strlen (pinvalue) > aodf->max_length)
|
|
|
|
|
{
|
|
|
|
|
log_error ("p15: PIN is too large; maximum length is %lu\n",
|
|
|
|
|
aodf->max_length);
|
|
|
|
|
err = gpg_error (GPG_ERR_BAD_PIN);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (err)
|
|
|
|
|
{
|
|
|
|
|
xfree (pinvalue);
|
|
|
|
|
return err;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
errstr = NULL;
|
|
|
|
|
err = 0;
|
|
|
|
|
switch (aodf->pintype)
|
|
|
|
|
{
|
|
|
|
|
case PIN_TYPE_BCD:
|
|
|
|
|
case PIN_TYPE_ASCII_NUMERIC:
|
|
|
|
|
for (s=pinvalue; digitp (s); s++)
|
|
|
|
|
;
|
|
|
|
|
if (*s)
|
|
|
|
|
{
|
|
|
|
|
errstr = "Non-numeric digits found in PIN";
|
|
|
|
|
err = gpg_error (GPG_ERR_BAD_PIN);
|
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
case PIN_TYPE_UTF8:
|
|
|
|
|
break;
|
|
|
|
|
case PIN_TYPE_HALF_NIBBLE_BCD:
|
|
|
|
|
errstr = "PIN type Half-Nibble-BCD is not supported";
|
|
|
|
|
break;
|
|
|
|
|
case PIN_TYPE_ISO9564_1:
|
|
|
|
|
errstr = "PIN type ISO9564-1 is not supported";
|
|
|
|
|
break;
|
|
|
|
|
default:
|
|
|
|
|
errstr = "Unknown PIN type";
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
if (errstr)
|
|
|
|
|
{
|
|
|
|
|
log_error ("p15: can't verify PIN: %s\n", errstr);
|
|
|
|
|
xfree (pinvalue);
|
|
|
|
|
return err? err : gpg_error (GPG_ERR_BAD_PIN_METHOD);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
if (aodf->pintype == PIN_TYPE_BCD )
|
|
|
|
|
{
|
|
|
|
|
char *paddedpin;
|
|
|
|
|
int ndigits;
|
|
|
|
|
|
|
|
|
|
for (ndigits=0, s=pinvalue; *s; ndigits++, s++)
|
|
|
|
|
;
|
|
|
|
|
paddedpin = xtrymalloc (aodf->stored_length+1);
|
|
|
|
|
if (!paddedpin)
|
|
|
|
|
{
|
|
|
|
|
err = gpg_error_from_syserror ();
|
|
|
|
|
xfree (pinvalue);
|
|
|
|
|
return err;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
i = 0;
|
|
|
|
|
paddedpin[i++] = 0x20 | (ndigits & 0x0f);
|
|
|
|
|
for (s=pinvalue; i < aodf->stored_length && *s && s[1]; s = s+2 )
|
|
|
|
|
paddedpin[i++] = (((*s - '0') << 4) | ((s[1] - '0') & 0x0f));
|
|
|
|
|
if (i < aodf->stored_length && *s)
|
|
|
|
|
paddedpin[i++] = (((*s - '0') << 4)
|
|
|
|
|
|((aodf->pad_char_valid?aodf->pad_char:0)&0x0f));
|
|
|
|
|
|
|
|
|
|
if (aodf->pinflags.needs_padding)
|
|
|
|
|
{
|
|
|
|
|
while (i < aodf->stored_length)
|
|
|
|
|
paddedpin[i++] = aodf->pad_char_valid? aodf->pad_char : 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
xfree (pinvalue);
|
|
|
|
|
pinvalue = paddedpin;
|
|
|
|
|
pinvaluelen = i;
|
|
|
|
|
}
|
|
|
|
|
else if (aodf->pinflags.needs_padding)
|
|
|
|
|
{
|
|
|
|
|
char *paddedpin;
|
|
|
|
|
|
|
|
|
|
paddedpin = xtrymalloc (aodf->stored_length+1);
|
|
|
|
|
if (!paddedpin)
|
|
|
|
|
{
|
|
|
|
|
err = gpg_error_from_syserror ();
|
|
|
|
|
xfree (pinvalue);
|
|
|
|
|
return err;
|
|
|
|
|
}
|
|
|
|
|
for (i=0, s=pinvalue; i < aodf->stored_length && *s; i++, s++)
|
|
|
|
|
paddedpin[i] = *s;
|
|
|
|
|
/* Not sure what padding char to use if none has been set.
|
|
|
|
|
For now we use 0x00; maybe a space would be better. */
|
|
|
|
|
for (; i < aodf->stored_length; i++)
|
|
|
|
|
paddedpin[i] = aodf->pad_char_valid? aodf->pad_char : 0;
|
|
|
|
|
paddedpin[i] = 0;
|
|
|
|
|
pinvaluelen = i;
|
|
|
|
|
xfree (pinvalue);
|
|
|
|
|
pinvalue = paddedpin;
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
pinvaluelen = strlen (pinvalue);
|
|
|
|
|
|
|
|
|
|
/* log_printhex (pinvalue, pinvaluelen, */
|
2020-04-07 20:38:15 +02:00
|
|
|
|
/* "about to verify with ref %lu pin:", pin_reference); */
|
|
|
|
|
err = iso7816_verify (app_get_slot (app), pin_reference,
|
2020-04-01 11:58:08 +02:00
|
|
|
|
pinvalue, pinvaluelen);
|
|
|
|
|
xfree (pinvalue);
|
|
|
|
|
if (err)
|
|
|
|
|
{
|
|
|
|
|
log_error ("p15: PIN verification failed: %s\n", gpg_strerror (err));
|
|
|
|
|
return err;
|
|
|
|
|
}
|
|
|
|
|
if (opt.verbose)
|
|
|
|
|
log_info ("p15: PIN verification succeeded\n");
|
2021-04-29 15:26:36 +02:00
|
|
|
|
if (prkdf)
|
|
|
|
|
prkdf->pin_verified = 1;
|
2020-04-01 11:58:08 +02:00
|
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
2011-02-04 12:57:53 +01:00
|
|
|
|
/* Handler for the PKSIGN command.
|
2005-04-27 12:09:21 +00:00
|
|
|
|
|
|
|
|
|
Create the signature and return the allocated result in OUTDATA.
|
|
|
|
|
If a PIN is required, the PINCB will be used to ask for the PIN;
|
|
|
|
|
that callback should return the PIN in an allocated buffer and
|
|
|
|
|
store that as the 3rd argument. */
|
2011-02-04 12:57:53 +01:00
|
|
|
|
static gpg_error_t
|
2021-02-19 09:50:22 +01:00
|
|
|
|
do_sign (app_t app, ctrl_t ctrl, const char *keyidstr, int hashalgo,
|
2005-05-18 10:48:06 +00:00
|
|
|
|
gpg_error_t (*pincb)(void*, const char *, char **),
|
2005-04-27 12:09:21 +00:00
|
|
|
|
void *pincb_arg,
|
|
|
|
|
const void *indata, size_t indatalen,
|
|
|
|
|
unsigned char **outdata, size_t *outdatalen )
|
2004-12-02 07:48:09 +00:00
|
|
|
|
{
|
|
|
|
|
gpg_error_t err;
|
2005-04-27 12:09:21 +00:00
|
|
|
|
prkdf_object_t prkdf; /* The private key object. */
|
|
|
|
|
aodf_object_t aodf; /* The associated authentication object. */
|
2005-09-06 18:42:13 +00:00
|
|
|
|
int mse_done = 0; /* Set to true if the MSE has been done. */
|
2021-04-29 15:26:36 +02:00
|
|
|
|
unsigned int digestlen; /* Length of the hash. */
|
2020-03-31 19:55:15 +02:00
|
|
|
|
int exmode, le_value;
|
2021-04-29 15:26:36 +02:00
|
|
|
|
unsigned char oidbuf[64];
|
|
|
|
|
size_t oidbuflen;
|
|
|
|
|
size_t n;
|
|
|
|
|
unsigned char *indata_buffer = NULL; /* Malloced helper. */
|
2005-04-27 12:09:21 +00:00
|
|
|
|
|
2021-02-19 09:50:22 +01:00
|
|
|
|
(void)ctrl;
|
2020-03-31 19:55:15 +02:00
|
|
|
|
|
2021-04-29 15:26:36 +02:00
|
|
|
|
if (!keyidstr || !*keyidstr || !indatalen)
|
2005-04-27 12:09:21 +00:00
|
|
|
|
return gpg_error (GPG_ERR_INV_VALUE);
|
|
|
|
|
|
|
|
|
|
err = prkdf_object_from_keyidstr (app, keyidstr, &prkdf);
|
|
|
|
|
if (err)
|
|
|
|
|
return err;
|
2021-04-29 15:26:36 +02:00
|
|
|
|
if (!(prkdf->usageflags.sign
|
|
|
|
|
|| prkdf->usageflags.sign_recover
|
|
|
|
|
|| prkdf->usageflags.non_repudiation
|
|
|
|
|
|| prkdf->gpgusage.cert
|
|
|
|
|
|| prkdf->gpgusage.sign
|
|
|
|
|
|| prkdf->gpgusage.auth ))
|
2005-04-27 19:47:53 +00:00
|
|
|
|
{
|
2020-03-31 10:40:21 +02:00
|
|
|
|
log_error ("p15: key %s may not be used for signing\n", keyidstr);
|
2005-04-27 19:47:53 +00:00
|
|
|
|
return gpg_error (GPG_ERR_WRONG_KEY_USAGE);
|
|
|
|
|
}
|
|
|
|
|
|
2005-04-27 12:09:21 +00:00
|
|
|
|
if (!prkdf->authid)
|
|
|
|
|
{
|
2020-03-31 10:40:21 +02:00
|
|
|
|
log_error ("p15: no authentication object defined for %s\n", keyidstr);
|
2005-04-27 12:09:21 +00:00
|
|
|
|
/* fixme: we might want to go ahead and do without PIN
|
|
|
|
|
verification. */
|
|
|
|
|
return gpg_error (GPG_ERR_UNSUPPORTED_OPERATION);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Find the authentication object to this private key object. */
|
|
|
|
|
for (aodf = app->app_local->auth_object_info; aodf; aodf = aodf->next)
|
2011-02-04 12:57:53 +01:00
|
|
|
|
if (aodf->objidlen == prkdf->authidlen
|
2005-04-27 12:09:21 +00:00
|
|
|
|
&& !memcmp (aodf->objid, prkdf->authid, prkdf->authidlen))
|
|
|
|
|
break;
|
|
|
|
|
if (!aodf)
|
2021-04-29 15:26:36 +02:00
|
|
|
|
log_info ("p15: no authentication for %s needed\n", keyidstr);
|
2020-03-31 19:55:15 +02:00
|
|
|
|
|
|
|
|
|
/* We need some more info about the key - get the keygrip to
|
|
|
|
|
* populate these fields. */
|
|
|
|
|
err = keygrip_from_prkdf (app, prkdf);
|
|
|
|
|
if (err)
|
|
|
|
|
{
|
|
|
|
|
log_error ("p15: keygrip_from_prkdf failed: %s\n", gpg_strerror (err));
|
|
|
|
|
return err;
|
|
|
|
|
}
|
|
|
|
|
|
2021-04-29 15:26:36 +02:00
|
|
|
|
|
|
|
|
|
digestlen = gcry_md_get_algo_dlen (hashalgo);
|
|
|
|
|
|
|
|
|
|
/* We handle ECC separately from RSA so that we do not need to touch
|
|
|
|
|
* working code. In particular we prepare the input data before the
|
|
|
|
|
* verify and a possible MSE. */
|
|
|
|
|
if (prkdf->is_ecc)
|
|
|
|
|
{
|
|
|
|
|
if (digestlen != 32 && digestlen != 48 && digestlen != 64)
|
|
|
|
|
{
|
|
|
|
|
log_error ("p15: ECC signing not possible: dlen=%u\n", digestlen);
|
|
|
|
|
err = gpg_error (GPG_ERR_DIGEST_ALGO);
|
|
|
|
|
goto leave;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (indatalen == digestlen)
|
|
|
|
|
; /* Already prepared. */
|
|
|
|
|
else if (indatalen > digestlen)
|
|
|
|
|
{
|
|
|
|
|
/* Assume a PKCS#1 prefix and remove it. */
|
|
|
|
|
oidbuflen = sizeof oidbuf;
|
|
|
|
|
err = gcry_md_get_asnoid (hashalgo, &oidbuf, &oidbuflen);
|
|
|
|
|
if (err)
|
|
|
|
|
{
|
|
|
|
|
log_error ("p15: no OID for hash algo %d\n", hashalgo);
|
|
|
|
|
err = gpg_error (GPG_ERR_INTERNAL);
|
|
|
|
|
goto leave;
|
|
|
|
|
}
|
|
|
|
|
if (indatalen != oidbuflen + digestlen
|
|
|
|
|
|| memcmp (indata, oidbuf, oidbuflen))
|
|
|
|
|
{
|
|
|
|
|
log_error ("p15: input data too long for ECC: len=%zu\n",
|
|
|
|
|
indatalen);
|
|
|
|
|
err = gpg_error (GPG_ERR_INV_VALUE);
|
|
|
|
|
goto leave;
|
|
|
|
|
}
|
|
|
|
|
indata = (const char*)indata + oidbuflen;
|
|
|
|
|
indatalen -= oidbuflen;
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
log_error ("p15: input data too short for ECC: len=%zu\n",
|
|
|
|
|
indatalen);
|
|
|
|
|
err = gpg_error (GPG_ERR_INV_VALUE);
|
|
|
|
|
goto leave;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
else /* Prepare RSA input. */
|
|
|
|
|
{
|
|
|
|
|
unsigned int framelen;
|
|
|
|
|
unsigned char *frame;
|
|
|
|
|
int i;
|
|
|
|
|
|
|
|
|
|
framelen = (prkdf->keynbits+7) / 8;
|
|
|
|
|
if (!framelen)
|
|
|
|
|
{
|
|
|
|
|
log_error ("p15: key length unknown"
|
|
|
|
|
" - can't prepare PKCS#v1.5 frame\n");
|
|
|
|
|
err = gpg_error (GPG_ERR_INV_VALUE);
|
|
|
|
|
goto leave;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
oidbuflen = sizeof oidbuf;
|
|
|
|
|
if (!hashalgo)
|
|
|
|
|
{
|
|
|
|
|
/* We assume that indata already has the required
|
|
|
|
|
* digestinfo; thus merely prepend the padding below. */
|
|
|
|
|
}
|
|
|
|
|
else if ((err = gcry_md_get_asnoid (hashalgo, &oidbuf, &oidbuflen)))
|
|
|
|
|
{
|
|
|
|
|
log_debug ("p15: no OID for hash algo %d\n", hashalgo);
|
|
|
|
|
goto leave;
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
if (indatalen == digestlen)
|
|
|
|
|
{
|
|
|
|
|
/* Plain hash in INDATA; prepend the digestinfo. */
|
|
|
|
|
indata_buffer = xtrymalloc (oidbuflen + indatalen);
|
|
|
|
|
if (!indata_buffer)
|
|
|
|
|
{
|
|
|
|
|
err = gpg_error_from_syserror ();
|
|
|
|
|
goto leave;
|
|
|
|
|
}
|
|
|
|
|
memcpy (indata_buffer, oidbuf, oidbuflen);
|
|
|
|
|
memcpy (indata_buffer+oidbuflen, indata, indatalen);
|
|
|
|
|
indata = indata_buffer;
|
|
|
|
|
indatalen = oidbuflen + indatalen;
|
|
|
|
|
}
|
|
|
|
|
else if (indatalen == oidbuflen + digestlen
|
|
|
|
|
&& !memcmp (indata, oidbuf, oidbuflen))
|
|
|
|
|
; /* We already got the correct prefix. */
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
err = gpg_error (GPG_ERR_INV_VALUE);
|
|
|
|
|
log_error ("p15: bad input for signing with RSA and hash %d\n",
|
|
|
|
|
hashalgo);
|
|
|
|
|
goto leave;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
/* Now prepend the pkcs#v1.5 padding. We require at least 8
|
|
|
|
|
* byte of padding and 3 extra bytes for the prefix and the
|
|
|
|
|
* delimiting nul. */
|
|
|
|
|
if (!indatalen || indatalen + 8 + 4 > framelen)
|
|
|
|
|
{
|
|
|
|
|
err = gpg_error (GPG_ERR_INV_VALUE);
|
|
|
|
|
log_error ("p15: input does not fit into a %u bit PKCS#v1.5 frame\n",
|
|
|
|
|
8*framelen);
|
|
|
|
|
goto leave;
|
|
|
|
|
}
|
|
|
|
|
frame = xtrymalloc (framelen);
|
|
|
|
|
if (!frame)
|
|
|
|
|
{
|
|
|
|
|
err = gpg_error_from_syserror ();
|
|
|
|
|
goto leave;
|
|
|
|
|
}
|
|
|
|
|
if (app->app_local->card_type == CARD_TYPE_BELPIC)
|
|
|
|
|
{
|
|
|
|
|
/* This card wants only the plain hash w/o any prefix. */
|
|
|
|
|
/* FIXME: We may want to remove this code because it is unlikely
|
|
|
|
|
* that such cards are still in use. */
|
|
|
|
|
memcpy (frame, indata, indatalen);
|
|
|
|
|
framelen = indatalen;
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
n = 0;
|
|
|
|
|
frame[n++] = 0;
|
|
|
|
|
frame[n++] = 1; /* Block type. */
|
|
|
|
|
i = framelen - indatalen - 3 ;
|
|
|
|
|
memset (frame+n, 0xff, i);
|
|
|
|
|
n += i;
|
|
|
|
|
frame[n++] = 0; /* Delimiter. */
|
|
|
|
|
memcpy (frame+n, indata, indatalen);
|
|
|
|
|
n += indatalen;
|
|
|
|
|
log_assert (n == framelen);
|
|
|
|
|
}
|
|
|
|
|
/* And now put it into the indata_buffer. */
|
|
|
|
|
xfree (indata_buffer);
|
|
|
|
|
indata_buffer = frame;
|
|
|
|
|
indata = indata_buffer;
|
|
|
|
|
indatalen = framelen;
|
|
|
|
|
}
|
|
|
|
|
|
2020-04-01 11:58:08 +02:00
|
|
|
|
/* Prepare PIN verification. This is split so that we can do
|
|
|
|
|
* MSE operation for some task after having selected the key file but
|
|
|
|
|
* before sending the verify APDU. */
|
|
|
|
|
err = prepare_verify_pin (app, keyidstr, prkdf, aodf);
|
2005-04-27 19:47:53 +00:00
|
|
|
|
if (err)
|
2020-04-01 11:58:08 +02:00
|
|
|
|
return err;
|
2005-09-06 18:42:13 +00:00
|
|
|
|
|
|
|
|
|
/* Due to the fact that the non-repudiation signature on a BELPIC
|
2006-10-23 14:02:13 +00:00
|
|
|
|
card requires a verify immediately before the DSO we set the
|
More cleanup of "allow to".
* README, agent/command.c, agent/keyformat.txt, common/i18n.c,
common/iobuf.c, common/keyserver.h, dirmngr/cdblib.c,
dirmngr/ldap-wrapper.c, doc/DETAILS, doc/TRANSLATE,
doc/announce-2.1.txt, doc/gpg.texi, doc/gpgsm.texi,
doc/scdaemon.texi, doc/tools.texi, doc/whats-new-in-2.1.txt,
g10/export.c, g10/getkey.c, g10/import.c, g10/keyedit.c, m4/ksba.m4,
m4/libgcrypt.m4, m4/ntbtls.m4, po/ca.po, po/cs.po, po/da.po,
po/de.po, po/el.po, po/eo.po, po/es.po, po/et.po, po/fi.po,
po/fr.po, po/gl.po, po/hu.po, po/id.po, po/it.po, po/ja.po,
po/nb.po, po/pl.po, po/pt.po, po/ro.po, po/ru.po, po/sk.po,
po/sv.po, po/tr.po, po/uk.po, po/zh_CN.po, po/zh_TW.po,
scd/app-p15.c, scd/ccid-driver.c, scd/command.c, sm/gpgsm.c,
sm/sign.c, tools/gpgconf-comp.c, tools/gpgtar.h: replace "Allow to"
with clearer text.
In standard English, the normal construction is "${XXX} allows ${YYY}
to" -- that is, the subject (${XXX}) of the sentence is allowing the
object (${YYY}) to do something. When the object is missing, the
phrasing sounds awkward, even if the object is implied by context.
There's almost always a better construction that isn't as awkward.
These changes should make the language a bit clearer.
Signed-off-by: Daniel Kahn Gillmor <dkg@fifthhorseman.net>
2016-08-01 22:19:17 -04:00
|
|
|
|
MSE before we do the verification. Other cards might also allow
|
|
|
|
|
this but I don't want to break anything, thus we do it only
|
2021-04-29 15:26:36 +02:00
|
|
|
|
for the BELPIC card here.
|
|
|
|
|
FIXME: see comment above about these cards. */
|
2005-09-06 18:42:13 +00:00
|
|
|
|
if (app->app_local->card_type == CARD_TYPE_BELPIC)
|
|
|
|
|
{
|
|
|
|
|
unsigned char mse[5];
|
2011-02-04 12:57:53 +01:00
|
|
|
|
|
2005-09-06 18:42:13 +00:00
|
|
|
|
mse[0] = 4; /* Length of the template. */
|
|
|
|
|
mse[1] = 0x80; /* Algorithm reference tag. */
|
2008-09-29 15:02:55 +00:00
|
|
|
|
if (hashalgo == MD_USER_TLS_MD5SHA1)
|
2006-10-24 14:45:34 +00:00
|
|
|
|
mse[2] = 0x01; /* Let card do pkcs#1 0xFF padding. */
|
|
|
|
|
else
|
|
|
|
|
mse[2] = 0x02; /* RSASSA-PKCS1-v1.5 using SHA1. */
|
2005-09-06 18:42:13 +00:00
|
|
|
|
mse[3] = 0x84; /* Private key reference tag. */
|
|
|
|
|
mse[4] = prkdf->key_reference_valid? prkdf->key_reference : 0x82;
|
|
|
|
|
|
2021-04-29 15:26:36 +02:00
|
|
|
|
err = iso7816_manage_security_env (app_get_slot (app),
|
2005-09-06 18:42:13 +00:00
|
|
|
|
0x41, 0xB6,
|
|
|
|
|
mse, sizeof mse);
|
|
|
|
|
mse_done = 1;
|
|
|
|
|
}
|
|
|
|
|
if (err)
|
|
|
|
|
{
|
2020-03-31 10:40:21 +02:00
|
|
|
|
log_error ("p15: MSE failed: %s\n", gpg_strerror (err));
|
2021-04-13 14:02:18 +02:00
|
|
|
|
goto leave;
|
2005-09-06 18:42:13 +00:00
|
|
|
|
}
|
|
|
|
|
|
2020-04-01 11:58:08 +02:00
|
|
|
|
/* Now that we have all the information available run the actual PIN
|
|
|
|
|
* verification.*/
|
|
|
|
|
err = verify_pin (app, pincb, pincb_arg, prkdf, aodf);
|
|
|
|
|
if (err)
|
|
|
|
|
return err;
|
2005-09-06 18:42:13 +00:00
|
|
|
|
|
2020-04-01 14:07:56 +02:00
|
|
|
|
/* Manage security environment needs to be tweaked for certain cards. */
|
2005-09-06 18:42:13 +00:00
|
|
|
|
if (mse_done)
|
|
|
|
|
err = 0;
|
|
|
|
|
else if (app->app_local->card_type == CARD_TYPE_TCOS)
|
2005-04-27 12:09:21 +00:00
|
|
|
|
{
|
|
|
|
|
/* TCOS creates signatures always using the local key 0. MSE
|
|
|
|
|
may not be used. */
|
|
|
|
|
}
|
|
|
|
|
else if (app->app_local->card_type == CARD_TYPE_MICARDO)
|
|
|
|
|
{
|
2005-04-27 19:47:53 +00:00
|
|
|
|
if (!prkdf->pathlen)
|
|
|
|
|
err = gpg_error (GPG_ERR_BUG);
|
|
|
|
|
else
|
|
|
|
|
err = micardo_mse (app, prkdf->path[prkdf->pathlen-1]);
|
2005-04-27 12:09:21 +00:00
|
|
|
|
}
|
|
|
|
|
else if (prkdf->key_reference_valid)
|
|
|
|
|
{
|
|
|
|
|
unsigned char mse[3];
|
2011-02-04 12:57:53 +01:00
|
|
|
|
|
2005-04-27 12:09:21 +00:00
|
|
|
|
mse[0] = 0x84; /* Select asym. key. */
|
|
|
|
|
mse[1] = 1;
|
|
|
|
|
mse[2] = prkdf->key_reference;
|
2004-12-02 07:48:09 +00:00
|
|
|
|
|
2021-04-29 15:26:36 +02:00
|
|
|
|
err = iso7816_manage_security_env (app_get_slot (app),
|
2005-04-27 12:09:21 +00:00
|
|
|
|
0x41, 0xB6,
|
|
|
|
|
mse, sizeof mse);
|
2005-04-27 19:47:53 +00:00
|
|
|
|
}
|
|
|
|
|
if (err)
|
|
|
|
|
{
|
2020-03-31 10:40:21 +02:00
|
|
|
|
log_error ("p15: MSE failed: %s\n", gpg_strerror (err));
|
2021-04-13 14:02:18 +02:00
|
|
|
|
goto leave;
|
2005-04-27 12:09:21 +00:00
|
|
|
|
}
|
2004-12-02 07:48:09 +00:00
|
|
|
|
|
2022-02-21 12:15:15 +01:00
|
|
|
|
if (prkdf->keyalgo == GCRY_PK_RSA && prkdf->keynbits >= 2048)
|
2020-03-31 19:55:15 +02:00
|
|
|
|
{
|
|
|
|
|
exmode = 1;
|
|
|
|
|
le_value = prkdf->keynbits / 8;
|
|
|
|
|
}
|
2005-09-06 18:42:13 +00:00
|
|
|
|
else
|
2020-03-31 19:55:15 +02:00
|
|
|
|
{
|
|
|
|
|
exmode = 0;
|
|
|
|
|
le_value = 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
err = iso7816_compute_ds (app_get_slot (app),
|
2021-04-29 15:26:36 +02:00
|
|
|
|
exmode, indata, indatalen,
|
2020-03-31 19:55:15 +02:00
|
|
|
|
le_value, outdata, outdatalen);
|
|
|
|
|
|
2021-04-29 15:26:36 +02:00
|
|
|
|
leave:
|
|
|
|
|
xfree (indata_buffer);
|
2005-04-27 12:09:21 +00:00
|
|
|
|
return err;
|
2004-12-02 07:48:09 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2011-02-04 12:57:53 +01:00
|
|
|
|
/* Handler for the PKAUTH command.
|
2005-09-09 11:18:08 +00:00
|
|
|
|
|
2006-10-10 11:11:04 +00:00
|
|
|
|
This is basically the same as the PKSIGN command but we first check
|
2005-09-09 11:18:08 +00:00
|
|
|
|
that the requested key is suitable for authentication; that is, it
|
|
|
|
|
must match the criteria used for the attribute $AUTHKEYID. See
|
|
|
|
|
do_sign for calling conventions; there is no HASHALGO, though. */
|
2011-02-04 12:57:53 +01:00
|
|
|
|
static gpg_error_t
|
2021-02-19 09:50:22 +01:00
|
|
|
|
do_auth (app_t app, ctrl_t ctrl, const char *keyidstr,
|
2005-09-09 11:18:08 +00:00
|
|
|
|
gpg_error_t (*pincb)(void*, const char *, char **),
|
|
|
|
|
void *pincb_arg,
|
|
|
|
|
const void *indata, size_t indatalen,
|
|
|
|
|
unsigned char **outdata, size_t *outdatalen )
|
|
|
|
|
{
|
|
|
|
|
gpg_error_t err;
|
|
|
|
|
prkdf_object_t prkdf;
|
2006-10-24 14:45:34 +00:00
|
|
|
|
int algo;
|
2005-09-09 11:18:08 +00:00
|
|
|
|
|
|
|
|
|
if (!keyidstr || !*keyidstr)
|
|
|
|
|
return gpg_error (GPG_ERR_INV_VALUE);
|
|
|
|
|
|
|
|
|
|
err = prkdf_object_from_keyidstr (app, keyidstr, &prkdf);
|
|
|
|
|
if (err)
|
|
|
|
|
return err;
|
2021-04-29 15:26:36 +02:00
|
|
|
|
if (!(prkdf->usageflags.sign || prkdf->gpgusage.auth))
|
2005-09-09 11:18:08 +00:00
|
|
|
|
{
|
2020-03-31 10:40:21 +02:00
|
|
|
|
log_error ("p15: key %s may not be used for authentication\n", keyidstr);
|
2005-09-09 11:18:08 +00:00
|
|
|
|
return gpg_error (GPG_ERR_WRONG_KEY_USAGE);
|
|
|
|
|
}
|
2006-10-24 14:45:34 +00:00
|
|
|
|
|
2008-09-29 15:02:55 +00:00
|
|
|
|
algo = indatalen == 36? MD_USER_TLS_MD5SHA1 : GCRY_MD_SHA1;
|
2021-02-19 09:50:22 +01:00
|
|
|
|
return do_sign (app, ctrl, keyidstr, algo, pincb, pincb_arg,
|
2005-09-09 11:18:08 +00:00
|
|
|
|
indata, indatalen, outdata, outdatalen);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2020-04-01 14:07:56 +02:00
|
|
|
|
/* Handler for the PKDECRYPT command. Decrypt the data in INDATA and
|
|
|
|
|
* return the allocated result in OUTDATA. If a PIN is required the
|
|
|
|
|
* PINCB will be used to ask for the PIN; it should return the PIN in
|
|
|
|
|
* an allocated buffer and put it into PIN. */
|
|
|
|
|
static gpg_error_t
|
2021-02-19 09:50:22 +01:00
|
|
|
|
do_decipher (app_t app, ctrl_t ctrl, const char *keyidstr,
|
2020-04-01 14:07:56 +02:00
|
|
|
|
gpg_error_t (*pincb)(void*, const char *, char **),
|
|
|
|
|
void *pincb_arg,
|
|
|
|
|
const void *indata, size_t indatalen,
|
|
|
|
|
unsigned char **outdata, size_t *outdatalen,
|
|
|
|
|
unsigned int *r_info)
|
|
|
|
|
{
|
|
|
|
|
gpg_error_t err;
|
|
|
|
|
prkdf_object_t prkdf; /* The private key object. */
|
|
|
|
|
aodf_object_t aodf; /* The associated authentication object. */
|
|
|
|
|
int exmode, le_value, padind;
|
|
|
|
|
|
2021-02-19 09:50:22 +01:00
|
|
|
|
(void)ctrl;
|
2020-04-01 14:07:56 +02:00
|
|
|
|
(void)r_info;
|
|
|
|
|
|
|
|
|
|
if (!keyidstr || !*keyidstr)
|
|
|
|
|
return gpg_error (GPG_ERR_INV_VALUE);
|
|
|
|
|
if (!indatalen || !indata || !outdatalen || !outdata)
|
|
|
|
|
return gpg_error (GPG_ERR_INV_ARG);
|
|
|
|
|
|
|
|
|
|
err = prkdf_object_from_keyidstr (app, keyidstr, &prkdf);
|
|
|
|
|
if (err)
|
|
|
|
|
return err;
|
2021-04-29 15:26:36 +02:00
|
|
|
|
if (!(prkdf->usageflags.decrypt
|
|
|
|
|
|| prkdf->usageflags.unwrap
|
|
|
|
|
|| prkdf->gpgusage.encr ))
|
2020-04-01 14:07:56 +02:00
|
|
|
|
{
|
2021-04-29 15:26:36 +02:00
|
|
|
|
log_error ("p15: key %s may not be used for decryption\n", keyidstr);
|
2020-04-01 14:07:56 +02:00
|
|
|
|
return gpg_error (GPG_ERR_WRONG_KEY_USAGE);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Find the authentication object to this private key object. */
|
|
|
|
|
if (!prkdf->authid)
|
|
|
|
|
{
|
|
|
|
|
log_error ("p15: no authentication object defined for %s\n", keyidstr);
|
|
|
|
|
/* fixme: we might want to go ahead and do without PIN
|
|
|
|
|
verification. */
|
|
|
|
|
return gpg_error (GPG_ERR_UNSUPPORTED_OPERATION);
|
|
|
|
|
}
|
|
|
|
|
for (aodf = app->app_local->auth_object_info; aodf; aodf = aodf->next)
|
|
|
|
|
if (aodf->objidlen == prkdf->authidlen
|
|
|
|
|
&& !memcmp (aodf->objid, prkdf->authid, prkdf->authidlen))
|
|
|
|
|
break;
|
|
|
|
|
if (!aodf)
|
2021-04-29 15:26:36 +02:00
|
|
|
|
log_info ("p15: no authentication for %s needed\n", keyidstr);
|
2020-04-01 14:07:56 +02:00
|
|
|
|
|
2020-04-01 17:43:05 +02:00
|
|
|
|
/* We need some more info about the key - get the keygrip to
|
|
|
|
|
* populate these fields. */
|
|
|
|
|
err = keygrip_from_prkdf (app, prkdf);
|
|
|
|
|
if (err)
|
|
|
|
|
{
|
|
|
|
|
log_error ("p15: keygrip_from_prkdf failed: %s\n", gpg_strerror (err));
|
|
|
|
|
return err;
|
|
|
|
|
}
|
2020-04-01 14:07:56 +02:00
|
|
|
|
|
|
|
|
|
/* Verify the PIN. */
|
|
|
|
|
err = prepare_verify_pin (app, keyidstr, prkdf, aodf);
|
|
|
|
|
if (!err)
|
|
|
|
|
err = verify_pin (app, pincb, pincb_arg, prkdf, aodf);
|
|
|
|
|
if (err)
|
|
|
|
|
return err;
|
|
|
|
|
|
2021-04-29 15:26:36 +02:00
|
|
|
|
if (prkdf->is_ecc && IS_CARDOS_5(app))
|
|
|
|
|
{
|
|
|
|
|
|
|
|
|
|
err = iso7816_manage_security_env (app_get_slot (app), 0xF3, 0x01,
|
|
|
|
|
NULL, 0);
|
|
|
|
|
if (err)
|
|
|
|
|
{
|
|
|
|
|
log_error ("p15: MSE failed: %s\n", gpg_strerror (err));
|
|
|
|
|
return err;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2020-04-01 14:07:56 +02:00
|
|
|
|
|
|
|
|
|
/* The next is guess work for CardOS. */
|
2020-04-07 16:00:11 +02:00
|
|
|
|
if (app->app_local->card_product == CARD_PRODUCT_DTRUST)
|
|
|
|
|
{
|
|
|
|
|
/* From analyzing an USB trace of a Windows signing application
|
|
|
|
|
* we see that the SE is simply reset to 0x14. It seems to be
|
|
|
|
|
* sufficient to do this for decryption; signing still works
|
|
|
|
|
* with the standard code despite that our trace showed that
|
|
|
|
|
* there the SE is restored to 0x09. Note that the special
|
|
|
|
|
* D-Trust AID is in any case select by prepare_verify_pin.
|
|
|
|
|
*
|
|
|
|
|
* Hey, D-Trust please hand over the specs so that you can
|
|
|
|
|
* actually sell your cards and we can properly implement it;
|
|
|
|
|
* other vendors understand this and do not demand ridiculous
|
|
|
|
|
* paper work or complicated procedures to get samples. */
|
|
|
|
|
err = iso7816_manage_security_env (app_get_slot (app),
|
|
|
|
|
0xF3, 0x14, NULL, 0);
|
|
|
|
|
|
|
|
|
|
}
|
|
|
|
|
else if (prkdf->key_reference_valid)
|
2020-04-01 14:07:56 +02:00
|
|
|
|
{
|
2021-04-29 15:26:36 +02:00
|
|
|
|
unsigned char mse[9];
|
|
|
|
|
int i;
|
2020-04-01 14:07:56 +02:00
|
|
|
|
|
2020-04-07 16:00:11 +02:00
|
|
|
|
/* Note: This works with CardOS but the D-Trust card has the
|
|
|
|
|
* problem that the next created signature would be broken. */
|
|
|
|
|
|
2021-04-29 15:26:36 +02:00
|
|
|
|
i = 0;
|
|
|
|
|
if (!prkdf->is_ecc)
|
|
|
|
|
{
|
|
|
|
|
mse[i++] = 0x80; /* Algorithm reference. */
|
|
|
|
|
mse[i++] = 1;
|
|
|
|
|
mse[i++] = 0x0a; /* RSA, no padding. */
|
|
|
|
|
}
|
|
|
|
|
mse[i++] = 0x84; /* Key reference. */
|
|
|
|
|
mse[i++] = 1;
|
|
|
|
|
mse[i++] = prkdf->key_reference;
|
|
|
|
|
if (prkdf->is_ecc && IS_CARDOS_5(app))
|
|
|
|
|
{
|
|
|
|
|
mse[i++] = 0x95; /* ???. */
|
|
|
|
|
mse[i++] = 1;
|
|
|
|
|
mse[i++] = 0x40;
|
|
|
|
|
}
|
|
|
|
|
log_assert (i <= DIM(mse));
|
2020-04-01 14:07:56 +02:00
|
|
|
|
err = iso7816_manage_security_env (app_get_slot (app), 0x41, 0xB8,
|
2021-04-29 15:26:36 +02:00
|
|
|
|
mse, i);
|
2020-04-01 14:07:56 +02:00
|
|
|
|
}
|
2020-04-07 16:00:11 +02:00
|
|
|
|
/* Check for MSE error. */
|
|
|
|
|
if (err)
|
|
|
|
|
{
|
|
|
|
|
log_error ("p15: MSE failed: %s\n", gpg_strerror (err));
|
|
|
|
|
return err;
|
|
|
|
|
}
|
2020-04-01 14:07:56 +02:00
|
|
|
|
|
|
|
|
|
exmode = le_value = 0;
|
|
|
|
|
padind = 0;
|
2022-02-21 12:15:15 +01:00
|
|
|
|
if (prkdf->keyalgo == GCRY_PK_RSA && prkdf->keynbits >= 2048)
|
2020-04-01 14:07:56 +02:00
|
|
|
|
{
|
|
|
|
|
exmode = 1; /* Extended length w/o a limit. */
|
|
|
|
|
le_value = prkdf->keynbits / 8;
|
|
|
|
|
}
|
|
|
|
|
|
2020-04-07 16:00:11 +02:00
|
|
|
|
if (app->app_local->card_product == CARD_PRODUCT_DTRUST)
|
|
|
|
|
padind = 0x81;
|
|
|
|
|
|
2021-04-29 15:26:36 +02:00
|
|
|
|
if (prkdf->is_ecc && IS_CARDOS_5(app))
|
|
|
|
|
{
|
|
|
|
|
if ((indatalen & 1) && *(const char *)indata == 0x04)
|
|
|
|
|
{
|
|
|
|
|
/* Strip indicator byte. */
|
|
|
|
|
indatalen--;
|
|
|
|
|
indata = (const char *)indata + 1;
|
|
|
|
|
}
|
|
|
|
|
err = iso7816_pso_csv (app_get_slot (app), exmode,
|
|
|
|
|
indata, indatalen,
|
|
|
|
|
le_value,
|
|
|
|
|
outdata, outdatalen);
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
err = iso7816_decipher (app_get_slot (app), exmode,
|
|
|
|
|
indata, indatalen,
|
|
|
|
|
le_value, padind,
|
|
|
|
|
outdata, outdatalen);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return err;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* Perform a simple verify operation for the PIN specified by
|
|
|
|
|
* KEYIDSTR. Note that we require a key reference which is then used
|
|
|
|
|
* to select the authentication object. Return GPG_ERR_NO_PIN if a
|
|
|
|
|
* PIN is not required for using the private key KEYIDSTR. */
|
|
|
|
|
static gpg_error_t
|
|
|
|
|
do_check_pin (app_t app, ctrl_t ctrl, const char *keyidstr,
|
|
|
|
|
gpg_error_t (*pincb)(void*, const char *, char **),
|
|
|
|
|
void *pincb_arg)
|
|
|
|
|
{
|
|
|
|
|
gpg_error_t err;
|
|
|
|
|
prkdf_object_t prkdf; /* The private key object. */
|
|
|
|
|
aodf_object_t aodf; /* The associated authentication object. */
|
|
|
|
|
|
|
|
|
|
(void)ctrl;
|
|
|
|
|
|
|
|
|
|
if (!keyidstr || !*keyidstr)
|
|
|
|
|
return gpg_error (GPG_ERR_INV_VALUE);
|
|
|
|
|
|
|
|
|
|
err = prkdf_object_from_keyidstr (app, keyidstr, &prkdf);
|
|
|
|
|
if (err
|
|
|
|
|
&& gpg_err_code (err) != GPG_ERR_INV_ID
|
|
|
|
|
&& gpg_err_code (err) != GPG_ERR_NOT_FOUND)
|
|
|
|
|
return err;
|
|
|
|
|
|
|
|
|
|
if (err) /* Not found or invalid - assume it is the label. */
|
|
|
|
|
{
|
|
|
|
|
prkdf = NULL;
|
|
|
|
|
for (aodf = app->app_local->auth_object_info; aodf; aodf = aodf->next)
|
|
|
|
|
if (aodf->label && !ascii_strcasecmp (aodf->label, keyidstr))
|
|
|
|
|
break;
|
|
|
|
|
if (!aodf)
|
|
|
|
|
return err; /* Re-use the original error code. */
|
|
|
|
|
}
|
|
|
|
|
else /* Find the authentication object to this private key object. */
|
|
|
|
|
{
|
|
|
|
|
if (!prkdf->authid)
|
|
|
|
|
{
|
|
|
|
|
log_error ("p15: no authentication object defined for %s\n",
|
|
|
|
|
keyidstr);
|
|
|
|
|
return gpg_error (GPG_ERR_UNSUPPORTED_OPERATION);
|
|
|
|
|
}
|
|
|
|
|
for (aodf = app->app_local->auth_object_info; aodf; aodf = aodf->next)
|
|
|
|
|
if (aodf->objidlen == prkdf->authidlen
|
|
|
|
|
&& !memcmp (aodf->objid, prkdf->authid, prkdf->authidlen))
|
|
|
|
|
break;
|
|
|
|
|
if (!aodf) /* None found. */
|
|
|
|
|
return gpg_error (GPG_ERR_NO_PIN);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
err = prepare_verify_pin (app, keyidstr, prkdf, aodf);
|
|
|
|
|
if (!err)
|
|
|
|
|
err = verify_pin (app, pincb, pincb_arg, prkdf, aodf);
|
|
|
|
|
|
|
|
|
|
return err;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* Process the various keygrip based info requests. */
|
|
|
|
|
static gpg_error_t
|
|
|
|
|
do_with_keygrip (app_t app, ctrl_t ctrl, int action,
|
|
|
|
|
const char *want_keygripstr, int capability)
|
|
|
|
|
{
|
|
|
|
|
gpg_error_t err;
|
|
|
|
|
char *serialno = NULL;
|
|
|
|
|
int as_data = 0;
|
|
|
|
|
prkdf_object_t prkdf;
|
|
|
|
|
|
|
|
|
|
/* First a quick check for valid parameters. */
|
|
|
|
|
switch (action)
|
|
|
|
|
{
|
|
|
|
|
case KEYGRIP_ACTION_LOOKUP:
|
|
|
|
|
if (!want_keygripstr)
|
|
|
|
|
{
|
|
|
|
|
err = gpg_error (GPG_ERR_NOT_FOUND);
|
|
|
|
|
goto leave;
|
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
case KEYGRIP_ACTION_SEND_DATA:
|
|
|
|
|
as_data = 1;
|
|
|
|
|
break;
|
|
|
|
|
case KEYGRIP_ACTION_WRITE_STATUS:
|
|
|
|
|
break;
|
|
|
|
|
default:
|
|
|
|
|
err = gpg_error (GPG_ERR_INV_ARG);
|
|
|
|
|
goto leave;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Allocate the s/n string if needed. */
|
|
|
|
|
if (action != KEYGRIP_ACTION_LOOKUP)
|
|
|
|
|
{
|
|
|
|
|
serialno = app_get_serialno (app);
|
|
|
|
|
if (!serialno)
|
|
|
|
|
{
|
|
|
|
|
err = gpg_error_from_syserror ();
|
|
|
|
|
goto leave;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
for (prkdf = app->app_local->private_key_info;
|
|
|
|
|
prkdf; prkdf = prkdf->next)
|
|
|
|
|
{
|
|
|
|
|
if (keygrip_from_prkdf (app, prkdf))
|
|
|
|
|
continue;
|
|
|
|
|
|
|
|
|
|
if (action == KEYGRIP_ACTION_LOOKUP)
|
|
|
|
|
{
|
|
|
|
|
if (!strcmp (prkdf->keygrip, want_keygripstr))
|
|
|
|
|
{
|
|
|
|
|
err = 0; /* Found */
|
|
|
|
|
goto leave;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
else if (!want_keygripstr || !strcmp (prkdf->keygrip, want_keygripstr))
|
|
|
|
|
{
|
|
|
|
|
char *keyref;
|
|
|
|
|
|
|
|
|
|
if (capability == GCRY_PK_USAGE_SIGN)
|
|
|
|
|
{
|
|
|
|
|
if (!(prkdf->usageflags.sign || prkdf->usageflags.sign_recover
|
|
|
|
|
|| prkdf->usageflags.non_repudiation))
|
|
|
|
|
continue;
|
|
|
|
|
}
|
|
|
|
|
else if (capability == GCRY_PK_USAGE_ENCR)
|
|
|
|
|
{
|
|
|
|
|
if (!(prkdf->usageflags.decrypt || prkdf->usageflags.unwrap))
|
|
|
|
|
continue;
|
|
|
|
|
}
|
|
|
|
|
else if (capability == GCRY_PK_USAGE_AUTH)
|
|
|
|
|
{
|
|
|
|
|
if (!(prkdf->usageflags.sign || prkdf->usageflags.sign_recover))
|
|
|
|
|
continue;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
keyref = keyref_from_prkdf (app, prkdf);
|
|
|
|
|
if (!keyref)
|
|
|
|
|
{
|
|
|
|
|
err = gpg_error_from_syserror ();
|
|
|
|
|
goto leave;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
send_keyinfo (ctrl, as_data, prkdf->keygrip, serialno, keyref);
|
|
|
|
|
xfree (keyref);
|
|
|
|
|
if (want_keygripstr)
|
|
|
|
|
{
|
|
|
|
|
err = 0; /* Found */
|
|
|
|
|
goto leave;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Return an error so that the dispatcher keeps on looping over the
|
|
|
|
|
* other applications. For clarity we use a different error code
|
|
|
|
|
* when listing all keys. Note that in lookup mode WANT_KEYGRIPSTR
|
|
|
|
|
* is not NULL. */
|
|
|
|
|
if (!want_keygripstr)
|
|
|
|
|
err = gpg_error (GPG_ERR_TRUE);
|
|
|
|
|
else
|
|
|
|
|
err = gpg_error (GPG_ERR_NOT_FOUND);
|
|
|
|
|
|
|
|
|
|
leave:
|
|
|
|
|
xfree (serialno);
|
2020-04-01 14:07:56 +02:00
|
|
|
|
return err;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2005-09-05 14:36:36 +00:00
|
|
|
|
|
|
|
|
|
/* Assume that EF(DIR) has been selected. Read its content and figure
|
2005-09-06 18:42:13 +00:00
|
|
|
|
out the home EF of pkcs#15. Return that home DF or 0 if not found
|
|
|
|
|
and the value at the address of BELPIC indicates whether it was
|
|
|
|
|
found by the belpic aid. */
|
2005-09-05 14:36:36 +00:00
|
|
|
|
static unsigned short
|
2005-09-06 18:42:13 +00:00
|
|
|
|
read_home_df (int slot, int *r_belpic)
|
2005-09-05 14:36:36 +00:00
|
|
|
|
{
|
|
|
|
|
gpg_error_t err;
|
|
|
|
|
unsigned char *buffer;
|
|
|
|
|
const unsigned char *p, *pp;
|
|
|
|
|
size_t buflen, n, nn;
|
|
|
|
|
unsigned short result = 0;
|
|
|
|
|
|
2005-09-06 18:42:13 +00:00
|
|
|
|
*r_belpic = 0;
|
|
|
|
|
|
2005-09-05 14:36:36 +00:00
|
|
|
|
err = iso7816_read_binary (slot, 0, 0, &buffer, &buflen);
|
|
|
|
|
if (err)
|
|
|
|
|
{
|
2020-03-31 10:40:21 +02:00
|
|
|
|
log_error ("p15: error reading EF(DIR): %s\n", gpg_strerror (err));
|
2005-09-05 14:36:36 +00:00
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* FIXME: We need to scan all records. */
|
|
|
|
|
p = find_tlv (buffer, buflen, 0x61, &n);
|
|
|
|
|
if (p && n)
|
|
|
|
|
{
|
|
|
|
|
pp = find_tlv (p, n, 0x4f, &nn);
|
2005-09-06 18:42:13 +00:00
|
|
|
|
if (pp && ((nn == sizeof pkcs15_aid && !memcmp (pp, pkcs15_aid, nn))
|
|
|
|
|
|| (*r_belpic = (nn == sizeof pkcs15be_aid
|
|
|
|
|
&& !memcmp (pp, pkcs15be_aid, nn)))))
|
2005-09-05 14:36:36 +00:00
|
|
|
|
{
|
|
|
|
|
pp = find_tlv (p, n, 0x50, &nn);
|
2020-03-31 10:40:21 +02:00
|
|
|
|
if (pp && opt.verbose)
|
|
|
|
|
log_info ("p15: application label from EF(DIR) is '%.*s'\n",
|
2005-09-05 14:36:36 +00:00
|
|
|
|
(int)nn, pp);
|
|
|
|
|
pp = find_tlv (p, n, 0x51, &nn);
|
|
|
|
|
if (pp && nn == 4 && *pp == 0x3f && !pp[1])
|
|
|
|
|
{
|
|
|
|
|
result = ((pp[2] << 8) | pp[3]);
|
2020-03-31 10:40:21 +02:00
|
|
|
|
if (opt.verbose)
|
|
|
|
|
log_info ("p15: application directory is 0x%04hX\n", result);
|
2005-09-05 14:36:36 +00:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
xfree (buffer);
|
|
|
|
|
return result;
|
|
|
|
|
}
|
2005-04-27 12:09:21 +00:00
|
|
|
|
|
|
|
|
|
|
2011-02-04 12:57:53 +01:00
|
|
|
|
/*
|
|
|
|
|
Select the PKCS#15 application on the card in SLOT.
|
2005-09-05 14:36:36 +00:00
|
|
|
|
*/
|
2005-04-27 12:09:21 +00:00
|
|
|
|
gpg_error_t
|
|
|
|
|
app_select_p15 (app_t app)
|
2004-12-02 07:48:09 +00:00
|
|
|
|
{
|
2021-04-29 15:26:36 +02:00
|
|
|
|
int slot = app_get_slot (app);
|
2004-12-02 07:48:09 +00:00
|
|
|
|
int rc;
|
2005-04-27 12:09:21 +00:00
|
|
|
|
unsigned short def_home_df = 0;
|
|
|
|
|
card_type_t card_type = CARD_TYPE_UNKNOWN;
|
2005-09-05 14:36:36 +00:00
|
|
|
|
int direct = 0;
|
2005-09-06 18:42:13 +00:00
|
|
|
|
int is_belpic = 0;
|
2021-04-29 15:26:36 +02:00
|
|
|
|
unsigned char *fci = NULL;
|
|
|
|
|
size_t fcilen;
|
2005-09-06 18:42:13 +00:00
|
|
|
|
|
2021-04-29 15:26:36 +02:00
|
|
|
|
rc = iso7816_select_application_ext (slot, pkcs15_aid, sizeof pkcs15_aid, 1,
|
|
|
|
|
&fci, &fcilen);
|
2005-09-05 14:36:36 +00:00
|
|
|
|
if (rc)
|
|
|
|
|
{ /* Not found: Try to locate it from 2F00. We use direct path
|
|
|
|
|
selection here because it seems that the Belgian eID card
|
|
|
|
|
does only allow for that. Many other cards supports this
|
2005-11-28 11:52:25 +00:00
|
|
|
|
selection method too. Note, that we don't use
|
|
|
|
|
select_application above for the Belgian card - the call
|
2020-03-30 21:18:12 +02:00
|
|
|
|
works but it seems that it does not switch to the correct DF.
|
2005-11-28 11:52:25 +00:00
|
|
|
|
Using the 2f02 just works. */
|
2005-09-05 14:36:36 +00:00
|
|
|
|
unsigned short path[1] = { 0x2f00 };
|
|
|
|
|
|
2021-01-26 17:42:55 +01:00
|
|
|
|
rc = iso7816_select_path (slot, path, 1, 0);
|
2005-09-05 14:36:36 +00:00
|
|
|
|
if (!rc)
|
|
|
|
|
{
|
|
|
|
|
direct = 1;
|
2005-09-06 18:42:13 +00:00
|
|
|
|
def_home_df = read_home_df (slot, &is_belpic);
|
2005-09-05 14:36:36 +00:00
|
|
|
|
if (def_home_df)
|
|
|
|
|
{
|
|
|
|
|
path[0] = def_home_df;
|
2021-01-26 17:42:55 +01:00
|
|
|
|
rc = iso7816_select_path (slot, path, 1, 0);
|
2005-09-05 14:36:36 +00:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
if (rc)
|
|
|
|
|
{ /* Still not found: Try the default DF. */
|
2021-04-29 15:26:36 +02:00
|
|
|
|
def_home_df = DEFAULT_HOME_DF;
|
2017-03-06 13:39:46 +09:00
|
|
|
|
rc = iso7816_select_file (slot, def_home_df, 1);
|
2005-04-27 12:09:21 +00:00
|
|
|
|
}
|
|
|
|
|
if (!rc)
|
|
|
|
|
{
|
2005-09-06 18:42:13 +00:00
|
|
|
|
/* Determine the type of the card. The general case is to look
|
|
|
|
|
it up from the ATR table. For the Belgian eID card we know
|
|
|
|
|
it instantly from the AID. */
|
|
|
|
|
if (is_belpic)
|
|
|
|
|
{
|
|
|
|
|
card_type = CARD_TYPE_BELPIC;
|
|
|
|
|
}
|
2005-04-27 12:09:21 +00:00
|
|
|
|
else
|
|
|
|
|
{
|
2005-09-06 18:42:13 +00:00
|
|
|
|
unsigned char *atr;
|
|
|
|
|
size_t atrlen;
|
|
|
|
|
int i;
|
|
|
|
|
|
2021-04-29 15:26:36 +02:00
|
|
|
|
atr = apdu_get_atr (app_get_slot (app), &atrlen);
|
2005-09-06 18:42:13 +00:00
|
|
|
|
if (!atr)
|
|
|
|
|
rc = gpg_error (GPG_ERR_INV_CARD);
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
for (i=0; card_atr_list[i].atrlen; i++)
|
|
|
|
|
if (card_atr_list[i].atrlen == atrlen
|
|
|
|
|
&& !memcmp (card_atr_list[i].atr, atr, atrlen))
|
|
|
|
|
{
|
|
|
|
|
card_type = card_atr_list[i].type;
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
xfree (atr);
|
|
|
|
|
}
|
2005-04-27 12:09:21 +00:00
|
|
|
|
}
|
|
|
|
|
}
|
2004-12-02 07:48:09 +00:00
|
|
|
|
if (!rc)
|
|
|
|
|
{
|
2021-02-19 09:17:06 +01:00
|
|
|
|
app->apptype = APPTYPE_P15;
|
2004-12-02 07:48:09 +00:00
|
|
|
|
|
|
|
|
|
app->app_local = xtrycalloc (1, sizeof *app->app_local);
|
|
|
|
|
if (!app->app_local)
|
|
|
|
|
{
|
2006-09-14 16:50:33 +00:00
|
|
|
|
rc = gpg_error_from_syserror ();
|
2004-12-02 07:48:09 +00:00
|
|
|
|
goto leave;
|
|
|
|
|
}
|
|
|
|
|
|
2021-04-29 15:26:36 +02:00
|
|
|
|
/* Set the home DF from the FCI returned by the select. */
|
|
|
|
|
if (!def_home_df && fci)
|
|
|
|
|
{
|
|
|
|
|
const unsigned char *s;
|
|
|
|
|
size_t n;
|
|
|
|
|
|
|
|
|
|
s = find_tlv (fci, fcilen, 0x83, &n);
|
|
|
|
|
if (s && n == 2)
|
|
|
|
|
def_home_df = buf16_to_ushort (s);
|
|
|
|
|
else
|
2021-06-18 17:32:01 +02:00
|
|
|
|
{
|
|
|
|
|
if (fcilen)
|
|
|
|
|
log_printhex (fci, fcilen, "fci:");
|
|
|
|
|
log_info ("p15: select did not return the DF - using default\n");
|
|
|
|
|
def_home_df = DEFAULT_HOME_DF;
|
|
|
|
|
}
|
2021-04-29 15:26:36 +02:00
|
|
|
|
}
|
2005-04-27 12:09:21 +00:00
|
|
|
|
app->app_local->home_df = def_home_df;
|
|
|
|
|
|
|
|
|
|
/* Store the card type. FIXME: We might want to put this into
|
|
|
|
|
the common APP structure. */
|
|
|
|
|
app->app_local->card_type = card_type;
|
|
|
|
|
|
2020-04-07 16:00:11 +02:00
|
|
|
|
app->app_local->card_product = CARD_PRODUCT_UNKNOWN;
|
|
|
|
|
|
2005-09-05 14:36:36 +00:00
|
|
|
|
/* Store whether we may and should use direct path selection. */
|
2021-04-29 15:26:36 +02:00
|
|
|
|
switch (card_type)
|
|
|
|
|
{
|
|
|
|
|
case CARD_TYPE_CARDOS_50:
|
|
|
|
|
case CARD_TYPE_CARDOS_53:
|
|
|
|
|
direct = 1;
|
|
|
|
|
break;
|
2021-06-18 17:32:01 +02:00
|
|
|
|
case CARD_TYPE_AET:
|
|
|
|
|
app->app_local->no_extended_mode = 1;
|
|
|
|
|
break;
|
2021-04-29 15:26:36 +02:00
|
|
|
|
default:
|
|
|
|
|
/* Use whatever has been determined above. */
|
|
|
|
|
break;
|
|
|
|
|
}
|
2005-09-05 14:36:36 +00:00
|
|
|
|
app->app_local->direct_path_selection = direct;
|
|
|
|
|
|
2005-04-27 19:47:53 +00:00
|
|
|
|
/* Read basic information and thus check whether this is a real
|
2004-12-02 07:48:09 +00:00
|
|
|
|
card. */
|
|
|
|
|
rc = read_p15_info (app);
|
2005-04-27 12:09:21 +00:00
|
|
|
|
if (rc)
|
|
|
|
|
goto leave;
|
|
|
|
|
|
|
|
|
|
/* Special serial number munging. We need to check for a German
|
|
|
|
|
prototype card right here because we need to access to
|
|
|
|
|
EF(TokenInfo). We mark such a serial number by the using a
|
|
|
|
|
prefix of FF0100. */
|
2021-04-29 15:26:36 +02:00
|
|
|
|
if (APP_CARD(app)->serialnolen == 12
|
|
|
|
|
&& !memcmp (APP_CARD(app)->serialno,
|
|
|
|
|
"\xD2\x76\0\0\0\0\0\0\0\0\0\0", 12))
|
2004-12-02 07:48:09 +00:00
|
|
|
|
{
|
|
|
|
|
/* This is a German card with a silly serial number. Try to get
|
2005-04-27 12:09:21 +00:00
|
|
|
|
the serial number from the EF(TokenInfo). . */
|
|
|
|
|
unsigned char *p;
|
|
|
|
|
|
|
|
|
|
/* FIXME: actually get it from EF(TokenInfo). */
|
2011-02-04 12:57:53 +01:00
|
|
|
|
|
2021-04-29 15:26:36 +02:00
|
|
|
|
p = xtrymalloc (3 + APP_CARD(app)->serialnolen);
|
2004-12-02 07:48:09 +00:00
|
|
|
|
if (!p)
|
|
|
|
|
rc = gpg_error (gpg_err_code_from_errno (errno));
|
|
|
|
|
else
|
|
|
|
|
{
|
2005-04-27 12:09:21 +00:00
|
|
|
|
memcpy (p, "\xff\x01", 3);
|
2021-04-29 15:26:36 +02:00
|
|
|
|
memcpy (p+3, APP_CARD(app)->serialno, APP_CARD(app)->serialnolen);
|
|
|
|
|
APP_CARD(app)->serialnolen += 3;
|
|
|
|
|
xfree (APP_CARD(app)->serialno);
|
|
|
|
|
APP_CARD(app)->serialno = p;
|
2004-12-02 07:48:09 +00:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
app->fnc.deinit = do_deinit;
|
2021-04-29 15:26:36 +02:00
|
|
|
|
app->fnc.prep_reselect = NULL;
|
|
|
|
|
app->fnc.reselect = NULL;
|
2004-12-02 07:48:09 +00:00
|
|
|
|
app->fnc.learn_status = do_learn_status;
|
|
|
|
|
app->fnc.readcert = do_readcert;
|
2005-09-09 11:18:08 +00:00
|
|
|
|
app->fnc.getattr = do_getattr;
|
2004-12-02 07:48:09 +00:00
|
|
|
|
app->fnc.setattr = NULL;
|
|
|
|
|
app->fnc.genkey = NULL;
|
|
|
|
|
app->fnc.sign = do_sign;
|
2005-09-09 11:18:08 +00:00
|
|
|
|
app->fnc.auth = do_auth;
|
2020-04-01 14:07:56 +02:00
|
|
|
|
app->fnc.decipher = do_decipher;
|
2004-12-02 07:48:09 +00:00
|
|
|
|
app->fnc.change_pin = NULL;
|
2021-04-29 15:26:36 +02:00
|
|
|
|
app->fnc.check_pin = do_check_pin;
|
|
|
|
|
app->fnc.with_keygrip = do_with_keygrip;
|
2004-12-02 07:48:09 +00:00
|
|
|
|
|
|
|
|
|
leave:
|
|
|
|
|
if (rc)
|
2005-04-27 12:09:21 +00:00
|
|
|
|
do_deinit (app);
|
2004-12-02 07:48:09 +00:00
|
|
|
|
}
|
|
|
|
|
|
2021-04-29 15:26:36 +02:00
|
|
|
|
xfree (fci);
|
2004-12-02 07:48:09 +00:00
|
|
|
|
return rc;
|
|
|
|
|
}
|