2009-01-08 19:56:30 +00:00
|
|
|
|
/* app-nks.c - The Telesec NKS card application.
|
2020-04-17 16:09:05 +02:00
|
|
|
|
* Copyright (C) 2004, 2007-2009 Free Software Foundation, Inc.
|
|
|
|
|
* Copyright (C) 2004, 2007-2009, 2013-2015, 2020 g10 Code GmbH
|
2004-01-27 16:40:42 +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-01-27 16:40:42 +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/>.
|
2004-01-27 16:40:42 +00:00
|
|
|
|
*/
|
|
|
|
|
|
2009-03-05 19:19:37 +00:00
|
|
|
|
/* Notes:
|
2020-05-07 08:18:28 +02:00
|
|
|
|
*
|
|
|
|
|
* - We are now targeting TCOS 3 cards and it may happen that there is
|
|
|
|
|
* a regression towards TCOS 2 cards. Please report.
|
|
|
|
|
*
|
|
|
|
|
* - The NKS3 AUT key is not used. It seems that it is only useful for
|
|
|
|
|
* the internal authentication command and not accessible by other
|
|
|
|
|
* applications. The key itself is in the encryption class but the
|
|
|
|
|
* corresponding certificate has only the digitalSignature
|
|
|
|
|
* capability.
|
|
|
|
|
* Update: This changed for the Signature Card V2 (nks version 15)
|
|
|
|
|
*
|
|
|
|
|
* - If required, we automagically switch between the NKS application
|
|
|
|
|
* and the SigG or eSign application. This avoids to use the DINSIG
|
|
|
|
|
* application which is somewhat limited, has no support for Secure
|
|
|
|
|
* Messaging as required by TCOS 3 and has no way to change the PIN
|
|
|
|
|
* or even set the NullPIN. With the Signature Card v2 (nks version
|
|
|
|
|
* 15) the Esign application is used instead of the SigG.
|
|
|
|
|
*
|
|
|
|
|
* - We use the prefix NKS-DF01 for TCOS 2 cards and NKS-NKS3 for newer
|
|
|
|
|
* cards. This is because the NKS application has moved to DF02 with
|
|
|
|
|
* TCOS 3 and thus we better use a DF independent tag.
|
|
|
|
|
*
|
|
|
|
|
* - We use only the global PINs for the NKS application.
|
|
|
|
|
*
|
|
|
|
|
*
|
|
|
|
|
*
|
|
|
|
|
* Here is a table with PIN stati collected from 3 cards.
|
|
|
|
|
*
|
|
|
|
|
* | app | pwid | NKS3 | SIG_B | SIG_N |
|
|
|
|
|
* |-----+------+-----------+-----------+-----------|
|
|
|
|
|
* | NKS | 0x00 | null - | - - | - - |
|
|
|
|
|
* | | 0x01 | 0 3 | - - | - - |
|
|
|
|
|
* | | 0x02 | 3 null | 15 3 | 15 null |
|
2020-05-07 14:03:25 +02:00
|
|
|
|
* | | 0x03 | - 3 | null - | 3 - |
|
|
|
|
|
* | | 0x04 | | null 0 | 3 3 |
|
2020-05-07 08:18:28 +02:00
|
|
|
|
* | SIG | 0x00 | null - | - - | - - |
|
|
|
|
|
* | | 0x01 | 0 null | - null | - null |
|
|
|
|
|
* | | 0x02 | 3 null | 15 0 | 15 0 |
|
|
|
|
|
* | | 0x03 | - 0 | null null | null null |
|
|
|
|
|
* - SIG is either SIGG or ESIGN.
|
|
|
|
|
* - "-" indicates reference not found (SW 6A88).
|
|
|
|
|
* - "null" indicates a NULLPIN (SW 6985).
|
|
|
|
|
* - The first value in each cell is the global PIN;
|
|
|
|
|
* the second is the local PIN (high bit of pwid set).
|
|
|
|
|
* - The NKS3 card is some older test card.
|
|
|
|
|
* - The SIG_B is a Signature Card V2.0 with Brainpool curves.
|
|
|
|
|
* Here the PIN 0x82 has been changed from the NULLPIN.
|
|
|
|
|
* - The SIG_N is a Signature Card V2.0 with NIST curves.
|
2020-05-07 14:03:25 +02:00
|
|
|
|
* The PIN was enabled using the TCOS Windows tool.
|
2009-03-05 19:19:37 +00:00
|
|
|
|
*/
|
|
|
|
|
|
2004-01-27 16:40:42 +00:00
|
|
|
|
#include <config.h>
|
|
|
|
|
#include <errno.h>
|
|
|
|
|
#include <stdio.h>
|
|
|
|
|
#include <stdlib.h>
|
|
|
|
|
#include <string.h>
|
|
|
|
|
#include <time.h>
|
|
|
|
|
|
|
|
|
|
#include "scdaemon.h"
|
2017-03-07 20:21:23 +09:00
|
|
|
|
#include "../common/i18n.h"
|
2004-01-27 16:40:42 +00:00
|
|
|
|
#include "iso7816.h"
|
2017-03-07 20:21:23 +09:00
|
|
|
|
#include "../common/tlv.h"
|
2009-03-05 19:19:37 +00:00
|
|
|
|
#include "apdu.h"
|
2017-03-07 20:21:23 +09:00
|
|
|
|
#include "../common/host2net.h"
|
2009-03-05 19:19:37 +00:00
|
|
|
|
|
|
|
|
|
static char const aid_nks[] = { 0xD2, 0x76, 0x00, 0x00, 0x03, 0x01, 0x02 };
|
|
|
|
|
static char const aid_sigg[] = { 0xD2, 0x76, 0x00, 0x00, 0x66, 0x01 };
|
2020-04-17 16:09:05 +02:00
|
|
|
|
static char const aid_esign[] =
|
|
|
|
|
{ 0xA0, 0x00, 0x00, 0x01, 0x67, 0x45, 0x53, 0x49, 0x47, 0x4E };
|
2020-05-07 19:44:26 +02:00
|
|
|
|
static char const aid_idlm[] = { 0xD2, 0x76, 0x00, 0x00, 0x03, 0x0c, 0x01 };
|
2020-04-17 16:09:05 +02:00
|
|
|
|
|
2020-05-07 19:44:26 +02:00
|
|
|
|
|
|
|
|
|
/* The ids of the different apps on our TCOS cards. */
|
2020-04-17 16:09:05 +02:00
|
|
|
|
#define NKS_APP_NKS 0
|
|
|
|
|
#define NKS_APP_SIGG 1
|
|
|
|
|
#define NKS_APP_ESIGN 2
|
2020-05-07 19:44:26 +02:00
|
|
|
|
#define NKS_APP_IDLM 3
|
2009-03-05 19:19:37 +00:00
|
|
|
|
|
2004-01-27 16:40:42 +00:00
|
|
|
|
|
2007-02-14 16:25:38 +00:00
|
|
|
|
static struct
|
|
|
|
|
{
|
2020-04-17 16:09:05 +02:00
|
|
|
|
int nks_app_id;/* One of the NKS_APP_ constants. */
|
2004-01-28 16:21:57 +00:00
|
|
|
|
int fid; /* File ID. */
|
2020-04-17 16:09:05 +02:00
|
|
|
|
int nks_ver; /* 0 for NKS version 2, 3 for version 3, etc. */
|
2004-01-28 16:21:57 +00:00
|
|
|
|
int certtype; /* Type of certificate or 0 if it is not a certificate. */
|
2009-03-26 19:27:04 +00:00
|
|
|
|
int iskeypair; /* If true has the FID of the corresponding certificate. */
|
2020-04-17 16:09:05 +02:00
|
|
|
|
int isauthkey; /* True if file is a key usable for authentication. */
|
2004-01-28 16:21:57 +00:00
|
|
|
|
int issignkey; /* True if file is a key usable for signing. */
|
2020-04-17 16:09:05 +02:00
|
|
|
|
int isencrkey; /* True if file is a key usable for decryption. */
|
2009-03-26 19:27:04 +00:00
|
|
|
|
unsigned char kid; /* Corresponding key references. */
|
2004-01-27 16:40:42 +00:00
|
|
|
|
} filelist[] = {
|
2020-04-17 16:09:05 +02:00
|
|
|
|
{ 0, 0x4531, 0, 0, 0xC000, 1,1,0, 0x80}, /* EF_PK.NKS.SIG */
|
|
|
|
|
/* */ /* nks15: EF.PK.NKS.ADS */
|
2009-03-26 19:27:04 +00:00
|
|
|
|
{ 0, 0xC000, 0, 101 }, /* EF_C.NKS.SIG */
|
2020-04-17 16:09:05 +02:00
|
|
|
|
/* */ /* nks15: EF.C.ICC.ADS (sign key) */
|
|
|
|
|
|
|
|
|
|
{ 0, 0x4331, 0, 100 }, /* Unnamed. */
|
|
|
|
|
/* */ /* nks15: EF.C.ICC.RFU1 */
|
|
|
|
|
/* */ /* (second cert for sign key) */
|
|
|
|
|
|
2009-03-05 19:19:37 +00:00
|
|
|
|
{ 0, 0x4332, 0, 100 },
|
2020-04-17 16:09:05 +02:00
|
|
|
|
{ 0, 0xB000, 0, 110 }, /* EF_PK.RCA.NKS */
|
|
|
|
|
|
|
|
|
|
{ 0, 0x45B1, 0, 0, 0xC200, 0,0,1, 0x81}, /* EF_PK.NKS.ENC */
|
|
|
|
|
/* */ /* nks15: EF.PK.ICC.ENC1 */
|
|
|
|
|
{ 0, 0xC200, 0, 101 }, /* EF_C.NKS.ENC */
|
|
|
|
|
/* nks15: EF.C.ICC.ENC1 (Cert-encr) */
|
|
|
|
|
|
|
|
|
|
{ 0, 0x43B1, 0, 100 }, /* Unnamed */
|
|
|
|
|
/* */ /* nks15: EF.C.ICC.RFU2 */
|
|
|
|
|
/* */ /* (second cert for enc1 key) */
|
|
|
|
|
|
2009-03-05 19:19:37 +00:00
|
|
|
|
{ 0, 0x43B2, 0, 100 },
|
2020-04-17 16:09:05 +02:00
|
|
|
|
{ 0, 0x4371,15, 100 }, /* EF.C.ICC.RFU3 */
|
|
|
|
|
/* */ /* (second cert for auth key) */
|
|
|
|
|
|
|
|
|
|
{ 0, 0x45B2, 3, 0, 0xC201, 0,0,1, 0x83}, /* EF_PK.NKS.ENC1024 */
|
|
|
|
|
/* */ /* nks15: EF.PK.ICC.ENC2 */
|
2009-03-26 19:27:04 +00:00
|
|
|
|
{ 0, 0xC201, 3, 101 }, /* EF_C.NKS.ENC1024 */
|
2020-04-17 16:09:05 +02:00
|
|
|
|
|
|
|
|
|
{ 0, 0xC20E,15, 111 }, /* EF.C.CSP.RCA1 (RootCA 1) */
|
|
|
|
|
{ 0, 0xC208,15, 101 }, /* EF.C.CSP.SCA1 (SubCA 1) */
|
|
|
|
|
{ 0, 0xC10E,15, 111 }, /* EF.C.CSP.RCA2 (RootCA 2) */
|
|
|
|
|
{ 0, 0xC108,15, 101 }, /* EF.C.CSP.SCA2 (SubCA 2) */
|
|
|
|
|
|
|
|
|
|
{ 0, 0x4571,15, 0, 0xC500, 1,0,0, 0x82}, /* EF.PK.ICC.AUT */
|
|
|
|
|
{ 0, 0xC500,15, 101 }, /* EF.C.ICC.AUT (Cert-auth) */
|
|
|
|
|
|
|
|
|
|
{ 0, 0xC201,15, 101 }, /* EF.C.ICC.ENC2 (Cert-encr) */
|
|
|
|
|
/* (empty on delivery) */
|
|
|
|
|
|
|
|
|
|
{ 1, 0x4531, 3, 0, 0xC000, 0,1,1, 0x84}, /* EF_PK.CH.SIG */
|
2009-03-30 12:46:06 +00:00
|
|
|
|
{ 1, 0xC000, 0, 101 }, /* EF_C.CH.SIG */
|
2020-04-17 16:09:05 +02:00
|
|
|
|
|
2009-03-30 12:46:06 +00:00
|
|
|
|
{ 1, 0xC008, 3, 101 }, /* EF_C.CA.SIG */
|
|
|
|
|
{ 1, 0xC00E, 3, 111 }, /* EF_C.RCA.SIG */
|
2020-04-17 16:09:05 +02:00
|
|
|
|
|
2020-06-30 14:41:20 +02:00
|
|
|
|
{ 2, 0x4531, 15, 0, 0xC001, 0,1,1, 0x84}, /* EF_PK.CH.SIG */
|
2020-04-17 16:09:05 +02:00
|
|
|
|
{ 2, 0xC000, 15,101 }, /* EF.C.SCA.QES (SubCA) */
|
|
|
|
|
{ 2, 0xC001, 15,100 }, /* EF.C.ICC.QES (Cert) */
|
|
|
|
|
{ 2, 0xC00E, 15,111 }, /* EF.C.RCA.QES (RootCA */
|
2020-05-07 19:44:26 +02:00
|
|
|
|
|
|
|
|
|
{ 3, 0x4E03, 3, 100 }, /* EK_PK_03 */
|
|
|
|
|
{ 3, 0x4E04, 3, 100 }, /* EK_PK_04 */
|
|
|
|
|
{ 3, 0x4E05, 3, 100 }, /* EK_PK_05 */
|
|
|
|
|
{ 3, 0x4E06, 3, 100 }, /* EK_PK_06 */
|
|
|
|
|
{ 3, 0x4E07, 3, 100 }, /* EK_PK_07 */
|
|
|
|
|
{ 3, 0x4E08, 3, 100 }, /* EK_PK_08 */
|
|
|
|
|
|
2009-03-05 19:19:37 +00:00
|
|
|
|
{ 0, 0 }
|
2004-01-27 16:40:42 +00:00
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
2020-07-02 18:35:34 +02:00
|
|
|
|
/* Object to cache information gathered from FIDs. */
|
2020-05-04 19:01:16 +02:00
|
|
|
|
struct fid_cache_s {
|
|
|
|
|
struct fid_cache_s *next;
|
|
|
|
|
int fid; /* Zero for an unused slot. */
|
2020-05-07 08:18:28 +02:00
|
|
|
|
unsigned int got_keygrip:1; /* The keygrip and algo are valid. */
|
|
|
|
|
int algo;
|
2020-09-21 14:47:53 +02:00
|
|
|
|
char *algostr; /* malloced. */
|
2020-05-04 19:01:16 +02:00
|
|
|
|
char keygripstr[2*KEYGRIP_LEN+1];
|
|
|
|
|
};
|
|
|
|
|
|
2004-01-27 16:40:42 +00:00
|
|
|
|
|
2009-01-08 19:56:30 +00:00
|
|
|
|
/* Object with application (i.e. NKS) specific data. */
|
|
|
|
|
struct app_local_s {
|
2020-04-17 16:09:05 +02:00
|
|
|
|
int active_nks_app; /* One of the NKS_APP_ constants. */
|
|
|
|
|
|
2020-05-07 19:44:26 +02:00
|
|
|
|
int only_idlm; /* The application is fixed to IDLM (IDKey card). */
|
2020-04-17 16:09:05 +02:00
|
|
|
|
int qes_app_id; /* Either NKS_APP_SIGG or NKS_APP_ESIGN. */
|
|
|
|
|
|
2020-05-07 19:44:26 +02:00
|
|
|
|
int sigg_msig_checked;/* True if we checked for a mass signature card. */
|
2020-04-17 16:09:05 +02:00
|
|
|
|
int sigg_is_msig; /* True if this is a mass signature card. */
|
2009-05-08 15:07:45 +00:00
|
|
|
|
|
2020-04-17 16:09:05 +02:00
|
|
|
|
int need_app_select; /* Need to re-select the application. */
|
2009-05-08 15:07:45 +00:00
|
|
|
|
|
2020-05-04 19:01:16 +02:00
|
|
|
|
struct fid_cache_s *fid_cache; /* Linked list with cached infos. */
|
2009-01-08 19:56:30 +00:00
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
2009-03-05 19:19:37 +00:00
|
|
|
|
|
2020-04-17 16:09:05 +02:00
|
|
|
|
static gpg_error_t readcert_from_ef (app_t app, int fid,
|
|
|
|
|
unsigned char **cert, size_t *certlen);
|
|
|
|
|
static gpg_error_t switch_application (app_t app, int nks_app_id);
|
2020-07-02 18:35:34 +02:00
|
|
|
|
static const char *parse_pwidstr (app_t app, const char *pwidstr, int new_mode,
|
|
|
|
|
int *r_nks_app_id, int *r_pwid);
|
|
|
|
|
static gpg_error_t verify_pin (app_t app, int pwid, const char *desc,
|
|
|
|
|
gpg_error_t (*pincb)(void*, const char *,
|
|
|
|
|
char **),
|
|
|
|
|
void *pincb_arg);
|
2009-03-05 19:19:37 +00:00
|
|
|
|
|
2009-01-08 19:56:30 +00:00
|
|
|
|
|
|
|
|
|
|
2020-07-02 18:35:34 +02:00
|
|
|
|
static void
|
|
|
|
|
flush_fid_cache (app_t app)
|
|
|
|
|
{
|
|
|
|
|
while (app->app_local->fid_cache)
|
|
|
|
|
{
|
|
|
|
|
struct fid_cache_s *next = app->app_local->fid_cache->next;
|
2020-09-21 14:47:53 +02:00
|
|
|
|
if (app->app_local->fid_cache)
|
|
|
|
|
xfree (app->app_local->fid_cache->algostr);
|
2020-07-02 18:35:34 +02:00
|
|
|
|
xfree (app->app_local->fid_cache);
|
|
|
|
|
app->app_local->fid_cache = next;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2009-01-08 19:56:30 +00:00
|
|
|
|
/* Release local data. */
|
|
|
|
|
static void
|
|
|
|
|
do_deinit (app_t app)
|
|
|
|
|
{
|
|
|
|
|
if (app && app->app_local)
|
|
|
|
|
{
|
2020-07-02 18:35:34 +02:00
|
|
|
|
flush_fid_cache (app);
|
2009-01-08 19:56:30 +00:00
|
|
|
|
xfree (app->app_local);
|
|
|
|
|
app->app_local = NULL;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2009-05-08 15:07:45 +00:00
|
|
|
|
static int
|
|
|
|
|
all_zero_p (void *buffer, size_t length)
|
|
|
|
|
{
|
|
|
|
|
char *p;
|
|
|
|
|
|
|
|
|
|
for (p=buffer; length; length--, p++)
|
|
|
|
|
if (*p)
|
|
|
|
|
return 0;
|
|
|
|
|
return 1;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2020-05-07 14:03:25 +02:00
|
|
|
|
/* Return an allocated string with the serial number in a format to be
|
|
|
|
|
* show to the user. May return NULL on malloc problem. */
|
|
|
|
|
static char *
|
|
|
|
|
get_dispserialno (app_t app)
|
|
|
|
|
{
|
|
|
|
|
char *result;
|
|
|
|
|
|
|
|
|
|
/* We only need to strip the last zero which is not printed on the
|
|
|
|
|
* card. */
|
|
|
|
|
result = app_get_serialno (app);
|
|
|
|
|
if (result && *result && result[strlen(result)-1] == '0')
|
|
|
|
|
result[strlen(result)-1] = 0;
|
|
|
|
|
return result;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2020-04-17 16:09:05 +02:00
|
|
|
|
/* Read the file with PKFID, assume it contains a public key and
|
|
|
|
|
* return its keygrip in the caller provided 41 byte buffer R_GRIPSTR.
|
|
|
|
|
* This works only for RSA card. For the Signature Card v2 ECC is
|
|
|
|
|
* used and Read Record needs to be replaced by read binary. Given
|
|
|
|
|
* all the ECC parameters required, we don't do that but rely that the
|
|
|
|
|
* corresponding certificate at CFID is already available and get the
|
2020-05-07 08:18:28 +02:00
|
|
|
|
* public key from there. If R_ALGO is not NULL the public key
|
2020-09-21 14:47:53 +02:00
|
|
|
|
* algorithm for the returned KEYGRIP is stored there. If R_ALGOSTR
|
|
|
|
|
* is not NULL the public key algo string (e.g. "rsa2048") is stored
|
|
|
|
|
* there. */
|
2004-01-27 16:40:42 +00:00
|
|
|
|
static gpg_error_t
|
2020-05-07 08:18:28 +02:00
|
|
|
|
keygripstr_from_pk_file (app_t app, int pkfid, int cfid, char *r_gripstr,
|
2020-09-21 14:47:53 +02:00
|
|
|
|
int *r_algo, char **r_algostr)
|
2004-01-27 16:40:42 +00:00
|
|
|
|
{
|
|
|
|
|
gpg_error_t err;
|
|
|
|
|
unsigned char grip[20];
|
|
|
|
|
unsigned char *buffer[2];
|
|
|
|
|
size_t buflen[2];
|
2020-05-04 19:01:16 +02:00
|
|
|
|
gcry_sexp_t sexp = NULL;
|
2020-05-07 08:18:28 +02:00
|
|
|
|
int algo = 0; /* Public key algo. */
|
2020-09-21 14:47:53 +02:00
|
|
|
|
char *algostr = NULL; /* Public key algo string. */
|
2004-01-27 16:40:42 +00:00
|
|
|
|
int i;
|
2009-03-20 19:04:47 +00:00
|
|
|
|
int offset[2] = { 0, 0 };
|
2020-05-04 19:01:16 +02:00
|
|
|
|
struct fid_cache_s *ci;
|
|
|
|
|
|
|
|
|
|
for (ci = app->app_local->fid_cache; ci; ci = ci->next)
|
|
|
|
|
if (ci->fid && ci->fid == pkfid)
|
|
|
|
|
{
|
|
|
|
|
if (!ci->got_keygrip)
|
|
|
|
|
return gpg_error (GPG_ERR_NOT_FOUND);
|
2020-09-21 14:47:53 +02:00
|
|
|
|
if (r_algostr && !ci->algostr)
|
|
|
|
|
break; /* Not in the cache - try w/o cache. */
|
2020-05-04 19:01:16 +02:00
|
|
|
|
memcpy (r_gripstr, ci->keygripstr, 2*KEYGRIP_LEN+1);
|
2020-05-07 08:18:28 +02:00
|
|
|
|
if (r_algo)
|
|
|
|
|
*r_algo = ci->algo;
|
2020-09-21 14:47:53 +02:00
|
|
|
|
if (r_algostr)
|
|
|
|
|
{
|
|
|
|
|
*r_algostr = xtrystrdup (ci->algostr);
|
|
|
|
|
if (!*r_algostr)
|
|
|
|
|
return gpg_error_from_syserror ();
|
|
|
|
|
}
|
2020-05-04 19:01:16 +02:00
|
|
|
|
return 0; /* Found in cache. */
|
|
|
|
|
}
|
2009-03-20 19:04:47 +00:00
|
|
|
|
|
2020-05-07 19:44:26 +02:00
|
|
|
|
if (app->appversion == 15)
|
2020-04-17 16:09:05 +02:00
|
|
|
|
{
|
|
|
|
|
/* Signature Card v2 - get keygrip from the certificate. */
|
|
|
|
|
unsigned char *cert, *pk;
|
|
|
|
|
size_t certlen, pklen;
|
|
|
|
|
|
|
|
|
|
/* Fall back to certificate reading. */
|
|
|
|
|
err = readcert_from_ef (app, cfid, &cert, &certlen);
|
|
|
|
|
if (err)
|
|
|
|
|
{
|
|
|
|
|
log_error ("nks: error reading certificate %04X: %s\n",
|
|
|
|
|
cfid, gpg_strerror (err));
|
2020-09-21 14:47:53 +02:00
|
|
|
|
goto leave;
|
2020-04-17 16:09:05 +02:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
err = app_help_pubkey_from_cert (cert, certlen, &pk, &pklen);
|
|
|
|
|
xfree (cert);
|
|
|
|
|
if (err)
|
|
|
|
|
{
|
|
|
|
|
log_error ("nks: error parsing certificate %04X: %s\n",
|
|
|
|
|
cfid, gpg_strerror (err));
|
2020-09-21 14:47:53 +02:00
|
|
|
|
goto leave;
|
2020-04-17 16:09:05 +02:00
|
|
|
|
}
|
|
|
|
|
|
2020-09-21 14:47:53 +02:00
|
|
|
|
err = app_help_get_keygrip_string_pk (pk, pklen, r_gripstr, NULL,
|
|
|
|
|
&algo, &algostr);
|
2020-04-17 16:09:05 +02:00
|
|
|
|
xfree (pk);
|
|
|
|
|
if (err)
|
|
|
|
|
log_error ("nks: error getting keygrip for certificate %04X: %s\n",
|
|
|
|
|
cfid, gpg_strerror (err));
|
|
|
|
|
|
2020-05-04 19:01:16 +02:00
|
|
|
|
goto leave;
|
2020-04-17 16:09:05 +02:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
err = iso7816_select_file (app_get_slot (app), pkfid, 0);
|
2004-01-27 16:40:42 +00:00
|
|
|
|
if (err)
|
2020-09-21 14:47:53 +02:00
|
|
|
|
goto leave;
|
2019-06-19 08:50:40 +02:00
|
|
|
|
err = iso7816_read_record (app_get_slot (app), 1, 1, 0,
|
|
|
|
|
&buffer[0], &buflen[0]);
|
2004-01-27 16:40:42 +00:00
|
|
|
|
if (err)
|
2020-09-21 14:47:53 +02:00
|
|
|
|
goto leave;
|
2019-06-19 08:50:40 +02:00
|
|
|
|
err = iso7816_read_record (app_get_slot (app), 2, 1, 0,
|
|
|
|
|
&buffer[1], &buflen[1]);
|
2004-01-27 16:40:42 +00:00
|
|
|
|
if (err)
|
|
|
|
|
{
|
|
|
|
|
xfree (buffer[0]);
|
2020-09-21 14:47:53 +02:00
|
|
|
|
goto leave;
|
2004-01-27 16:40:42 +00:00
|
|
|
|
}
|
2009-03-20 19:04:47 +00:00
|
|
|
|
|
2020-05-07 19:44:26 +02:00
|
|
|
|
if (app->appversion < 3)
|
2004-01-27 16:40:42 +00:00
|
|
|
|
{
|
2009-01-08 19:56:30 +00:00
|
|
|
|
/* Old versions of NKS store the values in a TLV encoded format.
|
|
|
|
|
We need to do some checks. */
|
|
|
|
|
for (i=0; i < 2; i++)
|
|
|
|
|
{
|
|
|
|
|
/* Check that the value appears like an integer encoded as
|
|
|
|
|
Simple-TLV. We don't check the tag because the tests cards I
|
|
|
|
|
have use 1 for both, the modulus and the exponent - the
|
|
|
|
|
example in the documentation gives 2 for the exponent. */
|
|
|
|
|
if (buflen[i] < 3)
|
|
|
|
|
err = gpg_error (GPG_ERR_TOO_SHORT);
|
|
|
|
|
else if (buffer[i][1] != buflen[i]-2 )
|
|
|
|
|
err = gpg_error (GPG_ERR_INV_OBJ);
|
2009-03-20 19:04:47 +00:00
|
|
|
|
else
|
|
|
|
|
offset[i] = 2;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
/* Remove leading zeroes to get a correct keygrip. Take care of
|
|
|
|
|
negative numbers. We should also fix it the same way in
|
|
|
|
|
libgcrypt but we can't yet rely on it yet. */
|
|
|
|
|
for (i=0; i < 2; i++)
|
|
|
|
|
{
|
2011-02-04 12:57:53 +01:00
|
|
|
|
while (buflen[i]-offset[i] > 1
|
|
|
|
|
&& !buffer[i][offset[i]]
|
2009-03-20 19:04:47 +00:00
|
|
|
|
&& !(buffer[i][offset[i]+1] & 0x80))
|
|
|
|
|
offset[i]++;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Check whether negative values are not prefixed with a zero and
|
|
|
|
|
fix that. */
|
|
|
|
|
for (i=0; i < 2; i++)
|
|
|
|
|
{
|
|
|
|
|
if ((buflen[i]-offset[i]) && (buffer[i][offset[i]] & 0x80))
|
|
|
|
|
{
|
2011-02-04 12:57:53 +01:00
|
|
|
|
unsigned char *newbuf;
|
2009-03-20 19:04:47 +00:00
|
|
|
|
size_t newlen;
|
2011-02-04 12:57:53 +01:00
|
|
|
|
|
2009-03-20 19:04:47 +00:00
|
|
|
|
newlen = 1 + buflen[i] - offset[i];
|
|
|
|
|
newbuf = xtrymalloc (newlen);
|
|
|
|
|
if (!newlen)
|
|
|
|
|
{
|
|
|
|
|
xfree (buffer[0]);
|
|
|
|
|
xfree (buffer[1]);
|
2020-09-21 14:47:53 +02:00
|
|
|
|
err = gpg_error_from_syserror ();
|
|
|
|
|
goto leave;
|
2009-03-20 19:04:47 +00:00
|
|
|
|
}
|
|
|
|
|
newbuf[0] = 0;
|
|
|
|
|
memcpy (newbuf+1, buffer[i]+offset[i], buflen[i] - offset[i]);
|
|
|
|
|
xfree (buffer[i]);
|
|
|
|
|
buffer[i] = newbuf;
|
|
|
|
|
buflen[i] = newlen;
|
|
|
|
|
offset[i] = 0;
|
2009-01-08 19:56:30 +00:00
|
|
|
|
}
|
2004-01-27 16:40:42 +00:00
|
|
|
|
}
|
|
|
|
|
|
2020-05-07 08:18:28 +02:00
|
|
|
|
algo = GCRY_PK_RSA;
|
2004-01-27 16:40:42 +00:00
|
|
|
|
if (!err)
|
|
|
|
|
err = gcry_sexp_build (&sexp, NULL,
|
|
|
|
|
"(public-key (rsa (n %b) (e %b)))",
|
2009-03-20 19:04:47 +00:00
|
|
|
|
(int)buflen[0]-offset[0], buffer[0]+offset[0],
|
|
|
|
|
(int)buflen[1]-offset[1], buffer[1]+offset[1]);
|
2004-01-27 16:40:42 +00:00
|
|
|
|
|
|
|
|
|
xfree (buffer[0]);
|
|
|
|
|
xfree (buffer[1]);
|
|
|
|
|
if (err)
|
|
|
|
|
return err;
|
|
|
|
|
|
|
|
|
|
if (!gcry_pk_get_keygrip (sexp, grip))
|
|
|
|
|
{
|
|
|
|
|
err = gpg_error (GPG_ERR_INTERNAL); /* i.e. RSA not supported by
|
|
|
|
|
libgcrypt. */
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
2008-11-03 10:54:18 +00:00
|
|
|
|
bin2hex (grip, 20, r_gripstr);
|
2020-05-07 08:18:28 +02:00
|
|
|
|
if (r_algo)
|
|
|
|
|
*r_algo = algo;
|
2020-11-25 10:32:28 +09:00
|
|
|
|
algostr = pubkey_algo_string (sexp, NULL);
|
2004-01-27 16:40:42 +00:00
|
|
|
|
}
|
2020-05-04 19:01:16 +02:00
|
|
|
|
|
|
|
|
|
|
|
|
|
|
leave:
|
|
|
|
|
if (!err)
|
|
|
|
|
{
|
2020-09-21 14:47:53 +02:00
|
|
|
|
if (r_algostr)
|
|
|
|
|
{
|
|
|
|
|
*r_algostr = algostr;
|
|
|
|
|
algostr = NULL;
|
|
|
|
|
}
|
|
|
|
|
|
2020-05-04 19:01:16 +02:00
|
|
|
|
/* FIXME: We need to implement not_found caching. */
|
|
|
|
|
for (ci = app->app_local->fid_cache; ci; ci = ci->next)
|
|
|
|
|
if (ci->fid && ci->fid == pkfid)
|
|
|
|
|
{
|
|
|
|
|
/* Update the keygrip. */
|
|
|
|
|
memcpy (ci->keygripstr, r_gripstr, 2*KEYGRIP_LEN+1);
|
2020-05-07 08:18:28 +02:00
|
|
|
|
ci->algo = algo;
|
2020-09-21 14:47:53 +02:00
|
|
|
|
xfree (ci->algostr);
|
|
|
|
|
ci->algostr = algostr? xtrystrdup (algostr) : NULL;
|
2020-05-07 08:18:28 +02:00
|
|
|
|
ci->got_keygrip = 1;
|
2020-05-04 19:01:16 +02:00
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
if (!ci)
|
|
|
|
|
{
|
|
|
|
|
for (ci = app->app_local->fid_cache; ci; ci = ci->next)
|
|
|
|
|
if (!ci->fid)
|
|
|
|
|
break;
|
|
|
|
|
if (!ci)
|
|
|
|
|
ci = xtrycalloc (1, sizeof *ci);
|
|
|
|
|
if (!ci)
|
|
|
|
|
; /* Out of memory - it is a cache, so we ignore it. */
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
ci->fid = pkfid;
|
|
|
|
|
memcpy (ci->keygripstr, r_gripstr, 2*KEYGRIP_LEN+1);
|
2020-05-07 08:18:28 +02:00
|
|
|
|
ci->algo = algo;
|
2020-05-04 19:01:16 +02:00
|
|
|
|
ci->got_keygrip = 1;
|
|
|
|
|
ci->next = app->app_local->fid_cache;
|
|
|
|
|
app->app_local->fid_cache = ci;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
2004-01-27 16:40:42 +00:00
|
|
|
|
gcry_sexp_release (sexp);
|
2020-09-21 14:47:53 +02:00
|
|
|
|
xfree (algostr);
|
2004-01-27 16:40:42 +00:00
|
|
|
|
return err;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2020-05-04 19:01:16 +02:00
|
|
|
|
/* Parse KEYREF and return the index into the FILELIST at R_IDX.
|
2020-05-07 08:18:28 +02:00
|
|
|
|
* Returns 0 on success and switches to the requested application.
|
|
|
|
|
* The public key algo is stored at R_ALGO unless it is NULL. */
|
2020-05-04 19:01:16 +02:00
|
|
|
|
static gpg_error_t
|
2020-05-07 08:18:28 +02:00
|
|
|
|
find_fid_by_keyref (app_t app, const char *keyref, int *r_idx, int *r_algo)
|
2020-05-04 19:01:16 +02:00
|
|
|
|
{
|
|
|
|
|
gpg_error_t err;
|
|
|
|
|
int idx, fid, nks_app_id;
|
|
|
|
|
char keygripstr[2*KEYGRIP_LEN+1];
|
|
|
|
|
|
|
|
|
|
if (!keyref || !keyref[0])
|
|
|
|
|
err = gpg_error (GPG_ERR_INV_ID);
|
|
|
|
|
else if (keyref[0] != 'N' && strlen (keyref) == 40) /* This is a keygrip. */
|
|
|
|
|
{
|
|
|
|
|
struct fid_cache_s *ci;
|
|
|
|
|
|
|
|
|
|
for (ci = app->app_local->fid_cache; ci; ci = ci->next)
|
2020-05-07 08:18:28 +02:00
|
|
|
|
if (ci->fid && ci->got_keygrip && !strcmp (ci->keygripstr, keyref))
|
2020-05-04 19:01:16 +02:00
|
|
|
|
break;
|
|
|
|
|
if (ci) /* Cached */
|
|
|
|
|
{
|
|
|
|
|
for (idx=0; filelist[idx].fid; idx++)
|
|
|
|
|
if (filelist[idx].fid == ci->fid)
|
|
|
|
|
break;
|
|
|
|
|
if (!filelist[idx].fid)
|
|
|
|
|
{
|
|
|
|
|
log_debug ("nks: Ooops: Unkown FID cached!\n");
|
|
|
|
|
err = gpg_error (GPG_ERR_BUG);
|
|
|
|
|
goto leave;
|
|
|
|
|
}
|
|
|
|
|
err = switch_application (app, filelist[idx].nks_app_id);
|
|
|
|
|
if (err)
|
|
|
|
|
goto leave;
|
2020-05-07 08:18:28 +02:00
|
|
|
|
if (r_algo)
|
|
|
|
|
*r_algo = ci->algo;
|
2020-05-04 19:01:16 +02:00
|
|
|
|
}
|
|
|
|
|
else /* Not cached. */
|
|
|
|
|
{
|
|
|
|
|
for (idx=0; filelist[idx].fid; idx++)
|
|
|
|
|
{
|
|
|
|
|
if (!filelist[idx].iskeypair)
|
|
|
|
|
continue;
|
|
|
|
|
|
2020-05-07 19:44:26 +02:00
|
|
|
|
if (app->app_local->only_idlm)
|
|
|
|
|
{
|
|
|
|
|
if (filelist[idx].nks_app_id != NKS_APP_IDLM)
|
|
|
|
|
continue;
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
if (filelist[idx].nks_app_id != NKS_APP_NKS
|
|
|
|
|
&& filelist[idx].nks_app_id != app->app_local->qes_app_id)
|
|
|
|
|
continue;
|
|
|
|
|
|
|
|
|
|
err = switch_application (app, filelist[idx].nks_app_id);
|
|
|
|
|
if (err)
|
|
|
|
|
goto leave;
|
|
|
|
|
}
|
2020-05-04 19:01:16 +02:00
|
|
|
|
|
|
|
|
|
err = keygripstr_from_pk_file (app, filelist[idx].fid,
|
|
|
|
|
filelist[idx].iskeypair,
|
2020-09-21 14:47:53 +02:00
|
|
|
|
keygripstr, r_algo, NULL);
|
2020-05-04 19:01:16 +02:00
|
|
|
|
if (err)
|
|
|
|
|
{
|
2020-05-07 08:18:28 +02:00
|
|
|
|
log_info ("nks: no keygrip for FID 0x%04X: %s - ignored\n",
|
2020-05-04 19:01:16 +02:00
|
|
|
|
filelist[idx].fid, gpg_strerror (err));
|
|
|
|
|
continue;
|
|
|
|
|
}
|
|
|
|
|
if (!strcmp (keygripstr, keyref))
|
|
|
|
|
break; /* Found */
|
|
|
|
|
}
|
|
|
|
|
if (!filelist[idx].fid)
|
|
|
|
|
{
|
|
|
|
|
err = gpg_error (GPG_ERR_NOT_FOUND);
|
|
|
|
|
goto leave;
|
|
|
|
|
}
|
|
|
|
|
/* (No need to switch the app as that has already been done
|
|
|
|
|
* in the loop.) */
|
|
|
|
|
}
|
|
|
|
|
*r_idx = idx;
|
|
|
|
|
err = 0;
|
|
|
|
|
}
|
|
|
|
|
else /* This is a usual keyref. */
|
|
|
|
|
{
|
|
|
|
|
if (!ascii_strncasecmp (keyref, "NKS-NKS3.", 9))
|
|
|
|
|
nks_app_id = NKS_APP_NKS;
|
|
|
|
|
else if (!ascii_strncasecmp (keyref, "NKS-ESIGN.", 10)
|
|
|
|
|
&& app->app_local->qes_app_id == NKS_APP_ESIGN)
|
|
|
|
|
nks_app_id = NKS_APP_ESIGN;
|
|
|
|
|
else if (!ascii_strncasecmp (keyref, "NKS-SIGG.", 9)
|
|
|
|
|
&& app->app_local->qes_app_id == NKS_APP_SIGG)
|
|
|
|
|
nks_app_id = NKS_APP_SIGG;
|
2020-05-07 19:44:26 +02:00
|
|
|
|
else if (!ascii_strncasecmp (keyref, "NKS-IDLM.", 9))
|
|
|
|
|
nks_app_id = NKS_APP_IDLM;
|
2020-05-04 19:01:16 +02:00
|
|
|
|
else if (!ascii_strncasecmp (keyref, "NKS-DF01.", 9))
|
|
|
|
|
nks_app_id = NKS_APP_NKS;
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
err = gpg_error (GPG_ERR_INV_ID);
|
|
|
|
|
goto leave;
|
|
|
|
|
}
|
|
|
|
|
keyref += nks_app_id == NKS_APP_ESIGN? 10 : 9;
|
|
|
|
|
|
|
|
|
|
if (!hexdigitp (keyref) || !hexdigitp (keyref+1)
|
|
|
|
|
|| !hexdigitp (keyref+2) || !hexdigitp (keyref+3)
|
|
|
|
|
|| keyref[4])
|
|
|
|
|
{
|
|
|
|
|
err = gpg_error (GPG_ERR_INV_ID);
|
|
|
|
|
goto leave;
|
|
|
|
|
}
|
|
|
|
|
fid = xtoi_4 (keyref);
|
|
|
|
|
for (idx=0; filelist[idx].fid; idx++)
|
|
|
|
|
if (filelist[idx].iskeypair && filelist[idx].fid == fid
|
|
|
|
|
&& filelist[idx].nks_app_id == nks_app_id)
|
|
|
|
|
break;
|
|
|
|
|
if (!filelist[idx].fid)
|
|
|
|
|
{
|
|
|
|
|
err = gpg_error (GPG_ERR_NOT_FOUND);
|
|
|
|
|
goto leave;
|
|
|
|
|
}
|
|
|
|
|
*r_idx = idx;
|
|
|
|
|
|
|
|
|
|
err = switch_application (app, nks_app_id);
|
|
|
|
|
if (err)
|
|
|
|
|
goto leave;
|
2020-05-07 08:18:28 +02:00
|
|
|
|
if (r_algo)
|
|
|
|
|
{
|
|
|
|
|
/* We need to get the public key algo. */
|
|
|
|
|
err = keygripstr_from_pk_file (app, filelist[idx].fid,
|
|
|
|
|
filelist[idx].iskeypair,
|
2020-09-21 14:47:53 +02:00
|
|
|
|
keygripstr, r_algo, NULL);
|
2020-05-07 08:18:28 +02:00
|
|
|
|
if (err)
|
|
|
|
|
log_error ("nks: no keygrip for FID 0x%04X: %s\n",
|
|
|
|
|
filelist[idx].fid, gpg_strerror (err));
|
|
|
|
|
}
|
2020-05-04 19:01:16 +02:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
leave:
|
|
|
|
|
return err;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2009-03-05 19:19:37 +00:00
|
|
|
|
/* TCOS responds to a verify with empty data (i.e. without the Lc
|
2020-04-17 16:09:05 +02:00
|
|
|
|
* byte) with the status of the PIN. PWID is the PIN ID. NKS_APP_ID
|
|
|
|
|
* gives the application to first switch to. Returns:
|
2020-04-07 18:25:41 +02:00
|
|
|
|
* ISO7816_VERIFY_* codes or non-negative number of verification
|
|
|
|
|
* attempts left. */
|
2009-03-05 19:19:37 +00:00
|
|
|
|
static int
|
2020-04-17 16:09:05 +02:00
|
|
|
|
get_chv_status (app_t app, int nks_app_id, int pwid)
|
2009-03-05 19:19:37 +00:00
|
|
|
|
{
|
2020-04-17 16:09:05 +02:00
|
|
|
|
if (switch_application (app, nks_app_id))
|
|
|
|
|
return (nks_app_id == NKS_APP_NKS
|
|
|
|
|
? ISO7816_VERIFY_ERROR
|
|
|
|
|
: ISO7816_VERIFY_NO_PIN);
|
2009-03-05 19:19:37 +00:00
|
|
|
|
|
2020-04-07 18:25:41 +02:00
|
|
|
|
return iso7816_verify_status (app_get_slot (app), pwid);
|
2009-03-05 19:19:37 +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
|
2009-03-05 19:19:37 +00:00
|
|
|
|
do_getattr (app_t app, ctrl_t ctrl, const char *name)
|
|
|
|
|
{
|
|
|
|
|
static struct {
|
|
|
|
|
const char *name;
|
|
|
|
|
int special;
|
|
|
|
|
} table[] = {
|
|
|
|
|
{ "$AUTHKEYID", 1 },
|
2019-08-21 14:18:43 +02:00
|
|
|
|
{ "$ENCRKEYID", 2 },
|
|
|
|
|
{ "$SIGNKEYID", 3 },
|
2020-05-07 19:44:26 +02:00
|
|
|
|
{ "NKS-VERSION", 4 }, /* Legacy (printed decimal) */
|
2019-08-21 14:18:43 +02:00
|
|
|
|
{ "CHV-STATUS", 5 },
|
2020-05-07 14:03:25 +02:00
|
|
|
|
{ "$DISPSERIALNO",6 },
|
|
|
|
|
{ "SERIALNO", 0 }
|
2009-03-05 19:19:37 +00:00
|
|
|
|
};
|
|
|
|
|
gpg_error_t err = 0;
|
|
|
|
|
int idx;
|
2020-05-07 14:03:25 +02:00
|
|
|
|
char *p, *p2;
|
2009-03-05 19:19:37 +00:00
|
|
|
|
char buffer[100];
|
2020-05-07 19:44:26 +02:00
|
|
|
|
int nksver = app->appversion;
|
2009-03-05 19:19:37 +00:00
|
|
|
|
|
2020-04-17 16:09:05 +02:00
|
|
|
|
err = switch_application (app, NKS_APP_NKS);
|
2009-03-05 19:19:37 +00:00
|
|
|
|
if (err)
|
|
|
|
|
return err;
|
|
|
|
|
|
2020-05-07 14:03:25 +02:00
|
|
|
|
for (idx=0; (idx < DIM(table)
|
|
|
|
|
&& ascii_strcasecmp (table[idx].name, name)); idx++)
|
2009-03-05 19:19:37 +00:00
|
|
|
|
;
|
2020-05-07 14:03:25 +02:00
|
|
|
|
if (!(idx < DIM (table)))
|
2011-02-04 12:57:53 +01:00
|
|
|
|
return gpg_error (GPG_ERR_INV_NAME);
|
2009-03-05 19:19:37 +00:00
|
|
|
|
|
|
|
|
|
switch (table[idx].special)
|
|
|
|
|
{
|
2020-05-07 14:03:25 +02:00
|
|
|
|
case 0: /* SERIALNO */
|
|
|
|
|
{
|
|
|
|
|
p = app_get_serialno (app);
|
|
|
|
|
if (p)
|
|
|
|
|
{
|
|
|
|
|
send_status_direct (ctrl, "SERIALNO", p);
|
|
|
|
|
xfree (p);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
|
2009-03-05 19:19:37 +00:00
|
|
|
|
case 1: /* $AUTHKEYID */
|
|
|
|
|
{
|
2009-03-26 19:27:04 +00:00
|
|
|
|
/* NetKey 3.0 cards define an authentication key but according
|
|
|
|
|
to the specs this key is only usable for encryption and not
|
|
|
|
|
signing. it might work anyway but it has not yet been
|
|
|
|
|
tested - fixme. Thus for now we use the NKS signature key
|
2020-04-17 16:09:05 +02:00
|
|
|
|
for authentication for netkey 3. For the Signature Card
|
|
|
|
|
V2.0 the auth key is defined and thus we use it. */
|
|
|
|
|
const char *tmp = nksver == 15? "NKS-NKS3.4571" : "NKS-NKS3.4531";
|
2009-03-05 19:19:37 +00:00
|
|
|
|
send_status_info (ctrl, table[idx].name, tmp, strlen (tmp), NULL, 0);
|
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
|
2019-08-21 14:18:43 +02:00
|
|
|
|
case 2: /* $ENCRKEYID */
|
|
|
|
|
{
|
|
|
|
|
char const tmp[] = "NKS-NKS3.45B1";
|
|
|
|
|
send_status_info (ctrl, table[idx].name, tmp, strlen (tmp), NULL, 0);
|
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case 3: /* $SIGNKEYID */
|
|
|
|
|
{
|
|
|
|
|
char const tmp[] = "NKS-NKS3.4531";
|
|
|
|
|
send_status_info (ctrl, table[idx].name, tmp, strlen (tmp), NULL, 0);
|
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case 4: /* NKS-VERSION */
|
2020-05-07 19:44:26 +02:00
|
|
|
|
snprintf (buffer, sizeof buffer, "%d", app->appversion);
|
2009-03-05 19:19:37 +00:00
|
|
|
|
send_status_info (ctrl, table[idx].name,
|
|
|
|
|
buffer, strlen (buffer), NULL, 0);
|
|
|
|
|
break;
|
|
|
|
|
|
2019-08-21 14:18:43 +02:00
|
|
|
|
case 5: /* CHV-STATUS */
|
2009-03-05 19:19:37 +00:00
|
|
|
|
{
|
2020-05-07 14:03:25 +02:00
|
|
|
|
/* Return the status for the the PINs as described in the
|
|
|
|
|
* table below. See the macros ISO7816_VERIFY_* for a list
|
|
|
|
|
* for each slot. The order is
|
|
|
|
|
*
|
|
|
|
|
* | idx | name |
|
|
|
|
|
* |-----+------------|
|
|
|
|
|
* | 0 | PW1.CH |
|
|
|
|
|
* | 1 | PW2.CH |
|
|
|
|
|
* | 2 | PW1.CH.SIG |
|
|
|
|
|
* | 3 | PW2.CH.SIG |
|
|
|
|
|
*
|
|
|
|
|
* See parse_pwidstr for details of the mapping.
|
|
|
|
|
*/
|
2009-03-05 19:19:37 +00:00
|
|
|
|
int tmp[4];
|
2011-02-04 12:57:53 +01:00
|
|
|
|
|
2009-03-05 19:19:37 +00:00
|
|
|
|
/* We use a helper array so that we can control that there is
|
2020-05-07 14:03:25 +02:00
|
|
|
|
* no superfluous application switches. */
|
2020-05-07 19:44:26 +02:00
|
|
|
|
if (app->appversion == 15)
|
2020-05-07 14:03:25 +02:00
|
|
|
|
{
|
|
|
|
|
tmp[0] = get_chv_status (app, 0, 0x03);
|
|
|
|
|
tmp[1] = get_chv_status (app, 0, 0x04);
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
tmp[0] = get_chv_status (app, 0, 0x00);
|
|
|
|
|
tmp[1] = get_chv_status (app, 0, 0x01);
|
|
|
|
|
}
|
2020-04-17 16:09:05 +02:00
|
|
|
|
tmp[2] = get_chv_status (app, app->app_local->qes_app_id, 0x81);
|
2020-05-07 19:44:26 +02:00
|
|
|
|
if (app->appversion == 15)
|
2020-05-07 14:03:25 +02:00
|
|
|
|
tmp[3] = get_chv_status (app, app->app_local->qes_app_id, 0x82);
|
|
|
|
|
else
|
|
|
|
|
tmp[3] = get_chv_status (app, app->app_local->qes_app_id, 0x83);
|
|
|
|
|
snprintf (buffer, sizeof buffer, "%d %d %d %d",
|
|
|
|
|
tmp[0], tmp[1], tmp[2], tmp[3]);
|
2009-03-05 19:19:37 +00:00
|
|
|
|
send_status_info (ctrl, table[idx].name,
|
|
|
|
|
buffer, strlen (buffer), NULL, 0);
|
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
|
2020-05-07 14:03:25 +02:00
|
|
|
|
case 6: /* $DISPSERIALNO */
|
|
|
|
|
{
|
|
|
|
|
p = app_get_serialno (app);
|
|
|
|
|
p2 = get_dispserialno (app);
|
|
|
|
|
if (p && p2 && strcmp (p, p2))
|
|
|
|
|
send_status_info (ctrl, table[idx].name, p2, strlen (p2),
|
|
|
|
|
NULL, (size_t)0);
|
|
|
|
|
else /* No abbreviated S/N or identical to the full full S/N. */
|
|
|
|
|
err = gpg_error (GPG_ERR_INV_NAME); /* No Abbreviated S/N. */
|
|
|
|
|
xfree (p);
|
|
|
|
|
xfree (p2);
|
|
|
|
|
}
|
|
|
|
|
break;
|
2009-03-05 19:19:37 +00:00
|
|
|
|
|
|
|
|
|
default:
|
|
|
|
|
err = gpg_error (GPG_ERR_NOT_IMPLEMENTED);
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return err;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
2009-03-06 17:31:27 +00:00
|
|
|
|
static void
|
2020-04-17 16:09:05 +02:00
|
|
|
|
do_learn_status_core (app_t app, ctrl_t ctrl, unsigned int flags,
|
|
|
|
|
int nks_app_id)
|
2004-01-27 16:40:42 +00:00
|
|
|
|
{
|
|
|
|
|
gpg_error_t err;
|
|
|
|
|
char ct_buf[100], id_buf[100];
|
|
|
|
|
int i;
|
2009-03-06 17:31:27 +00:00
|
|
|
|
const char *tag;
|
2004-01-27 16:40:42 +00:00
|
|
|
|
|
2020-04-17 16:09:05 +02:00
|
|
|
|
if (nks_app_id == NKS_APP_ESIGN)
|
|
|
|
|
tag = "ESIGN";
|
|
|
|
|
else if (nks_app_id == NKS_APP_SIGG)
|
2009-03-06 17:31:27 +00:00
|
|
|
|
tag = "SIGG";
|
2020-05-07 19:44:26 +02:00
|
|
|
|
else if (nks_app_id == NKS_APP_IDLM)
|
|
|
|
|
tag = "IDLM";
|
|
|
|
|
else if (app->appversion < 3)
|
2009-03-06 17:31:27 +00:00
|
|
|
|
tag = "DF01";
|
|
|
|
|
else
|
|
|
|
|
tag = "NKS3";
|
2009-03-05 19:19:37 +00:00
|
|
|
|
|
|
|
|
|
/* Output information about all useful objects in the NKS application. */
|
|
|
|
|
for (i=0; filelist[i].fid; i++)
|
|
|
|
|
{
|
2020-05-07 19:44:26 +02:00
|
|
|
|
if (filelist[i].nks_ver > app->appversion)
|
2009-03-05 19:19:37 +00:00
|
|
|
|
continue;
|
|
|
|
|
|
2020-04-17 16:09:05 +02:00
|
|
|
|
if (filelist[i].nks_app_id != nks_app_id)
|
2009-03-05 19:19:37 +00:00
|
|
|
|
continue;
|
|
|
|
|
|
2019-09-04 10:43:18 +02:00
|
|
|
|
if (filelist[i].certtype && !(flags & APP_LEARN_FLAG_KEYPAIRINFO))
|
2009-03-05 19:19:37 +00:00
|
|
|
|
{
|
|
|
|
|
size_t len;
|
|
|
|
|
|
2019-06-19 08:50:40 +02:00
|
|
|
|
len = app_help_read_length_of_cert (app_get_slot (app),
|
2009-03-05 19:19:37 +00:00
|
|
|
|
filelist[i].fid, NULL);
|
|
|
|
|
if (len)
|
|
|
|
|
{
|
|
|
|
|
/* FIXME: We should store the length in the application's
|
|
|
|
|
context so that a following readcert does only need to
|
|
|
|
|
read that many bytes. */
|
|
|
|
|
snprintf (ct_buf, sizeof ct_buf, "%d", filelist[i].certtype);
|
2011-02-04 12:57:53 +01:00
|
|
|
|
snprintf (id_buf, sizeof id_buf, "NKS-%s.%04X",
|
2009-03-06 17:31:27 +00:00
|
|
|
|
tag, filelist[i].fid);
|
2009-03-05 19:19:37 +00:00
|
|
|
|
send_status_info (ctrl, "CERTINFO",
|
2011-02-04 12:57:53 +01:00
|
|
|
|
ct_buf, strlen (ct_buf),
|
|
|
|
|
id_buf, strlen (id_buf),
|
2009-03-05 19:19:37 +00:00
|
|
|
|
NULL, (size_t)0);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
else if (filelist[i].iskeypair)
|
|
|
|
|
{
|
|
|
|
|
char gripstr[40+1];
|
2020-04-17 16:09:05 +02:00
|
|
|
|
char usagebuf[5];
|
|
|
|
|
int usageidx = 0;
|
2020-09-21 14:47:53 +02:00
|
|
|
|
char *algostr = NULL;
|
2009-03-05 19:19:37 +00:00
|
|
|
|
|
2020-04-17 16:09:05 +02:00
|
|
|
|
err = keygripstr_from_pk_file (app, filelist[i].fid,
|
2020-09-21 14:47:53 +02:00
|
|
|
|
filelist[i].iskeypair, gripstr,
|
|
|
|
|
NULL, &algostr);
|
2009-03-05 19:19:37 +00:00
|
|
|
|
if (err)
|
|
|
|
|
log_error ("can't get keygrip from FID 0x%04X: %s\n",
|
|
|
|
|
filelist[i].fid, gpg_strerror (err));
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
snprintf (id_buf, sizeof id_buf, "NKS-%s.%04X",
|
2009-03-06 17:31:27 +00:00
|
|
|
|
tag, filelist[i].fid);
|
2020-04-17 16:09:05 +02:00
|
|
|
|
if (filelist[i].issignkey)
|
|
|
|
|
usagebuf[usageidx++] = 's';
|
|
|
|
|
if (filelist[i].isauthkey)
|
|
|
|
|
usagebuf[usageidx++] = 'a';
|
|
|
|
|
if (filelist[i].isencrkey)
|
|
|
|
|
usagebuf[usageidx++] = 'e';
|
|
|
|
|
usagebuf[usageidx] = 0;
|
2009-03-05 19:19:37 +00:00
|
|
|
|
send_status_info (ctrl, "KEYPAIRINFO",
|
2011-02-04 12:57:53 +01:00
|
|
|
|
gripstr, 40,
|
|
|
|
|
id_buf, strlen (id_buf),
|
2020-04-17 16:09:05 +02:00
|
|
|
|
usagebuf, strlen (usagebuf),
|
2020-09-21 14:47:53 +02:00
|
|
|
|
"-", (size_t)1,
|
|
|
|
|
algostr, strlen (algostr),
|
2009-03-05 19:19:37 +00:00
|
|
|
|
NULL, (size_t)0);
|
|
|
|
|
}
|
2020-09-21 14:47:53 +02:00
|
|
|
|
xfree (algostr);
|
2009-03-05 19:19:37 +00:00
|
|
|
|
}
|
|
|
|
|
}
|
2009-03-06 17:31:27 +00:00
|
|
|
|
}
|
2009-01-08 19:56:30 +00:00
|
|
|
|
|
2009-03-05 19:19:37 +00:00
|
|
|
|
|
2009-03-06 17:31:27 +00: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)
|
2009-03-06 17:31:27 +00:00
|
|
|
|
{
|
|
|
|
|
gpg_error_t err;
|
2004-01-27 16:40:42 +00:00
|
|
|
|
|
2020-05-07 19:44:26 +02:00
|
|
|
|
do_getattr (app, ctrl, "CHV-STATUS");
|
|
|
|
|
|
2020-04-17 16:09:05 +02:00
|
|
|
|
err = switch_application (app, NKS_APP_NKS);
|
2009-03-06 17:31:27 +00:00
|
|
|
|
if (err)
|
|
|
|
|
return err;
|
2011-02-04 12:57:53 +01:00
|
|
|
|
|
2020-05-07 19:44:26 +02:00
|
|
|
|
do_learn_status_core (app, ctrl, flags, app->app_local->active_nks_app);
|
|
|
|
|
|
|
|
|
|
if (app->app_local->only_idlm)
|
|
|
|
|
return 0; /* ready. */
|
2004-01-27 16:40:42 +00:00
|
|
|
|
|
2020-04-17 16:09:05 +02:00
|
|
|
|
err = switch_application (app, app->app_local->qes_app_id);
|
2009-03-06 17:31:27 +00:00
|
|
|
|
if (err)
|
|
|
|
|
return 0; /* Silently ignore if we can't switch to SigG. */
|
2004-01-27 16:40:42 +00:00
|
|
|
|
|
2020-04-17 16:09:05 +02:00
|
|
|
|
do_learn_status_core (app, ctrl, flags, app->app_local->qes_app_id);
|
2009-03-05 19:19:37 +00:00
|
|
|
|
|
2004-01-27 16:40:42 +00:00
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
2020-04-17 16:09:05 +02:00
|
|
|
|
/* Helper to read a certificate from the file FID. The function
|
|
|
|
|
* assumes that the the application has already been selected. */
|
2005-05-18 10:48:06 +00:00
|
|
|
|
static gpg_error_t
|
2020-04-17 16:09:05 +02:00
|
|
|
|
readcert_from_ef (app_t app, int fid, unsigned char **cert, size_t *certlen)
|
2004-01-27 16:40:42 +00:00
|
|
|
|
{
|
|
|
|
|
gpg_error_t err;
|
|
|
|
|
unsigned char *buffer;
|
|
|
|
|
const unsigned char *p;
|
|
|
|
|
size_t buflen, n;
|
|
|
|
|
int class, tag, constructed, ndef;
|
|
|
|
|
size_t totobjlen, objlen, hdrlen;
|
|
|
|
|
int rootca = 0;
|
|
|
|
|
|
|
|
|
|
*cert = NULL;
|
|
|
|
|
*certlen = 0;
|
2009-03-05 19:19:37 +00:00
|
|
|
|
|
2004-01-27 16:40:42 +00:00
|
|
|
|
/* Read the entire file. fixme: This could be optimized by first
|
|
|
|
|
reading the header to figure out how long the certificate
|
|
|
|
|
actually is. */
|
2019-06-19 08:50:40 +02:00
|
|
|
|
err = iso7816_select_file (app_get_slot (app), fid, 0);
|
2004-01-27 16:40:42 +00:00
|
|
|
|
if (err)
|
|
|
|
|
{
|
2020-04-17 16:09:05 +02:00
|
|
|
|
log_error ("nks: error selecting FID 0x%04X: %s\n",
|
|
|
|
|
fid, gpg_strerror (err));
|
2004-01-27 16:40:42 +00:00
|
|
|
|
return err;
|
|
|
|
|
}
|
|
|
|
|
|
2019-06-19 08:50:40 +02:00
|
|
|
|
err = iso7816_read_binary (app_get_slot (app), 0, 0, &buffer, &buflen);
|
2004-01-27 16:40:42 +00:00
|
|
|
|
if (err)
|
|
|
|
|
{
|
2020-04-17 16:09:05 +02:00
|
|
|
|
log_error ("nks: error reading certificate from FID 0x%04X: %s\n",
|
2004-01-27 16:40:42 +00:00
|
|
|
|
fid, gpg_strerror (err));
|
|
|
|
|
return err;
|
|
|
|
|
}
|
2011-02-04 12:57:53 +01:00
|
|
|
|
|
2004-01-27 16:40:42 +00:00
|
|
|
|
if (!buflen || *buffer == 0xff)
|
|
|
|
|
{
|
2020-04-17 16:09:05 +02:00
|
|
|
|
log_info ("nks: no certificate contained in FID 0x%04X\n", fid);
|
2004-01-27 16:40:42 +00:00
|
|
|
|
err = gpg_error (GPG_ERR_NOT_FOUND);
|
|
|
|
|
goto leave;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Now figure something out about the object. */
|
|
|
|
|
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 )
|
|
|
|
|
;
|
|
|
|
|
else if ( class == CLASS_UNIVERSAL && tag == TAG_SET && constructed )
|
|
|
|
|
rootca = 1;
|
|
|
|
|
else
|
|
|
|
|
return gpg_error (GPG_ERR_INV_OBJ);
|
|
|
|
|
totobjlen = objlen + hdrlen;
|
2020-04-17 16:09:05 +02:00
|
|
|
|
log_assert (totobjlen <= buflen);
|
2004-01-27 16:40:42 +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
|
|
|
|
|
2004-01-27 16:40:42 +00:00
|
|
|
|
if (rootca)
|
|
|
|
|
;
|
|
|
|
|
else if (class == CLASS_UNIVERSAL && tag == TAG_OBJECT_ID && !constructed)
|
|
|
|
|
{
|
|
|
|
|
const unsigned char *save_p;
|
2011-02-04 12:57:53 +01:00
|
|
|
|
|
2004-01-27 16:40:42 +00:00
|
|
|
|
/* 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)
|
2004-01-27 16:40:42 +00:00
|
|
|
|
goto leave;
|
|
|
|
|
if ( !(class == CLASS_UNIVERSAL && tag == TAG_SEQUENCE && constructed) )
|
|
|
|
|
return gpg_error (GPG_ERR_INV_OBJ);
|
|
|
|
|
totobjlen = objlen + hdrlen;
|
2020-04-17 16:09:05 +02:00
|
|
|
|
log_assert (save_p + totobjlen <= buffer + buflen);
|
2004-01-27 16:40:42 +00:00
|
|
|
|
memmove (buffer, save_p, totobjlen);
|
|
|
|
|
}
|
2011-02-04 12:57:53 +01:00
|
|
|
|
|
2004-01-27 16:40:42 +00:00
|
|
|
|
*cert = buffer;
|
|
|
|
|
buffer = NULL;
|
|
|
|
|
*certlen = totobjlen;
|
|
|
|
|
|
|
|
|
|
leave:
|
|
|
|
|
xfree (buffer);
|
|
|
|
|
return err;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2020-04-17 16:09:05 +02:00
|
|
|
|
/* Read the certificate with id CERTID (as returned by learn_status in
|
|
|
|
|
the CERTINFO status lines) and return it in the freshly allocated
|
|
|
|
|
buffer put into CERT and the length of the certificate put into
|
|
|
|
|
CERTLEN. */
|
|
|
|
|
static gpg_error_t
|
|
|
|
|
do_readcert (app_t app, const char *certid,
|
|
|
|
|
unsigned char **cert, size_t *certlen)
|
|
|
|
|
{
|
|
|
|
|
int i, fid;
|
|
|
|
|
gpg_error_t err;
|
|
|
|
|
int nks_app_id;
|
|
|
|
|
|
|
|
|
|
*cert = NULL;
|
|
|
|
|
*certlen = 0;
|
|
|
|
|
|
|
|
|
|
if (!strncmp (certid, "NKS-NKS3.", 9))
|
|
|
|
|
nks_app_id = NKS_APP_NKS;
|
|
|
|
|
else if (!strncmp (certid, "NKS-ESIGN.", 10))
|
|
|
|
|
nks_app_id = NKS_APP_ESIGN;
|
|
|
|
|
else if (!strncmp (certid, "NKS-SIGG.", 9))
|
|
|
|
|
nks_app_id = NKS_APP_SIGG;
|
|
|
|
|
else if (!strncmp (certid, "NKS-DF01.", 9))
|
|
|
|
|
nks_app_id = NKS_APP_NKS;
|
2020-05-07 19:44:26 +02:00
|
|
|
|
else if (!strncmp (certid, "NKS-IDLM.", 9))
|
|
|
|
|
nks_app_id = NKS_APP_IDLM;
|
2020-04-17 16:09:05 +02:00
|
|
|
|
else
|
|
|
|
|
return gpg_error (GPG_ERR_INV_ID);
|
|
|
|
|
certid += nks_app_id == NKS_APP_ESIGN? 10 : 9;
|
|
|
|
|
|
|
|
|
|
err = switch_application (app, nks_app_id);
|
|
|
|
|
if (err)
|
|
|
|
|
return err;
|
|
|
|
|
|
|
|
|
|
if (!hexdigitp (certid) || !hexdigitp (certid+1)
|
|
|
|
|
|| !hexdigitp (certid+2) || !hexdigitp (certid+3)
|
|
|
|
|
|| certid[4])
|
|
|
|
|
return gpg_error (GPG_ERR_INV_ID);
|
|
|
|
|
fid = xtoi_4 (certid);
|
|
|
|
|
for (i=0; filelist[i].fid; i++)
|
|
|
|
|
if ((filelist[i].certtype || filelist[i].iskeypair)
|
2020-06-30 14:41:20 +02:00
|
|
|
|
&& filelist[i].nks_app_id == nks_app_id
|
2020-04-17 16:09:05 +02:00
|
|
|
|
&& filelist[i].fid == fid)
|
|
|
|
|
break;
|
|
|
|
|
if (!filelist[i].fid)
|
|
|
|
|
return gpg_error (GPG_ERR_NOT_FOUND);
|
|
|
|
|
|
|
|
|
|
/* If the requested objects is a plain public key, redirect it to
|
|
|
|
|
the corresponding certificate. The whole system is a bit messy
|
|
|
|
|
because we sometime use the key directly or let the caller
|
|
|
|
|
retrieve the key from the certificate. The rationale for
|
|
|
|
|
that is to support not-yet stored certificates. */
|
|
|
|
|
if (filelist[i].iskeypair)
|
|
|
|
|
fid = filelist[i].iskeypair;
|
|
|
|
|
|
|
|
|
|
return readcert_from_ef (app, fid, cert, certlen);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2009-05-08 15:07:45 +00:00
|
|
|
|
/* Handle the READKEY command. On success a canonical encoded
|
|
|
|
|
S-expression with the public key will get stored at PK and its
|
|
|
|
|
length at PKLEN; the caller must release that buffer. On error PK
|
|
|
|
|
and PKLEN are not changed and an error code is returned. As of now
|
|
|
|
|
this function is only useful for the internal authentication key.
|
2020-04-17 16:09:05 +02:00
|
|
|
|
Other keys are automagically retrieved by means of the
|
2009-05-08 15:07:45 +00:00
|
|
|
|
certificate parsing code in commands.c:cmd_readkey. For internal
|
|
|
|
|
use PK and PKLEN may be NULL to just check for an existing key. */
|
|
|
|
|
static gpg_error_t
|
2019-04-03 17:31:09 +02:00
|
|
|
|
do_readkey (app_t app, ctrl_t ctrl, const char *keyid, unsigned int flags,
|
|
|
|
|
unsigned char **pk, size_t *pklen)
|
2009-05-08 15:07:45 +00:00
|
|
|
|
{
|
|
|
|
|
gpg_error_t err;
|
|
|
|
|
unsigned char *buffer[2];
|
|
|
|
|
size_t buflen[2];
|
|
|
|
|
unsigned short path[1] = { 0x4500 };
|
|
|
|
|
|
|
|
|
|
/* We use a generic name to retrieve PK.AUT.IFD-SPK. */
|
2020-05-07 19:44:26 +02:00
|
|
|
|
if (!strcmp (keyid, "$IFDAUTHKEY") && app->appversion >= 3)
|
2009-05-08 15:07:45 +00:00
|
|
|
|
;
|
|
|
|
|
else /* Return the error code expected by cmd_readkey. */
|
2011-02-04 12:57:53 +01:00
|
|
|
|
return gpg_error (GPG_ERR_UNSUPPORTED_OPERATION);
|
2009-05-08 15:07:45 +00:00
|
|
|
|
|
|
|
|
|
/* Access the KEYD file which is always in the master directory. */
|
2019-06-19 08:50:40 +02:00
|
|
|
|
err = iso7816_select_path (app_get_slot (app), path, DIM (path));
|
2009-05-08 15:07:45 +00:00
|
|
|
|
if (err)
|
|
|
|
|
return err;
|
|
|
|
|
/* Due to the above select we need to re-select our application. */
|
|
|
|
|
app->app_local->need_app_select = 1;
|
|
|
|
|
/* Get the two records. */
|
2019-06-19 08:50:40 +02:00
|
|
|
|
err = iso7816_read_record (app_get_slot (app), 5, 1, 0,
|
|
|
|
|
&buffer[0], &buflen[0]);
|
2009-05-08 15:07:45 +00:00
|
|
|
|
if (err)
|
|
|
|
|
return err;
|
|
|
|
|
if (all_zero_p (buffer[0], buflen[0]))
|
|
|
|
|
{
|
|
|
|
|
xfree (buffer[0]);
|
|
|
|
|
return gpg_error (GPG_ERR_NOT_FOUND);
|
|
|
|
|
}
|
2019-06-19 08:50:40 +02:00
|
|
|
|
err = iso7816_read_record (app_get_slot (app), 6, 1, 0,
|
|
|
|
|
&buffer[1], &buflen[1]);
|
2009-05-08 15:07:45 +00:00
|
|
|
|
if (err)
|
|
|
|
|
{
|
|
|
|
|
xfree (buffer[0]);
|
|
|
|
|
return err;
|
|
|
|
|
}
|
|
|
|
|
|
2019-04-03 17:31:09 +02:00
|
|
|
|
if ((flags & APP_READKEY_FLAG_INFO))
|
|
|
|
|
{
|
|
|
|
|
/* Not yet implemented but we won't get here for any regular
|
|
|
|
|
* keyrefs anyway, thus the top layer will provide the
|
|
|
|
|
* keypairinfo from the certificate. */
|
|
|
|
|
(void)ctrl;
|
|
|
|
|
}
|
|
|
|
|
|
2009-05-08 15:07:45 +00:00
|
|
|
|
if (pk && pklen)
|
|
|
|
|
{
|
|
|
|
|
*pk = make_canon_sexp_from_rsa_pk (buffer[0], buflen[0],
|
|
|
|
|
buffer[1], buflen[1],
|
|
|
|
|
pklen);
|
|
|
|
|
if (!*pk)
|
|
|
|
|
err = gpg_error_from_syserror ();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
xfree (buffer[0]);
|
|
|
|
|
xfree (buffer[1]);
|
|
|
|
|
return err;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2020-07-02 18:35:34 +02:00
|
|
|
|
/* Write the certificate (CERT,CERTLEN) to the card at CERTREFSTR.
|
|
|
|
|
* CERTREFSTR is of the form "NKS_<yyy>.<four_hexdigit_keyref>". */
|
|
|
|
|
static gpg_error_t
|
|
|
|
|
do_writecert (app_t app, ctrl_t ctrl,
|
|
|
|
|
const char *certid,
|
|
|
|
|
gpg_error_t (*pincb)(void*, const char *, char **),
|
|
|
|
|
void *pincb_arg,
|
|
|
|
|
const unsigned char *cert, size_t certlen)
|
|
|
|
|
{
|
|
|
|
|
gpg_error_t err;
|
|
|
|
|
int i, fid, pwid;
|
|
|
|
|
int nks_app_id, tmp_app_id;
|
|
|
|
|
const char *desc;
|
|
|
|
|
|
|
|
|
|
(void)ctrl;
|
|
|
|
|
|
|
|
|
|
if (!strncmp (certid, "NKS-NKS3.", 9))
|
|
|
|
|
nks_app_id = NKS_APP_NKS;
|
|
|
|
|
else if (!strncmp (certid, "NKS-ESIGN.", 10))
|
|
|
|
|
nks_app_id = NKS_APP_ESIGN;
|
|
|
|
|
else if (!strncmp (certid, "NKS-SIGG.", 9))
|
|
|
|
|
nks_app_id = NKS_APP_SIGG;
|
|
|
|
|
else if (!strncmp (certid, "NKS-DF01.", 9))
|
|
|
|
|
nks_app_id = NKS_APP_NKS;
|
|
|
|
|
else if (!strncmp (certid, "NKS-IDLM.", 9))
|
|
|
|
|
nks_app_id = NKS_APP_IDLM;
|
|
|
|
|
else
|
|
|
|
|
return gpg_error (GPG_ERR_INV_ID);
|
|
|
|
|
certid += nks_app_id == NKS_APP_ESIGN? 10 : 9;
|
|
|
|
|
|
|
|
|
|
err = switch_application (app, nks_app_id);
|
|
|
|
|
if (err)
|
|
|
|
|
return err;
|
|
|
|
|
|
|
|
|
|
if (!hexdigitp (certid) || !hexdigitp (certid+1)
|
|
|
|
|
|| !hexdigitp (certid+2) || !hexdigitp (certid+3)
|
|
|
|
|
|| certid[4])
|
|
|
|
|
return gpg_error (GPG_ERR_INV_ID);
|
|
|
|
|
fid = xtoi_4 (certid);
|
|
|
|
|
for (i=0; filelist[i].fid; i++)
|
|
|
|
|
if ((filelist[i].certtype || filelist[i].iskeypair)
|
|
|
|
|
&& filelist[i].nks_app_id == nks_app_id
|
|
|
|
|
&& filelist[i].fid == fid)
|
|
|
|
|
break;
|
|
|
|
|
if (!filelist[i].fid)
|
|
|
|
|
return gpg_error (GPG_ERR_NOT_FOUND);
|
|
|
|
|
|
|
|
|
|
/* If the requested objects is a plain public key, redirect it to
|
|
|
|
|
* the corresponding certificate. This makes it easier for the user
|
|
|
|
|
* to figure out which CERTID to use. For example gpg-card shows
|
|
|
|
|
* the id of the key and not of the certificate. */
|
|
|
|
|
if (filelist[i].iskeypair)
|
|
|
|
|
fid = filelist[i].iskeypair;
|
|
|
|
|
|
|
|
|
|
/* We have no selective flush mechanism and given the rare use of
|
|
|
|
|
* writecert it won't harm to flush the entire cache. */
|
|
|
|
|
flush_fid_cache (app);
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* The certificates we support all require PW1.CH. Note that we
|
|
|
|
|
* check that the nks_app_id matches which sorts out CERTID values
|
|
|
|
|
* which are subkecy to a different nks_app_id. */
|
|
|
|
|
desc = parse_pwidstr (app, "PW1.CH", 0, &tmp_app_id, &pwid);
|
|
|
|
|
if (!desc || tmp_app_id != nks_app_id)
|
|
|
|
|
return gpg_error (GPG_ERR_INV_ID);
|
|
|
|
|
err = verify_pin (app, pwid, desc, pincb, pincb_arg);
|
|
|
|
|
if (err)
|
|
|
|
|
return err;
|
|
|
|
|
|
|
|
|
|
/* Select the file and write the certificate. */
|
|
|
|
|
err = iso7816_select_file (app_get_slot (app), fid, 0);
|
|
|
|
|
if (err)
|
|
|
|
|
{
|
|
|
|
|
log_error ("nks: error selecting FID 0x%04X: %s\n",
|
|
|
|
|
fid, gpg_strerror (err));
|
|
|
|
|
return err;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
err = iso7816_update_binary (app_get_slot (app), 1, 0, cert, certlen);
|
|
|
|
|
if (err)
|
|
|
|
|
{
|
|
|
|
|
log_error ("nks: error updating certificate at FID 0x%04X: %s\n",
|
|
|
|
|
fid, gpg_strerror (err));
|
|
|
|
|
return err;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2009-05-13 11:42:34 +00:00
|
|
|
|
/* Handle the WRITEKEY command for NKS. This function expects a
|
|
|
|
|
canonical encoded S-expression with the public key in KEYDATA and
|
|
|
|
|
its length in KEYDATALEN. The only supported KEYID is
|
|
|
|
|
"$IFDAUTHKEY" to store the terminal key on the card. Bit 0 of
|
|
|
|
|
FLAGS indicates whether an existing key shall get overwritten.
|
|
|
|
|
PINCB and PINCB_ARG are the usual arguments for the pinentry
|
|
|
|
|
callback. */
|
|
|
|
|
static gpg_error_t
|
|
|
|
|
do_writekey (app_t app, ctrl_t ctrl,
|
|
|
|
|
const char *keyid, unsigned int flags,
|
|
|
|
|
gpg_error_t (*pincb)(void*, const char *, char **),
|
|
|
|
|
void *pincb_arg,
|
|
|
|
|
const unsigned char *keydata, size_t keydatalen)
|
|
|
|
|
{
|
|
|
|
|
gpg_error_t err;
|
|
|
|
|
int force = (flags & 1);
|
|
|
|
|
const unsigned char *rsa_n = NULL;
|
|
|
|
|
const unsigned char *rsa_e = NULL;
|
|
|
|
|
size_t rsa_n_len, rsa_e_len;
|
|
|
|
|
unsigned int nbits;
|
|
|
|
|
|
|
|
|
|
(void)ctrl;
|
|
|
|
|
(void)pincb;
|
|
|
|
|
(void)pincb_arg;
|
|
|
|
|
|
2020-05-07 19:44:26 +02:00
|
|
|
|
if (!strcmp (keyid, "$IFDAUTHKEY") && app->appversion >= 3)
|
2009-05-13 11:42:34 +00:00
|
|
|
|
;
|
|
|
|
|
else
|
|
|
|
|
return gpg_error (GPG_ERR_INV_ID);
|
2011-02-04 12:57:53 +01:00
|
|
|
|
|
2019-04-03 17:31:09 +02:00
|
|
|
|
if (!force && !do_readkey (app, ctrl, keyid, 0, NULL, NULL))
|
2009-05-13 11:42:34 +00:00
|
|
|
|
return gpg_error (GPG_ERR_EEXIST);
|
|
|
|
|
|
|
|
|
|
/* Parse the S-expression. */
|
|
|
|
|
err = get_rsa_pk_from_canon_sexp (keydata, keydatalen,
|
|
|
|
|
&rsa_n, &rsa_n_len, &rsa_e, &rsa_e_len);
|
2011-02-04 12:57:53 +01:00
|
|
|
|
if (err)
|
2009-05-13 11:42:34 +00:00
|
|
|
|
goto leave;
|
|
|
|
|
|
|
|
|
|
/* Check that the parameters match the requirements. */
|
|
|
|
|
nbits = app_help_count_bits (rsa_n, rsa_n_len);
|
|
|
|
|
if (nbits != 1024)
|
|
|
|
|
{
|
|
|
|
|
log_error (_("RSA modulus missing or not of size %d bits\n"), 1024);
|
|
|
|
|
err = gpg_error (GPG_ERR_BAD_PUBKEY);
|
|
|
|
|
goto leave;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
nbits = app_help_count_bits (rsa_e, rsa_e_len);
|
|
|
|
|
if (nbits < 2 || nbits > 32)
|
|
|
|
|
{
|
|
|
|
|
log_error (_("RSA public exponent missing or larger than %d bits\n"),
|
|
|
|
|
32);
|
|
|
|
|
err = gpg_error (GPG_ERR_BAD_PUBKEY);
|
|
|
|
|
goto leave;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* /\* Store them. *\/ */
|
|
|
|
|
/* err = verify_pin (app, 0, NULL, pincb, pincb_arg); */
|
|
|
|
|
/* if (err) */
|
|
|
|
|
/* goto leave; */
|
|
|
|
|
|
|
|
|
|
/* Send the MSE:Store_Public_Key. */
|
2020-05-04 19:01:16 +02:00
|
|
|
|
/* We will need to clear the cache here. */
|
2009-05-13 11:42:34 +00:00
|
|
|
|
err = gpg_error (GPG_ERR_NOT_IMPLEMENTED);
|
|
|
|
|
/* mse = xtrymalloc (1000); */
|
2011-02-04 12:57:53 +01:00
|
|
|
|
|
2009-05-13 11:42:34 +00:00
|
|
|
|
/* mse[0] = 0x80; /\* Algorithm reference. *\/ */
|
|
|
|
|
/* mse[1] = 1; */
|
|
|
|
|
/* mse[2] = 0x17; */
|
|
|
|
|
/* mse[3] = 0x84; /\* Private key reference. *\/ */
|
|
|
|
|
/* mse[4] = 1; */
|
|
|
|
|
/* mse[5] = 0x77; */
|
|
|
|
|
/* mse[6] = 0x7F; /\* Public key parameter. *\/ */
|
|
|
|
|
/* mse[7] = 0x49; */
|
|
|
|
|
/* mse[8] = 0x81; */
|
|
|
|
|
/* mse[9] = 3 + 0x80 + 2 + rsa_e_len; */
|
|
|
|
|
/* mse[10] = 0x81; /\* RSA modulus of 128 byte. *\/ */
|
|
|
|
|
/* mse[11] = 0x81; */
|
|
|
|
|
/* mse[12] = rsa_n_len; */
|
|
|
|
|
/* memcpy (mse+12, rsa_n, rsa_n_len); */
|
|
|
|
|
/* mse[10] = 0x82; /\* RSA public exponent of up to 4 bytes. *\/ */
|
|
|
|
|
/* mse[12] = rsa_e_len; */
|
|
|
|
|
/* memcpy (mse+12, rsa_e, rsa_e_len); */
|
2019-06-19 08:50:40 +02:00
|
|
|
|
/* err = iso7816_manage_security_env (app_get_slot (app), 0x81, 0xB6, */
|
2009-05-13 11:42:34 +00:00
|
|
|
|
/* mse, sizeof mse); */
|
|
|
|
|
|
|
|
|
|
leave:
|
|
|
|
|
return err;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2020-05-07 08:18:28 +02:00
|
|
|
|
/* Return an allocated string to be used as prompt. Returns NULL on
|
|
|
|
|
* malloc error. */
|
|
|
|
|
static char *
|
|
|
|
|
make_prompt (app_t app, int remaining, const char *firstline,
|
|
|
|
|
const char *extraline)
|
|
|
|
|
{
|
|
|
|
|
char *serial, *tmpbuf, *result;
|
|
|
|
|
|
|
|
|
|
serial = get_dispserialno (app);
|
|
|
|
|
|
|
|
|
|
/* 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",
|
|
|
|
|
serial,
|
|
|
|
|
"",
|
|
|
|
|
"");
|
|
|
|
|
xfree (serial);
|
|
|
|
|
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,
|
|
|
|
|
extraline? "%0A%0A":"", extraline,
|
|
|
|
|
NULL);
|
|
|
|
|
xfree (result);
|
|
|
|
|
result = tmpbuf;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return result;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2009-03-05 19:19:37 +00:00
|
|
|
|
static gpg_error_t
|
|
|
|
|
basic_pin_checks (const char *pinvalue, int minlen, int maxlen)
|
|
|
|
|
{
|
|
|
|
|
if (strlen (pinvalue) < minlen)
|
|
|
|
|
{
|
|
|
|
|
log_error ("PIN is too short; minimum length is %d\n", minlen);
|
|
|
|
|
return gpg_error (GPG_ERR_BAD_PIN);
|
|
|
|
|
}
|
|
|
|
|
if (strlen (pinvalue) > maxlen)
|
|
|
|
|
{
|
|
|
|
|
log_error ("PIN is too large; maximum length is %d\n", maxlen);
|
|
|
|
|
return gpg_error (GPG_ERR_BAD_PIN);
|
|
|
|
|
}
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2004-01-28 16:21:57 +00:00
|
|
|
|
/* Verify the PIN if required. */
|
2005-05-18 10:48:06 +00:00
|
|
|
|
static gpg_error_t
|
2009-03-05 19:19:37 +00:00
|
|
|
|
verify_pin (app_t app, int pwid, const char *desc,
|
2005-05-18 10:48:06 +00:00
|
|
|
|
gpg_error_t (*pincb)(void*, const char *, char **),
|
2004-01-28 16:21:57 +00:00
|
|
|
|
void *pincb_arg)
|
|
|
|
|
{
|
2007-03-07 20:55:14 +00:00
|
|
|
|
int rc;
|
2020-05-07 08:18:28 +02:00
|
|
|
|
pininfo_t pininfo;
|
|
|
|
|
char *prompt;
|
|
|
|
|
const char *extrapromptline = NULL;
|
|
|
|
|
int remaining, nullpin;
|
2007-03-07 20:55:14 +00:00
|
|
|
|
|
2009-03-05 19:19:37 +00:00
|
|
|
|
if (!desc)
|
2020-05-07 08:18:28 +02:00
|
|
|
|
desc = "||PIN";
|
2007-03-07 20:55:14 +00:00
|
|
|
|
|
|
|
|
|
memset (&pininfo, 0, sizeof pininfo);
|
2013-01-09 16:23:55 +09:00
|
|
|
|
pininfo.fixedlen = -1;
|
2020-05-07 14:03:25 +02:00
|
|
|
|
|
|
|
|
|
/* FIXME: TCOS allows to read the min. and max. values - do this. */
|
2020-05-07 19:44:26 +02:00
|
|
|
|
if (app->appversion == 15)
|
2020-05-07 14:03:25 +02:00
|
|
|
|
{
|
|
|
|
|
if (app->app_local->active_nks_app == NKS_APP_NKS && pwid == 0x03)
|
|
|
|
|
pininfo.minlen = 6;
|
|
|
|
|
else if (app->app_local->active_nks_app == NKS_APP_ESIGN && pwid == 0x81)
|
|
|
|
|
pininfo.minlen = 6;
|
|
|
|
|
else
|
|
|
|
|
pininfo.minlen = 8;
|
|
|
|
|
pininfo.maxlen = 24;
|
|
|
|
|
}
|
2020-05-07 19:44:26 +02:00
|
|
|
|
else if (app->app_local->active_nks_app == NKS_APP_IDLM)
|
|
|
|
|
{
|
|
|
|
|
if (pwid == 0x00)
|
|
|
|
|
pininfo.minlen = 6;
|
|
|
|
|
else
|
|
|
|
|
pininfo.minlen = 8;
|
|
|
|
|
pininfo.maxlen = 24;
|
|
|
|
|
}
|
2020-05-07 14:03:25 +02:00
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
/* For NKS3 we used these fixed values; let's keep this. */
|
|
|
|
|
pininfo.minlen = 6;
|
|
|
|
|
pininfo.maxlen = 16;
|
|
|
|
|
}
|
2007-03-07 20:55:14 +00:00
|
|
|
|
|
2020-05-07 08:18:28 +02:00
|
|
|
|
remaining = iso7816_verify_status (app_get_slot (app), pwid);
|
|
|
|
|
nullpin = (remaining == ISO7816_VERIFY_NULLPIN);
|
|
|
|
|
if (remaining < 0)
|
|
|
|
|
remaining = -1; /* We don't care about the concrete error. */
|
|
|
|
|
if (remaining < 3)
|
|
|
|
|
{
|
|
|
|
|
if (remaining >= 0)
|
|
|
|
|
log_info ("nks: PIN has %d attempts left\n", remaining);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (nullpin)
|
|
|
|
|
{
|
|
|
|
|
log_info ("nks: The NullPIN for PIN 0x%02x has not yet been changed\n",
|
|
|
|
|
pwid);
|
2020-05-07 14:03:25 +02:00
|
|
|
|
extrapromptline = _("Note: PIN has not yet been enabled.");
|
2020-05-07 08:18:28 +02:00
|
|
|
|
}
|
|
|
|
|
|
scd: Rename 'keypad' to 'pinpad'.
* NEWS: Mention scd changes.
* agent/divert-scd.c (getpin_cb): Change message.
* agent/call-scd.c (inq_needpin): Change the protocol to
POPUPPINPADPROMPT and DISMISSPINPADPROMPT.
* scd/command.c (pin_cb): Likewise.
* scd/apdu.c (struct reader_table_s): Rename member functions.
(check_pcsc_pinpad, pcsc_pinpad_verify, pcsc_pinpad_modify,
check_ccid_pinpad, ccid_pinpad_operation, apdu_check_pinpad
apdu_pinpad_verify, apdu_pinpad_modify): Rename.
* scd/apdu.h (SW_HOST_NO_PINPAD, apdu_check_pinpad)
(apdu_pinpad_verify, apdu_pinpad_modify): Rename.
* scd/iso7816.h (iso7816_check_pinpad): Rename.
* scd/iso7816.c (map_sw): Use SW_HOST_NO_PINPAD.
(iso7816_check_pinpad): Rename.
(iso7816_verify_kp, iso7816_change_reference_data_kp): Follow
the change.
* scd/ccid-driver.h (CCID_DRIVER_ERR_NO_PINPAD): Rename.
* scd/ccid-driver.c (ccid_transceive_secure): Use it.
* scd/app-dinsig.c (verify_pin): Follow the change.
* scd/app-nks.c (verify_pin): Follow the change.
* scd/app-openpgp.c (check_pinpad_request): Rename.
(parse_login_data, verify_a_chv, verify_chv3, do_change_pin): Follow
the change.
* scd/scdaemon.c (oDisablePinpad, oEnablePinpadVarlen): Rename.
* scd/scdaemon.h (opt): Rename to disable_pinpad,
enable_pinpad_varlen.
* tools/gpgconf-comp.c (gc_options_scdaemon): Rename to
disable-pinpad.
2013-02-07 10:07:51 +09:00
|
|
|
|
if (!opt.disable_pinpad
|
2019-06-19 08:50:40 +02:00
|
|
|
|
&& !iso7816_check_pinpad (app_get_slot (app), ISO7816_VERIFY, &pininfo) )
|
2007-03-07 20:55:14 +00:00
|
|
|
|
{
|
2020-05-07 08:18:28 +02:00
|
|
|
|
prompt = make_prompt (app, remaining, desc, extrapromptline);
|
|
|
|
|
rc = pincb (pincb_arg, prompt, NULL);
|
|
|
|
|
xfree (prompt);
|
2007-03-07 20:55:14 +00:00
|
|
|
|
if (rc)
|
|
|
|
|
{
|
|
|
|
|
log_info (_("PIN callback returned error: %s\n"),
|
|
|
|
|
gpg_strerror (rc));
|
|
|
|
|
return rc;
|
|
|
|
|
}
|
2011-02-04 12:57:53 +01:00
|
|
|
|
|
2019-06-19 08:50:40 +02:00
|
|
|
|
rc = iso7816_verify_kp (app_get_slot (app), pwid, &pininfo);
|
2009-03-06 17:31:27 +00:00
|
|
|
|
pincb (pincb_arg, NULL, NULL); /* Dismiss the prompt. */
|
2007-03-07 20:55:14 +00:00
|
|
|
|
}
|
|
|
|
|
else
|
2004-01-28 16:21:57 +00:00
|
|
|
|
{
|
|
|
|
|
char *pinvalue;
|
|
|
|
|
|
2020-05-07 08:18:28 +02:00
|
|
|
|
prompt = make_prompt (app, remaining, desc, extrapromptline);
|
|
|
|
|
rc = pincb (pincb_arg, prompt, &pinvalue);
|
|
|
|
|
xfree (prompt);
|
2004-01-28 16:21:57 +00:00
|
|
|
|
if (rc)
|
|
|
|
|
{
|
|
|
|
|
log_info ("PIN callback returned error: %s\n", gpg_strerror (rc));
|
|
|
|
|
return rc;
|
|
|
|
|
}
|
|
|
|
|
|
2009-03-05 19:19:37 +00:00
|
|
|
|
rc = basic_pin_checks (pinvalue, pininfo.minlen, pininfo.maxlen);
|
|
|
|
|
if (rc)
|
2004-01-28 16:21:57 +00:00
|
|
|
|
{
|
|
|
|
|
xfree (pinvalue);
|
2009-03-05 19:19:37 +00:00
|
|
|
|
return rc;
|
2004-01-28 16:21:57 +00:00
|
|
|
|
}
|
|
|
|
|
|
2019-06-19 08:50:40 +02:00
|
|
|
|
rc = iso7816_verify (app_get_slot (app), pwid,
|
|
|
|
|
pinvalue, strlen (pinvalue));
|
2004-01-28 16:21:57 +00:00
|
|
|
|
xfree (pinvalue);
|
|
|
|
|
}
|
|
|
|
|
|
2007-03-07 20:55:14 +00:00
|
|
|
|
if (rc)
|
|
|
|
|
{
|
|
|
|
|
if ( gpg_err_code (rc) == GPG_ERR_USE_CONDITIONS )
|
|
|
|
|
log_error (_("the NullPIN has not yet been changed\n"));
|
|
|
|
|
else
|
|
|
|
|
log_error ("verify PIN failed\n");
|
|
|
|
|
return rc;
|
|
|
|
|
}
|
|
|
|
|
|
2004-01-28 16:21:57 +00:00
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* 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 in the 3rd argument. */
|
2011-02-04 12:57:53 +01:00
|
|
|
|
static gpg_error_t
|
2020-01-07 18:45:33 +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 **),
|
|
|
|
|
void *pincb_arg,
|
|
|
|
|
const void *indata, size_t indatalen,
|
|
|
|
|
unsigned char **outdata, size_t *outdatalen )
|
2004-01-28 16:21:57 +00:00
|
|
|
|
{
|
|
|
|
|
static unsigned char sha1_prefix[15] = /* Object ID is 1.3.14.3.2.26 */
|
|
|
|
|
{ 0x30, 0x21, 0x30, 0x09, 0x06, 0x05, 0x2b, 0x0e, 0x03,
|
|
|
|
|
0x02, 0x1a, 0x05, 0x00, 0x04, 0x14 };
|
|
|
|
|
static unsigned char rmd160_prefix[15] = /* Object ID is 1.3.36.3.2.1 */
|
|
|
|
|
{ 0x30, 0x21, 0x30, 0x09, 0x06, 0x05, 0x2b, 0x24, 0x03,
|
|
|
|
|
0x02, 0x01, 0x05, 0x00, 0x04, 0x14 };
|
2020-05-04 19:01:16 +02:00
|
|
|
|
gpg_error_t err;
|
|
|
|
|
int idx;
|
2020-05-07 14:03:25 +02:00
|
|
|
|
int pwid;
|
2009-03-26 19:27:04 +00:00
|
|
|
|
unsigned char kid;
|
|
|
|
|
unsigned char data[83]; /* Must be large enough for a SHA-1 digest
|
|
|
|
|
+ the largest OID prefix. */
|
|
|
|
|
size_t datalen;
|
2004-01-28 16:21:57 +00:00
|
|
|
|
|
2020-01-07 18:45:33 +01:00
|
|
|
|
(void)ctrl;
|
|
|
|
|
|
2009-03-26 19:27:04 +00:00
|
|
|
|
switch (indatalen)
|
|
|
|
|
{
|
|
|
|
|
case 16: case 20: case 35: case 47: case 51: case 67: case 83: break;
|
|
|
|
|
default: return gpg_error (GPG_ERR_INV_VALUE);
|
|
|
|
|
}
|
2004-01-28 16:21:57 +00:00
|
|
|
|
|
2020-05-07 08:18:28 +02:00
|
|
|
|
err = find_fid_by_keyref (app, keyidstr, &idx, NULL);
|
2020-05-04 19:01:16 +02:00
|
|
|
|
if (err)
|
|
|
|
|
return err;
|
2009-03-06 17:31:27 +00:00
|
|
|
|
|
2020-05-04 19:01:16 +02:00
|
|
|
|
if (app->app_local->active_nks_app == NKS_APP_SIGG
|
|
|
|
|
&& app->app_local->sigg_is_msig)
|
2009-05-08 15:07:45 +00:00
|
|
|
|
{
|
|
|
|
|
log_info ("mass signature cards are not allowed\n");
|
|
|
|
|
return gpg_error (GPG_ERR_NOT_SUPPORTED);
|
|
|
|
|
}
|
|
|
|
|
|
2020-05-04 19:01:16 +02:00
|
|
|
|
if (!filelist[idx].issignkey)
|
2004-01-28 16:21:57 +00:00
|
|
|
|
return gpg_error (GPG_ERR_INV_ID);
|
2020-05-04 19:01:16 +02:00
|
|
|
|
|
|
|
|
|
kid = filelist[idx].kid;
|
2009-03-26 19:27:04 +00:00
|
|
|
|
|
|
|
|
|
/* Prepare the DER object from INDATA. */
|
2020-05-07 19:44:26 +02:00
|
|
|
|
if (app->appversion > 2 && (indatalen == 35
|
|
|
|
|
|| indatalen == 47
|
|
|
|
|
|| indatalen == 51
|
|
|
|
|
|| indatalen == 67
|
|
|
|
|
|| indatalen == 83))
|
2009-03-26 19:27:04 +00:00
|
|
|
|
{
|
|
|
|
|
/* The caller send data matching the length of the ASN.1 encoded
|
|
|
|
|
hash for SHA-{1,224,256,384,512}. Assume that is okay. */
|
2020-04-17 16:09:05 +02:00
|
|
|
|
log_assert (indatalen <= sizeof data);
|
2009-03-26 19:27:04 +00:00
|
|
|
|
memcpy (data, indata, indatalen);
|
|
|
|
|
datalen = indatalen;
|
|
|
|
|
}
|
|
|
|
|
else if (indatalen == 35)
|
2004-01-28 16:21:57 +00:00
|
|
|
|
{
|
|
|
|
|
/* Alright, the caller was so kind to send us an already
|
2009-03-26 19:27:04 +00:00
|
|
|
|
prepared DER object. This is for TCOS 2. */
|
2004-01-28 16:21:57 +00:00
|
|
|
|
if (hashalgo == GCRY_MD_SHA1 && !memcmp (indata, sha1_prefix, 15))
|
|
|
|
|
;
|
2009-03-26 19:27:04 +00:00
|
|
|
|
else if (hashalgo == GCRY_MD_RMD160 && !memcmp (indata,rmd160_prefix,15))
|
2004-01-28 16:21:57 +00:00
|
|
|
|
;
|
2011-02-04 12:57:53 +01:00
|
|
|
|
else
|
2004-01-28 16:21:57 +00:00
|
|
|
|
return gpg_error (GPG_ERR_UNSUPPORTED_ALGORITHM);
|
|
|
|
|
memcpy (data, indata, indatalen);
|
2009-03-26 19:27:04 +00:00
|
|
|
|
datalen = 35;
|
2004-01-28 16:21:57 +00:00
|
|
|
|
}
|
2009-03-26 19:27:04 +00:00
|
|
|
|
else if (indatalen == 20)
|
2004-01-28 16:21:57 +00:00
|
|
|
|
{
|
|
|
|
|
if (hashalgo == GCRY_MD_SHA1)
|
|
|
|
|
memcpy (data, sha1_prefix, 15);
|
|
|
|
|
else if (hashalgo == GCRY_MD_RMD160)
|
|
|
|
|
memcpy (data, rmd160_prefix, 15);
|
2011-02-04 12:57:53 +01:00
|
|
|
|
else
|
2004-01-28 16:21:57 +00:00
|
|
|
|
return gpg_error (GPG_ERR_UNSUPPORTED_ALGORITHM);
|
|
|
|
|
memcpy (data+15, indata, indatalen);
|
2009-03-26 19:27:04 +00:00
|
|
|
|
datalen = 35;
|
2004-01-28 16:21:57 +00:00
|
|
|
|
}
|
2009-03-26 19:27:04 +00:00
|
|
|
|
else
|
|
|
|
|
return gpg_error (GPG_ERR_INV_VALUE);
|
|
|
|
|
|
2004-01-28 16:21:57 +00:00
|
|
|
|
|
2009-03-26 19:27:04 +00:00
|
|
|
|
/* Send an MSE for PSO:Computer_Signature. */
|
2020-05-07 19:44:26 +02:00
|
|
|
|
if (app->appversion > 2)
|
2009-03-26 19:27:04 +00:00
|
|
|
|
{
|
|
|
|
|
unsigned char mse[6];
|
2011-02-04 12:57:53 +01:00
|
|
|
|
|
2009-03-26 19:27:04 +00:00
|
|
|
|
mse[0] = 0x80; /* Algorithm reference. */
|
|
|
|
|
mse[1] = 1;
|
|
|
|
|
mse[2] = 2; /* RSA, card does pkcs#1 v1.5 padding, no ASN.1 check. */
|
|
|
|
|
mse[3] = 0x84; /* Private key reference. */
|
|
|
|
|
mse[4] = 1;
|
|
|
|
|
mse[5] = kid;
|
2020-05-04 19:01:16 +02:00
|
|
|
|
err = iso7816_manage_security_env (app_get_slot (app), 0x41, 0xB6,
|
|
|
|
|
mse, sizeof mse);
|
2009-03-26 19:27:04 +00:00
|
|
|
|
}
|
2020-05-07 14:03:25 +02:00
|
|
|
|
|
|
|
|
|
/* We use the Global PIN 1 */
|
2020-05-07 19:44:26 +02:00
|
|
|
|
if (app->appversion == 15)
|
2020-05-07 14:03:25 +02:00
|
|
|
|
pwid = 0x03;
|
|
|
|
|
else
|
|
|
|
|
pwid = 0x00;
|
|
|
|
|
|
2020-05-04 19:01:16 +02:00
|
|
|
|
if (!err)
|
2020-05-07 14:03:25 +02:00
|
|
|
|
err = verify_pin (app, pwid, NULL, pincb, pincb_arg);
|
2009-03-26 19:27:04 +00:00
|
|
|
|
/* Compute the signature. */
|
2020-05-04 19:01:16 +02:00
|
|
|
|
if (!err)
|
|
|
|
|
err = iso7816_compute_ds (app_get_slot (app), 0, data, datalen, 0,
|
|
|
|
|
outdata, outdatalen);
|
|
|
|
|
return err;
|
2004-01-28 16:21:57 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* 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. */
|
2011-02-04 12:57:53 +01:00
|
|
|
|
static gpg_error_t
|
2020-01-07 18:45:33 +01:00
|
|
|
|
do_decipher (app_t app, ctrl_t ctrl, const char *keyidstr,
|
2005-05-18 10:48:06 +00:00
|
|
|
|
gpg_error_t (*pincb)(void*, const char *, char **),
|
2004-01-28 16:21:57 +00:00
|
|
|
|
void *pincb_arg,
|
|
|
|
|
const void *indata, size_t indatalen,
|
2013-08-26 17:29:54 +02:00
|
|
|
|
unsigned char **outdata, size_t *outdatalen,
|
|
|
|
|
unsigned int *r_info)
|
2004-01-28 16:21:57 +00:00
|
|
|
|
{
|
2020-05-04 19:01:16 +02:00
|
|
|
|
gpg_error_t err;
|
|
|
|
|
int idx;
|
2009-03-30 12:46:06 +00:00
|
|
|
|
int kid;
|
2020-05-07 08:18:28 +02:00
|
|
|
|
int algo;
|
|
|
|
|
int pwid;
|
|
|
|
|
int padind;
|
|
|
|
|
int extended_mode;
|
2004-01-28 16:21:57 +00:00
|
|
|
|
|
2020-01-07 18:45:33 +01:00
|
|
|
|
(void)ctrl;
|
2013-08-26 17:29:54 +02:00
|
|
|
|
(void)r_info;
|
|
|
|
|
|
2020-05-04 19:01:16 +02:00
|
|
|
|
if (!indatalen)
|
2004-01-28 16:21:57 +00:00
|
|
|
|
return gpg_error (GPG_ERR_INV_VALUE);
|
|
|
|
|
|
2020-05-07 08:18:28 +02:00
|
|
|
|
err = find_fid_by_keyref (app, keyidstr, &idx, &algo);
|
2020-05-04 19:01:16 +02:00
|
|
|
|
if (err)
|
|
|
|
|
return err;
|
2009-03-06 17:31:27 +00:00
|
|
|
|
|
2020-05-04 19:01:16 +02:00
|
|
|
|
if (!filelist[idx].isencrkey)
|
2004-01-28 16:21:57 +00:00
|
|
|
|
return gpg_error (GPG_ERR_INV_ID);
|
2020-05-04 19:01:16 +02:00
|
|
|
|
|
|
|
|
|
kid = filelist[idx].kid;
|
2009-03-30 12:46:06 +00:00
|
|
|
|
|
2020-05-07 19:44:26 +02:00
|
|
|
|
if (app->appversion <= 2)
|
2020-05-07 08:18:28 +02:00
|
|
|
|
{
|
|
|
|
|
static const unsigned char mse[] =
|
|
|
|
|
{
|
|
|
|
|
0x80, 1, 0x10, /* Select algorithm RSA. */
|
|
|
|
|
0x84, 1, 0x81 /* Select local secret key 1 for decryption. */
|
|
|
|
|
};
|
|
|
|
|
err = iso7816_manage_security_env (app_get_slot (app), 0xC1, 0xB8,
|
|
|
|
|
mse, sizeof mse);
|
|
|
|
|
extended_mode = 0;
|
|
|
|
|
padind = 0x81;
|
|
|
|
|
}
|
|
|
|
|
else if (algo == GCRY_PK_ECC)
|
|
|
|
|
{
|
|
|
|
|
unsigned char mse[3];
|
|
|
|
|
mse[0] = 0x84; /* Private key reference. */
|
|
|
|
|
mse[1] = 1;
|
|
|
|
|
mse[2] = kid;
|
|
|
|
|
err = iso7816_manage_security_env (app_get_slot (app), 0x41, 0xB8,
|
|
|
|
|
mse, sizeof mse);
|
|
|
|
|
extended_mode = 0;
|
|
|
|
|
padind = 0x00;
|
|
|
|
|
}
|
|
|
|
|
else
|
2009-03-30 12:46:06 +00:00
|
|
|
|
{
|
|
|
|
|
unsigned char mse[6];
|
|
|
|
|
mse[0] = 0x80; /* Algorithm reference. */
|
|
|
|
|
mse[1] = 1;
|
|
|
|
|
mse[2] = 0x0a; /* RSA no padding. (0x1A is pkcs#1.5 padding.) */
|
|
|
|
|
mse[3] = 0x84; /* Private key reference. */
|
|
|
|
|
mse[4] = 1;
|
|
|
|
|
mse[5] = kid;
|
2020-05-04 19:01:16 +02:00
|
|
|
|
err = iso7816_manage_security_env (app_get_slot (app), 0x41, 0xB8,
|
|
|
|
|
mse, sizeof mse);
|
2020-05-07 08:18:28 +02:00
|
|
|
|
extended_mode = 1;
|
|
|
|
|
padind = 0x81;
|
2009-03-30 12:46:06 +00:00
|
|
|
|
}
|
2020-05-07 08:18:28 +02:00
|
|
|
|
if (err)
|
2009-03-30 12:46:06 +00:00
|
|
|
|
{
|
2020-05-07 08:18:28 +02:00
|
|
|
|
log_error ("nks: MSE failed: %s\n", gpg_strerror (err));
|
|
|
|
|
goto leave;
|
2009-03-30 12:46:06 +00:00
|
|
|
|
}
|
2004-01-28 16:21:57 +00:00
|
|
|
|
|
2020-05-07 14:03:25 +02:00
|
|
|
|
/* We use the Global PIN 1 */
|
2020-05-07 19:44:26 +02:00
|
|
|
|
if (app->appversion == 15)
|
2020-05-07 08:18:28 +02:00
|
|
|
|
pwid = 0x03;
|
|
|
|
|
else
|
|
|
|
|
pwid = 0x00;
|
2009-03-30 12:46:06 +00:00
|
|
|
|
|
2020-05-07 08:18:28 +02:00
|
|
|
|
err = verify_pin (app, pwid, NULL, pincb, pincb_arg);
|
|
|
|
|
if (err)
|
|
|
|
|
goto leave;
|
|
|
|
|
|
|
|
|
|
err = iso7816_decipher (app_get_slot (app), extended_mode,
|
|
|
|
|
indata, indatalen, 0, padind,
|
|
|
|
|
outdata, outdatalen);
|
|
|
|
|
|
|
|
|
|
leave:
|
2020-05-04 19:01:16 +02:00
|
|
|
|
return err;
|
2004-01-28 16:21:57 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2009-03-05 19:19:37 +00:00
|
|
|
|
|
|
|
|
|
/* Parse a password ID string. Returns NULL on error or a string
|
2020-05-07 08:18:28 +02:00
|
|
|
|
* suitable as passphrase prompt on success. On success stores the
|
|
|
|
|
* reference value for the password at R_PWID and a flag indicating
|
|
|
|
|
* which app is to be used at R_NKS_APP_ID. If NEW_MODE is true, the
|
2020-05-07 14:03:25 +02:00
|
|
|
|
* returned description is suitable for a new password. Here is a
|
|
|
|
|
* take mapping the PWIDSTR to the used PWIDs:
|
|
|
|
|
*
|
|
|
|
|
* | pwidstr | | NKS3 | NKS15 | IDKEY1 |
|
|
|
|
|
* |------------+--------------+------+-------+--------|
|
|
|
|
|
* | PW1.CH | Global PIN 1 | 0x00 | 0x03 | 0x00 |
|
|
|
|
|
* | PW2.CH | Global PIN 2 | 0x01 | 0x04 | 0x01 |
|
|
|
|
|
* | PW1.CH.SIG | SigG PIN 1 | 0x81 | 0x81 | - |
|
|
|
|
|
* | PW2.CH.SIG | SigG PIN 2 | 0x83 | 0x82 | - |
|
2020-05-07 08:18:28 +02:00
|
|
|
|
*
|
2020-05-07 14:03:25 +02:00
|
|
|
|
* The names for PWIDSTR are taken from the NKS3 specs; the specs of
|
|
|
|
|
* other cards use different names but we keep using the. PIN1 can be
|
|
|
|
|
* used to unlock PIN2 and vice versa; for consistence with other
|
|
|
|
|
* cards we name PIN2 a "PUK". The IDKEY card also features a Card
|
|
|
|
|
* Reset Key (CR Key 0x01) which can also be used to reset PIN1.
|
2020-05-07 08:18:28 +02:00
|
|
|
|
*
|
2020-05-07 14:03:25 +02:00
|
|
|
|
* For testing it is possible to specify the PWID directly; the
|
|
|
|
|
* prompts are then not very descriptive:
|
2020-05-07 08:18:28 +02:00
|
|
|
|
*
|
|
|
|
|
* NKS.0xnn - Switch to NKS and select id 0xnn
|
|
|
|
|
* SIGG.0xnn - Switch to SigG and select id 0xnn
|
|
|
|
|
* ESIGN.0xnn - Switch to ESIGN and select id 0xnn
|
2009-03-05 19:19:37 +00:00
|
|
|
|
*/
|
|
|
|
|
static const char *
|
2020-05-07 14:03:25 +02:00
|
|
|
|
parse_pwidstr (app_t app, const char *pwidstr, int new_mode,
|
|
|
|
|
int *r_nks_app_id, int *r_pwid)
|
2009-03-05 19:19:37 +00:00
|
|
|
|
{
|
|
|
|
|
const char *desc;
|
2020-05-07 19:44:26 +02:00
|
|
|
|
int nks15 = app->appversion == 15;
|
2009-03-05 19:19:37 +00:00
|
|
|
|
|
|
|
|
|
if (!pwidstr)
|
|
|
|
|
desc = NULL;
|
|
|
|
|
else if (!strcmp (pwidstr, "PW1.CH"))
|
|
|
|
|
{
|
2020-04-17 16:09:05 +02:00
|
|
|
|
*r_nks_app_id = NKS_APP_NKS;
|
2020-05-07 14:03:25 +02:00
|
|
|
|
*r_pwid = nks15? 0x03 : 0x00;
|
2009-03-05 19:19:37 +00:00
|
|
|
|
/* TRANSLATORS: Do not translate the "|*|" prefixes but keep
|
|
|
|
|
them verbatim at the start of the string. */
|
|
|
|
|
desc = (new_mode
|
|
|
|
|
? _("|N|Please enter a new PIN for the standard keys.")
|
|
|
|
|
: _("||Please enter the PIN for the standard keys."));
|
|
|
|
|
}
|
|
|
|
|
else if (!strcmp (pwidstr, "PW2.CH"))
|
|
|
|
|
{
|
2020-04-17 16:09:05 +02:00
|
|
|
|
*r_nks_app_id = NKS_APP_NKS;
|
2020-05-07 14:03:25 +02:00
|
|
|
|
*r_pwid = nks15? 0x04 : 0x01;
|
2009-03-05 19:19:37 +00:00
|
|
|
|
desc = (new_mode
|
|
|
|
|
? _("|NP|Please enter a new PIN Unblocking Code (PUK) "
|
|
|
|
|
"for the standard keys.")
|
|
|
|
|
: _("|P|Please enter the PIN Unblocking Code (PUK) "
|
|
|
|
|
"for the standard keys."));
|
|
|
|
|
}
|
2020-05-07 19:44:26 +02:00
|
|
|
|
else if (!strcmp (pwidstr, "PW1.CH.SIG") && !app->app_local->only_idlm)
|
2009-03-05 19:19:37 +00:00
|
|
|
|
{
|
2020-05-07 14:03:25 +02:00
|
|
|
|
*r_nks_app_id = app->app_local->qes_app_id;
|
2009-03-05 19:19:37 +00:00
|
|
|
|
*r_pwid = 0x81;
|
|
|
|
|
desc = (new_mode
|
|
|
|
|
? _("|N|Please enter a new PIN for the key to create "
|
|
|
|
|
"qualified signatures.")
|
|
|
|
|
: _("||Please enter the PIN for the key to create "
|
|
|
|
|
"qualified signatures."));
|
|
|
|
|
}
|
2020-05-07 19:44:26 +02:00
|
|
|
|
else if (!strcmp (pwidstr, "PW2.CH.SIG") && !app->app_local->only_idlm)
|
2009-03-05 19:19:37 +00:00
|
|
|
|
{
|
2020-05-07 14:03:25 +02:00
|
|
|
|
*r_nks_app_id = app->app_local->qes_app_id;
|
|
|
|
|
*r_pwid = nks15? 0x82 : 0x83;
|
2009-03-05 19:19:37 +00:00
|
|
|
|
desc = (new_mode
|
|
|
|
|
? _("|NP|Please enter a new PIN Unblocking Code (PUK) "
|
|
|
|
|
"for the key to create qualified signatures.")
|
|
|
|
|
: _("|P|Please enter the PIN Unblocking Code (PUK) "
|
|
|
|
|
"for the key to create qualified signatures."));
|
|
|
|
|
}
|
2020-05-07 08:18:28 +02:00
|
|
|
|
else if (!strncmp (pwidstr, "NKS.0x", 6)
|
|
|
|
|
&& hexdigitp (pwidstr+6) && hexdigitp (pwidstr+7) && !pwidstr[8])
|
|
|
|
|
{
|
|
|
|
|
/* Hack to help debugging. */
|
|
|
|
|
*r_nks_app_id = NKS_APP_NKS;
|
|
|
|
|
*r_pwid = xtoi_2 (pwidstr+6);
|
|
|
|
|
desc = (new_mode
|
|
|
|
|
? "|N|Please enter a new PIN for the given NKS pwid"
|
|
|
|
|
: "||Please enter the PIN for the given NKS pwid" );
|
|
|
|
|
}
|
|
|
|
|
else if (!strncmp (pwidstr, "SIGG.0x", 7)
|
|
|
|
|
&& hexdigitp (pwidstr+7) && hexdigitp (pwidstr+8) && !pwidstr[9])
|
|
|
|
|
{
|
|
|
|
|
/* Hack to help debugging. */
|
|
|
|
|
*r_nks_app_id = NKS_APP_SIGG;
|
|
|
|
|
*r_pwid = xtoi_2 (pwidstr+7);
|
|
|
|
|
desc = (new_mode
|
|
|
|
|
? "|N|Please enter a new PIN for the given SIGG pwid"
|
|
|
|
|
: "||Please enter the PIN for the given SIGG pwid" );
|
|
|
|
|
}
|
|
|
|
|
else if (!strncmp (pwidstr, "ESIGN.0x", 8)
|
|
|
|
|
&& hexdigitp (pwidstr+8) && hexdigitp (pwidstr+9) && !pwidstr[10])
|
|
|
|
|
{
|
|
|
|
|
/* Hack to help debugging. */
|
|
|
|
|
*r_nks_app_id = NKS_APP_ESIGN;
|
|
|
|
|
*r_pwid = xtoi_2 (pwidstr+8);
|
|
|
|
|
desc = (new_mode
|
|
|
|
|
? "|N|Please enter a new PIN for the given ESIGN pwid"
|
|
|
|
|
: "||Please enter the PIN for the given ESIGN pwid" );
|
|
|
|
|
}
|
2020-05-07 19:44:26 +02:00
|
|
|
|
else if (!strncmp (pwidstr, "IDLM.0x", 7)
|
|
|
|
|
&& hexdigitp (pwidstr+7) && hexdigitp (pwidstr+8) && !pwidstr[9])
|
|
|
|
|
{
|
|
|
|
|
/* Hack to help debugging. */
|
|
|
|
|
*r_nks_app_id = NKS_APP_IDLM;
|
|
|
|
|
*r_pwid = xtoi_2 (pwidstr+7);
|
|
|
|
|
desc = (new_mode
|
|
|
|
|
? "|N|Please enter a new PIN for the given IDLM pwid"
|
|
|
|
|
: "||Please enter the PIN for the given IDLM pwid" );
|
|
|
|
|
}
|
2009-03-05 19:19:37 +00:00
|
|
|
|
else
|
2013-08-30 10:28:26 +02:00
|
|
|
|
{
|
|
|
|
|
*r_pwid = 0; /* Only to avoid gcc warning in calling function. */
|
|
|
|
|
desc = NULL; /* Error. */
|
|
|
|
|
}
|
2009-03-05 19:19:37 +00:00
|
|
|
|
|
|
|
|
|
return desc;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* Handle the PASSWD command. See parse_pwidstr() for allowed values
|
|
|
|
|
for CHVNOSTR. */
|
2011-02-04 12:57:53 +01:00
|
|
|
|
static gpg_error_t
|
|
|
|
|
do_change_pin (app_t app, ctrl_t ctrl, const char *pwidstr,
|
2008-06-24 16:00:29 +00:00
|
|
|
|
unsigned int flags,
|
|
|
|
|
gpg_error_t (*pincb)(void*, const char *, char **),
|
|
|
|
|
void *pincb_arg)
|
|
|
|
|
{
|
|
|
|
|
gpg_error_t err;
|
2009-03-05 19:19:37 +00:00
|
|
|
|
char *newpin = NULL;
|
|
|
|
|
char *oldpin = NULL;
|
|
|
|
|
size_t newpinlen;
|
2008-06-24 16:00:29 +00:00
|
|
|
|
size_t oldpinlen;
|
2020-04-17 16:09:05 +02:00
|
|
|
|
int nks_app_id;
|
2009-03-05 19:19:37 +00:00
|
|
|
|
const char *newdesc;
|
|
|
|
|
int pwid;
|
SCD: API cleanup for keypad handling.
* scd/iso7816.h (struct pininfo_s): Rename from iso7816_pininfo_s.
Change meaning of MODE.
(pininfo_t): Rename from iso7816_pininfo_t.
* scd/sc-copykeys.c: Include "iso7816.h".
* scd/scdaemon.c, scd/command.c: Likewise.
* scd/ccid-driver.c: Include "scdaemon.h" and "iso7816.h".
(ccid_transceive_secure): Follow the change of PININFO_T.
* scd/app.c: Include "apdu.h" after "iso7816.h".
* scd/iso7816.c (iso7816_check_keypad, iso7816_verify_kp)
(iso7816_change_reference_data_kp): Follow the change of API.
* scd/apdu.c (struct reader_table_s): Change API of CHECK_KEYPAD,
KEYPAD_VERIFY, KEYPAD_MODIFY to have arg of PININFO_T.
(check_pcsc_keypad, check_ccid_keypad): Likewise.
(apdu_check_keypad, apdu_keypad_verify, apdu_keypad_modify): Likewise.
(pcsc_keypad_verify, pcsc_keypad_modify, ct_send_apdu)
(pcsc_send_apdu_direct, pcsc_send_apdu_wrapped, pcsc_send_apdu)
(send_apdu_ccid, ccid_keypad_operation, my_rapdu_send_apdu, send_apdu)
(send_le): Follow the change of API.
* scd/apdu.h (apdu_check_keypad, apdu_keypad_verify)
(apdu_keypad_modify): Change the API.
* scd/app-dinsig.c, scd/app-nks.c, scd/app-openpgp.c: Follow the
change.
2013-01-09 14:10:08 +09:00
|
|
|
|
pininfo_t pininfo;
|
2020-05-07 08:18:28 +02:00
|
|
|
|
int remaining;
|
|
|
|
|
char *prompt;
|
2008-06-24 16:00:29 +00:00
|
|
|
|
|
2008-10-20 13:53:23 +00:00
|
|
|
|
(void)ctrl;
|
|
|
|
|
|
2009-03-05 19:19:37 +00:00
|
|
|
|
/* The minimum length is enforced by TCOS, the maximum length is
|
|
|
|
|
just a reasonable value. */
|
|
|
|
|
memset (&pininfo, 0, sizeof pininfo);
|
|
|
|
|
pininfo.minlen = 6;
|
|
|
|
|
pininfo.maxlen = 16;
|
2011-02-04 12:57:53 +01:00
|
|
|
|
|
2020-05-07 14:03:25 +02:00
|
|
|
|
newdesc = parse_pwidstr (app, pwidstr, 1, &nks_app_id, &pwid);
|
2009-03-05 19:19:37 +00:00
|
|
|
|
if (!newdesc)
|
|
|
|
|
return gpg_error (GPG_ERR_INV_ID);
|
|
|
|
|
|
2019-01-21 14:06:51 +01:00
|
|
|
|
if ((flags & APP_CHANGE_FLAG_CLEAR))
|
|
|
|
|
return gpg_error (GPG_ERR_UNSUPPORTED_OPERATION);
|
|
|
|
|
|
2020-04-17 16:09:05 +02:00
|
|
|
|
err = switch_application (app, nks_app_id);
|
2009-03-05 19:19:37 +00:00
|
|
|
|
if (err)
|
|
|
|
|
return err;
|
2008-06-24 16:00:29 +00:00
|
|
|
|
|
|
|
|
|
if ((flags & APP_CHANGE_FLAG_NULLPIN))
|
|
|
|
|
{
|
2009-03-05 19:19:37 +00:00
|
|
|
|
/* With the nullpin flag, we do not verify the PIN - it would
|
|
|
|
|
fail if the Nullpin is still set. */
|
|
|
|
|
oldpin = xtrycalloc (1, 6);
|
|
|
|
|
if (!oldpin)
|
|
|
|
|
{
|
|
|
|
|
err = gpg_error_from_syserror ();
|
|
|
|
|
goto leave;
|
|
|
|
|
}
|
2020-05-07 19:44:26 +02:00
|
|
|
|
if (app->appversion == 15)
|
2020-05-07 08:18:28 +02:00
|
|
|
|
{
|
|
|
|
|
memset (oldpin, '0', 5);
|
|
|
|
|
oldpinlen = 5; /* 5 ascii zeroes. */
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
oldpinlen = 6; /* 6 binary Nuls. */
|
|
|
|
|
}
|
2008-06-24 16:00:29 +00:00
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
2009-03-05 19:19:37 +00:00
|
|
|
|
const char *desc;
|
|
|
|
|
int dummy1, dummy2;
|
|
|
|
|
|
|
|
|
|
if ((flags & APP_CHANGE_FLAG_RESET))
|
|
|
|
|
{
|
|
|
|
|
/* Reset mode: Ask for the alternate PIN. */
|
|
|
|
|
const char *altpwidstr;
|
|
|
|
|
|
|
|
|
|
if (!strcmp (pwidstr, "PW1.CH"))
|
|
|
|
|
altpwidstr = "PW2.CH";
|
|
|
|
|
else if (!strcmp (pwidstr, "PW2.CH"))
|
|
|
|
|
altpwidstr = "PW1.CH";
|
|
|
|
|
else if (!strcmp (pwidstr, "PW1.CH.SIG"))
|
|
|
|
|
altpwidstr = "PW2.CH.SIG";
|
|
|
|
|
else if (!strcmp (pwidstr, "PW2.CH.SIG"))
|
|
|
|
|
altpwidstr = "PW1.CH.SIG";
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
err = gpg_error (GPG_ERR_BUG);
|
|
|
|
|
goto leave;
|
|
|
|
|
}
|
2020-05-07 14:03:25 +02:00
|
|
|
|
desc = parse_pwidstr (app, altpwidstr, 0, &dummy1, &dummy2);
|
2020-06-25 11:25:58 +02:00
|
|
|
|
remaining = iso7816_verify_status (app_get_slot (app), dummy2);
|
2009-03-05 19:19:37 +00:00
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
/* Regular change mode: Ask for the old PIN. */
|
2020-05-07 14:03:25 +02:00
|
|
|
|
desc = parse_pwidstr (app, pwidstr, 0, &dummy1, &dummy2);
|
2020-06-25 11:25:58 +02:00
|
|
|
|
remaining = iso7816_verify_status (app_get_slot (app), pwid);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (remaining < 0)
|
|
|
|
|
remaining = -1; /* We don't care about the concrete error. */
|
|
|
|
|
if (remaining < 3)
|
|
|
|
|
{
|
|
|
|
|
if (remaining >= 0)
|
|
|
|
|
log_info ("nks: PIN has %d attempts left\n", remaining);
|
2009-03-05 19:19:37 +00:00
|
|
|
|
}
|
2020-05-07 08:18:28 +02:00
|
|
|
|
|
|
|
|
|
prompt = make_prompt (app, remaining, desc, NULL);
|
|
|
|
|
err = pincb (pincb_arg, prompt, &oldpin);
|
|
|
|
|
xfree (prompt);
|
2008-06-24 16:00:29 +00:00
|
|
|
|
if (err)
|
2009-03-05 19:19:37 +00:00
|
|
|
|
{
|
|
|
|
|
log_error ("error getting old PIN: %s\n", gpg_strerror (err));
|
|
|
|
|
goto leave;
|
|
|
|
|
}
|
|
|
|
|
oldpinlen = strlen (oldpin);
|
|
|
|
|
err = basic_pin_checks (oldpin, pininfo.minlen, pininfo.maxlen);
|
|
|
|
|
if (err)
|
|
|
|
|
goto leave;
|
2008-06-24 16:00:29 +00:00
|
|
|
|
}
|
|
|
|
|
|
2020-05-07 08:18:28 +02:00
|
|
|
|
|
|
|
|
|
prompt = make_prompt (app, -1, newdesc, NULL);
|
|
|
|
|
err = pincb (pincb_arg, prompt, &newpin);
|
|
|
|
|
xfree (prompt);
|
2008-06-24 16:00:29 +00:00
|
|
|
|
if (err)
|
|
|
|
|
{
|
|
|
|
|
log_error (_("error getting new PIN: %s\n"), gpg_strerror (err));
|
2009-03-05 19:19:37 +00:00
|
|
|
|
goto leave;
|
2008-06-24 16:00:29 +00:00
|
|
|
|
}
|
2009-03-05 19:19:37 +00:00
|
|
|
|
newpinlen = strlen (newpin);
|
2011-02-04 12:57:53 +01:00
|
|
|
|
|
2009-03-05 19:19:37 +00:00
|
|
|
|
err = basic_pin_checks (newpin, pininfo.minlen, pininfo.maxlen);
|
|
|
|
|
if (err)
|
|
|
|
|
goto leave;
|
|
|
|
|
|
|
|
|
|
if ((flags & APP_CHANGE_FLAG_RESET))
|
|
|
|
|
{
|
|
|
|
|
char *data;
|
|
|
|
|
size_t datalen = oldpinlen + newpinlen;
|
2008-06-24 16:00:29 +00:00
|
|
|
|
|
2009-03-05 19:19:37 +00:00
|
|
|
|
data = xtrymalloc (datalen);
|
|
|
|
|
if (!data)
|
|
|
|
|
{
|
|
|
|
|
err = gpg_error_from_syserror ();
|
|
|
|
|
goto leave;
|
|
|
|
|
}
|
|
|
|
|
memcpy (data, oldpin, oldpinlen);
|
|
|
|
|
memcpy (data+oldpinlen, newpin, newpinlen);
|
2019-06-19 08:50:40 +02:00
|
|
|
|
err = iso7816_reset_retry_counter_with_rc (app_get_slot (app), pwid,
|
2009-03-05 19:19:37 +00:00
|
|
|
|
data, datalen);
|
|
|
|
|
wipememory (data, datalen);
|
|
|
|
|
xfree (data);
|
|
|
|
|
}
|
2011-02-04 12:57:53 +01:00
|
|
|
|
else
|
2019-06-19 08:50:40 +02:00
|
|
|
|
err = iso7816_change_reference_data (app_get_slot (app), pwid,
|
2009-03-05 19:19:37 +00:00
|
|
|
|
oldpin, oldpinlen,
|
|
|
|
|
newpin, newpinlen);
|
|
|
|
|
leave:
|
|
|
|
|
xfree (oldpin);
|
|
|
|
|
xfree (newpin);
|
2008-06-24 16:00:29 +00:00
|
|
|
|
return err;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* Perform a simple verify operation. KEYIDSTR should be NULL or empty. */
|
2011-02-04 12:57:53 +01:00
|
|
|
|
static gpg_error_t
|
2020-01-07 18:45:33 +01:00
|
|
|
|
do_check_pin (app_t app, ctrl_t ctrl, const char *pwidstr,
|
2008-06-24 16:00:29 +00:00
|
|
|
|
gpg_error_t (*pincb)(void*, const char *, char **),
|
|
|
|
|
void *pincb_arg)
|
|
|
|
|
{
|
2009-03-05 19:19:37 +00:00
|
|
|
|
gpg_error_t err;
|
|
|
|
|
int pwid;
|
2020-04-17 16:09:05 +02:00
|
|
|
|
int nks_app_id;
|
2009-03-05 19:19:37 +00:00
|
|
|
|
const char *desc;
|
|
|
|
|
|
2020-01-07 18:45:33 +01:00
|
|
|
|
(void)ctrl;
|
|
|
|
|
|
2020-05-07 14:03:25 +02:00
|
|
|
|
desc = parse_pwidstr (app, pwidstr, 0, &nks_app_id, &pwid);
|
2009-03-05 19:19:37 +00:00
|
|
|
|
if (!desc)
|
|
|
|
|
return gpg_error (GPG_ERR_INV_ID);
|
|
|
|
|
|
2020-04-17 16:09:05 +02:00
|
|
|
|
err = switch_application (app, nks_app_id);
|
2009-03-05 19:19:37 +00:00
|
|
|
|
if (err)
|
|
|
|
|
return err;
|
|
|
|
|
|
|
|
|
|
return verify_pin (app, pwid, desc, pincb, pincb_arg);
|
2008-06-24 16:00:29 +00:00
|
|
|
|
}
|
|
|
|
|
|
2004-01-27 16:40:42 +00:00
|
|
|
|
|
2020-05-04 19:01:16 +02:00
|
|
|
|
/* 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 keygripstr[2*KEYGRIP_LEN+1];
|
|
|
|
|
char *serialno = NULL;
|
|
|
|
|
char idbuf[20];
|
|
|
|
|
int data = 0;
|
|
|
|
|
int idx;
|
|
|
|
|
const char *tagstr;
|
|
|
|
|
|
|
|
|
|
/* 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:
|
|
|
|
|
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 (idx=0; filelist[idx].fid; idx++)
|
|
|
|
|
{
|
2020-05-07 19:44:26 +02:00
|
|
|
|
if (filelist[idx].nks_ver > app->appversion)
|
2020-05-04 19:01:16 +02:00
|
|
|
|
continue;
|
|
|
|
|
|
|
|
|
|
if (!filelist[idx].iskeypair)
|
|
|
|
|
continue;
|
|
|
|
|
|
2020-05-07 19:44:26 +02:00
|
|
|
|
if (app->app_local->only_idlm)
|
|
|
|
|
{
|
|
|
|
|
if (filelist[idx].nks_app_id != NKS_APP_IDLM)
|
|
|
|
|
continue;
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
if (filelist[idx].nks_app_id != NKS_APP_NKS
|
|
|
|
|
&& filelist[idx].nks_app_id != app->app_local->qes_app_id)
|
|
|
|
|
continue;
|
|
|
|
|
err = switch_application (app, filelist[idx].nks_app_id);
|
|
|
|
|
if (err)
|
|
|
|
|
goto leave;
|
|
|
|
|
}
|
2020-05-04 19:01:16 +02:00
|
|
|
|
|
|
|
|
|
err = keygripstr_from_pk_file (app, filelist[idx].fid,
|
2020-09-21 14:47:53 +02:00
|
|
|
|
filelist[idx].iskeypair, keygripstr,
|
|
|
|
|
NULL, NULL);
|
2020-05-04 19:01:16 +02:00
|
|
|
|
if (err)
|
|
|
|
|
{
|
|
|
|
|
log_error ("can't get keygrip from FID 0x%04X: %s\n",
|
|
|
|
|
filelist[idx].fid, gpg_strerror (err));
|
|
|
|
|
continue;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (action == KEYGRIP_ACTION_LOOKUP)
|
|
|
|
|
{
|
|
|
|
|
if (!strcmp (keygripstr, want_keygripstr))
|
|
|
|
|
{
|
|
|
|
|
err = 0; /* Found */
|
|
|
|
|
goto leave;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
else if (!want_keygripstr || !strcmp (keygripstr, want_keygripstr))
|
|
|
|
|
{
|
|
|
|
|
if (capability == GCRY_PK_USAGE_SIGN)
|
|
|
|
|
{
|
|
|
|
|
if (!filelist[idx].issignkey)
|
|
|
|
|
continue;
|
|
|
|
|
}
|
|
|
|
|
if (capability == GCRY_PK_USAGE_ENCR)
|
|
|
|
|
{
|
|
|
|
|
if (!filelist[idx].isencrkey)
|
|
|
|
|
continue;
|
|
|
|
|
}
|
|
|
|
|
if (capability == GCRY_PK_USAGE_AUTH)
|
|
|
|
|
{
|
|
|
|
|
if (!filelist[idx].isauthkey)
|
|
|
|
|
continue;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (app->app_local->active_nks_app == NKS_APP_ESIGN)
|
|
|
|
|
tagstr = "ESIGN";
|
|
|
|
|
else if (app->app_local->active_nks_app == NKS_APP_SIGG)
|
|
|
|
|
tagstr = "SIGG";
|
2020-05-07 19:44:26 +02:00
|
|
|
|
else if (app->app_local->active_nks_app == NKS_APP_IDLM)
|
|
|
|
|
tagstr = "IDLM";
|
|
|
|
|
else if (app->appversion < 3)
|
2020-05-04 19:01:16 +02:00
|
|
|
|
tagstr = "DF01";
|
|
|
|
|
else
|
|
|
|
|
tagstr = "NKS3";
|
|
|
|
|
snprintf (idbuf, sizeof idbuf, "NKS-%s.%04X",
|
|
|
|
|
tagstr, filelist[idx].fid);
|
|
|
|
|
send_keyinfo (ctrl, data, keygripstr, serialno, idbuf);
|
|
|
|
|
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);
|
|
|
|
|
return err;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2009-01-08 19:56:30 +00:00
|
|
|
|
/* Return the version of the NKS application. */
|
|
|
|
|
static int
|
|
|
|
|
get_nks_version (int slot)
|
|
|
|
|
{
|
|
|
|
|
unsigned char *result = NULL;
|
|
|
|
|
size_t resultlen;
|
|
|
|
|
int type;
|
|
|
|
|
|
2011-02-04 12:57:53 +01:00
|
|
|
|
if (iso7816_apdu_direct (slot, "\x80\xaa\x06\x00\x00", 5, 0,
|
2019-01-20 11:41:23 +01:00
|
|
|
|
NULL, &result, &resultlen))
|
2009-01-08 19:56:30 +00:00
|
|
|
|
return 2; /* NKS 2 does not support this command. */
|
2020-04-17 16:09:05 +02:00
|
|
|
|
/* Example values: 04 11 19 22 21 6A 20 80 03 03 01 01 01 00 00 00
|
|
|
|
|
* 05 a0 22 3e c8 0c 04 20 0f 01 b6 01 01 00 00 02
|
|
|
|
|
* vv tt ccccccccccccccccc aa bb cc vv ff rr rr xx
|
|
|
|
|
* vendor -----------+ | | | | | | | | | |
|
|
|
|
|
* chip type -----------+ | | | | | | | | |
|
|
|
|
|
* chip id ----------------+ | | | | | | | |
|
|
|
|
|
* card type --------------------------------+ | | | | | | |
|
|
|
|
|
* OS version of card type ---------------------+ | | | | | |
|
|
|
|
|
* OS release of card type ------------------------+ | | | | |
|
|
|
|
|
* Completion code version number --------------------+ | | | |
|
|
|
|
|
* File system version ----------------------------------+ | | |
|
|
|
|
|
* RFU (00) ------------------------------------------------+ | |
|
|
|
|
|
* RFU (00) ---------------------------------------------------+ |
|
|
|
|
|
* Authentication key identifier ---------------------------------+
|
|
|
|
|
*
|
|
|
|
|
* vendor 4 := Philips
|
|
|
|
|
* 5 := Infinion
|
|
|
|
|
* card type 3 := TCOS 3
|
2020-05-07 19:44:26 +02:00
|
|
|
|
* 15 := TCOS Signature Card (bb,cc is the ROM mask version)
|
2020-04-17 16:09:05 +02:00
|
|
|
|
* Completion code version number Bit 7..5 := pre-completion code version
|
|
|
|
|
* Bit 4..0 := completion code version
|
|
|
|
|
* (pre-completion by chip vendor)
|
|
|
|
|
* (completion by OS developer)
|
|
|
|
|
*/
|
2009-01-08 19:56:30 +00:00
|
|
|
|
if (resultlen < 16)
|
|
|
|
|
type = 0; /* Invalid data returned. */
|
|
|
|
|
else
|
|
|
|
|
type = result[8];
|
|
|
|
|
xfree (result);
|
|
|
|
|
return type;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2020-04-17 16:09:05 +02:00
|
|
|
|
/* Switch to the NKS app identified by NKS_APP_ID if not yet done.
|
|
|
|
|
* Returns 0 on success. */
|
2009-03-05 19:19:37 +00:00
|
|
|
|
static gpg_error_t
|
2020-04-17 16:09:05 +02:00
|
|
|
|
switch_application (app_t app, int nks_app_id)
|
2009-03-05 19:19:37 +00:00
|
|
|
|
{
|
|
|
|
|
gpg_error_t err;
|
|
|
|
|
|
2020-05-07 19:44:26 +02:00
|
|
|
|
if (app->app_local->only_idlm)
|
|
|
|
|
return 0; /* No switching at all */
|
2020-04-17 16:09:05 +02:00
|
|
|
|
if (app->app_local->active_nks_app == nks_app_id
|
2009-05-08 15:07:45 +00:00
|
|
|
|
&& !app->app_local->need_app_select)
|
2009-03-05 19:19:37 +00:00
|
|
|
|
return 0; /* Already switched. */
|
|
|
|
|
|
2020-04-17 16:09:05 +02:00
|
|
|
|
log_info ("nks: switching to %s\n",
|
|
|
|
|
nks_app_id == NKS_APP_ESIGN? "eSign" :
|
|
|
|
|
nks_app_id == NKS_APP_SIGG? "SigG" : "NKS");
|
|
|
|
|
|
|
|
|
|
if (nks_app_id == NKS_APP_ESIGN)
|
|
|
|
|
err = iso7816_select_application (app_get_slot (app),
|
|
|
|
|
aid_esign, sizeof aid_esign, 0);
|
|
|
|
|
else if (nks_app_id == NKS_APP_SIGG)
|
2019-06-19 08:50:40 +02:00
|
|
|
|
err = iso7816_select_application (app_get_slot (app),
|
|
|
|
|
aid_sigg, sizeof aid_sigg, 0);
|
2009-03-05 19:19:37 +00:00
|
|
|
|
else
|
2019-06-19 08:50:40 +02:00
|
|
|
|
err = iso7816_select_application (app_get_slot (app),
|
|
|
|
|
aid_nks, sizeof aid_nks, 0);
|
2009-05-08 15:07:45 +00:00
|
|
|
|
|
2020-04-17 16:09:05 +02:00
|
|
|
|
if (!err && nks_app_id == NKS_APP_SIGG
|
2020-05-07 19:44:26 +02:00
|
|
|
|
&& app->appversion >= 3
|
2009-05-08 15:07:45 +00:00
|
|
|
|
&& !app->app_local->sigg_msig_checked)
|
|
|
|
|
{
|
|
|
|
|
/* Check whether this card is a mass signature card. */
|
|
|
|
|
unsigned char *buffer;
|
|
|
|
|
size_t buflen;
|
|
|
|
|
const unsigned char *tmpl;
|
|
|
|
|
size_t tmpllen;
|
2011-02-04 12:57:53 +01:00
|
|
|
|
|
2009-05-08 15:07:45 +00:00
|
|
|
|
app->app_local->sigg_msig_checked = 1;
|
|
|
|
|
app->app_local->sigg_is_msig = 1;
|
2019-06-19 08:50:40 +02:00
|
|
|
|
err = iso7816_select_file (app_get_slot (app), 0x5349, 0);
|
2009-05-08 15:07:45 +00:00
|
|
|
|
if (!err)
|
2019-06-19 08:50:40 +02:00
|
|
|
|
err = iso7816_read_record (app_get_slot (app), 1, 1, 0,
|
|
|
|
|
&buffer, &buflen);
|
2009-05-08 15:07:45 +00:00
|
|
|
|
if (!err)
|
|
|
|
|
{
|
|
|
|
|
tmpl = find_tlv (buffer, buflen, 0x7a, &tmpllen);
|
2011-02-04 12:57:53 +01:00
|
|
|
|
if (tmpl && tmpllen == 12
|
2009-05-08 15:07:45 +00:00
|
|
|
|
&& !memcmp (tmpl,
|
|
|
|
|
"\x93\x02\x00\x01\xA4\x06\x83\x01\x81\x83\x01\x83",
|
|
|
|
|
12))
|
|
|
|
|
app->app_local->sigg_is_msig = 0;
|
|
|
|
|
xfree (buffer);
|
|
|
|
|
}
|
|
|
|
|
if (app->app_local->sigg_is_msig)
|
2020-04-17 16:09:05 +02:00
|
|
|
|
log_info ("nks: This is a mass signature card\n");
|
2009-05-08 15:07:45 +00:00
|
|
|
|
}
|
2011-02-04 12:57:53 +01:00
|
|
|
|
|
2009-03-05 19:19:37 +00:00
|
|
|
|
if (!err)
|
2009-05-08 15:07:45 +00:00
|
|
|
|
{
|
|
|
|
|
app->app_local->need_app_select = 0;
|
2020-04-17 16:09:05 +02:00
|
|
|
|
app->app_local->active_nks_app = nks_app_id;
|
2009-05-08 15:07:45 +00:00
|
|
|
|
}
|
2009-03-05 19:19:37 +00:00
|
|
|
|
else
|
2020-04-17 16:09:05 +02:00
|
|
|
|
log_error ("nks: error switching to %s: %s\n",
|
|
|
|
|
nks_app_id == NKS_APP_ESIGN? "eSign" :
|
|
|
|
|
nks_app_id == NKS_APP_SIGG? "SigG" : "NKS",
|
|
|
|
|
gpg_strerror (err));
|
2009-03-05 19:19:37 +00:00
|
|
|
|
|
|
|
|
|
return err;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2009-01-08 19:56:30 +00:00
|
|
|
|
/* Select the NKS application. */
|
2005-05-18 10:48:06 +00:00
|
|
|
|
gpg_error_t
|
2006-09-06 16:35:52 +00:00
|
|
|
|
app_select_nks (app_t app)
|
2004-01-27 16:40:42 +00:00
|
|
|
|
{
|
2019-06-19 08:50:40 +02:00
|
|
|
|
int slot = app_get_slot (app);
|
2004-01-27 16:40:42 +00:00
|
|
|
|
int rc;
|
2020-05-07 19:44:26 +02:00
|
|
|
|
int is_idlm = 0;
|
2011-02-04 12:57:53 +01:00
|
|
|
|
|
2009-03-05 19:19:37 +00:00
|
|
|
|
rc = iso7816_select_application (slot, aid_nks, sizeof aid_nks, 0);
|
2020-05-07 19:44:26 +02:00
|
|
|
|
if (rc)
|
|
|
|
|
{
|
|
|
|
|
is_idlm = 1;
|
|
|
|
|
rc = iso7816_select_application (slot, aid_idlm, sizeof aid_idlm, 0);
|
|
|
|
|
}
|
2004-01-27 16:40:42 +00:00
|
|
|
|
if (!rc)
|
|
|
|
|
{
|
2019-06-19 14:30:16 +02:00
|
|
|
|
app->apptype = APPTYPE_NKS;
|
2004-01-27 16:40:42 +00:00
|
|
|
|
|
2009-01-08 19:56:30 +00:00
|
|
|
|
app->app_local = xtrycalloc (1, sizeof *app->app_local);
|
|
|
|
|
if (!app->app_local)
|
|
|
|
|
{
|
|
|
|
|
rc = gpg_error (gpg_err_code_from_errno (errno));
|
|
|
|
|
goto leave;
|
|
|
|
|
}
|
|
|
|
|
|
2020-05-07 19:44:26 +02:00
|
|
|
|
app->appversion = get_nks_version (slot);
|
|
|
|
|
app->app_local->only_idlm = is_idlm;
|
|
|
|
|
if (is_idlm) /* Set it once, there won't be any switching. */
|
|
|
|
|
app->app_local->active_nks_app = NKS_APP_IDLM;
|
|
|
|
|
|
2009-01-08 19:56:30 +00:00
|
|
|
|
if (opt.verbose)
|
2020-05-07 19:44:26 +02:00
|
|
|
|
{
|
|
|
|
|
log_info ("Detected NKS version: %d\n", app->appversion);
|
|
|
|
|
if (is_idlm)
|
|
|
|
|
log_info ("Using only the IDLM application\n");
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (app->appversion == 15)
|
2020-04-17 16:09:05 +02:00
|
|
|
|
app->app_local->qes_app_id = NKS_APP_ESIGN;
|
|
|
|
|
else
|
|
|
|
|
app->app_local->qes_app_id = NKS_APP_SIGG;
|
2009-01-08 19:56:30 +00:00
|
|
|
|
|
|
|
|
|
app->fnc.deinit = do_deinit;
|
2020-01-07 18:45:33 +01:00
|
|
|
|
app->fnc.prep_reselect = NULL;
|
scd: Add an re-select mechanism to switch apps.
* scd/app-common.h (struct app_ctx_s): Add func ptr 'reselect'.
* scd/app-piv.c (do_reselect): New.
(app_select_piv): Move AID constant to file scope.
* scd/app-openpgp.c (do_reselect): New.
(app_select_openpgp): Move AID constant to file scope.
* scd/app.c (apptype_from_name): New.
(check_application_conflict): Check against all apps of the card.
Always set current_apptype.
(select_additional_application): New.
(maybe_switch_app): New.
(app_write_learn_status, app_readcert, app_readkey, app_getattr)
(app_setattr, app_sign, app_auth, app_decipher, app_writecert)
(app_writekey, app_genkey, app_change_pin, app_check_pin): Use it here.
(app_do_with_keygrip): Force reselect on success.
(app_new_register): Move setting of CURRENT_APPTYPE to ...
(select_application): here so that it will be set to the requested
card.
* scd/command.c (open_card_with_request): Select additional
application if possible.
--
Noet that we will likely need to rework this even more so to get well
defined semantics for card access.
Signed-off-by: Werner Koch <wk@gnupg.org>
2019-06-25 08:30:04 +02:00
|
|
|
|
app->fnc.reselect = NULL;
|
2004-01-27 16:40:42 +00:00
|
|
|
|
app->fnc.learn_status = do_learn_status;
|
|
|
|
|
app->fnc.readcert = do_readcert;
|
2009-05-08 15:07:45 +00:00
|
|
|
|
app->fnc.readkey = do_readkey;
|
2009-03-05 19:19:37 +00:00
|
|
|
|
app->fnc.getattr = do_getattr;
|
2004-01-27 16:40:42 +00:00
|
|
|
|
app->fnc.setattr = NULL;
|
2020-07-02 18:35:34 +02:00
|
|
|
|
app->fnc.writecert = do_writecert;
|
2009-05-13 11:42:34 +00:00
|
|
|
|
app->fnc.writekey = do_writekey;
|
2004-01-27 16:40:42 +00:00
|
|
|
|
app->fnc.genkey = NULL;
|
2004-01-28 16:21:57 +00:00
|
|
|
|
app->fnc.sign = do_sign;
|
2004-01-27 16:40:42 +00:00
|
|
|
|
app->fnc.auth = NULL;
|
2004-01-28 16:21:57 +00:00
|
|
|
|
app->fnc.decipher = do_decipher;
|
2008-06-24 16:00:29 +00:00
|
|
|
|
app->fnc.change_pin = do_change_pin;
|
|
|
|
|
app->fnc.check_pin = do_check_pin;
|
2020-05-04 19:01:16 +02:00
|
|
|
|
app->fnc.with_keygrip = do_with_keygrip;
|
2004-01-27 16:40:42 +00:00
|
|
|
|
}
|
|
|
|
|
|
2009-01-08 19:56:30 +00:00
|
|
|
|
leave:
|
|
|
|
|
if (rc)
|
|
|
|
|
do_deinit (app);
|
2004-01-27 16:40:42 +00:00
|
|
|
|
return rc;
|
|
|
|
|
}
|