mirror of
git://git.gnupg.org/gnupg.git
synced 2025-01-06 12:33:23 +01:00
tkd: sorted out entities and initialize PKCS#11 on demand.
tkd: host side process which manages KEYs on TOKENs It's a KEYGRIP which identifies a KEY. it's TKD which administrates: where is a KEY on which TOKEN. cryptoki (ck): access object of PKCS#11 API. CRYPTOKI has slots. token: abstraction of a device. Each slots has slot_id. Slot_id corresponds a TOKEN. 1:1 map between slot_id and TOKEN. KEY: TOKEN may have multiple KEYs. Signed-off-by: NIIBE Yutaka <gniibe@fsij.org>
This commit is contained in:
parent
a53441c8dd
commit
22be07a08c
@ -248,7 +248,7 @@ cmd_slotlist (assuan_context_t ctx, char *line)
|
|||||||
line = skip_options (line);
|
line = skip_options (line);
|
||||||
(void)line;
|
(void)line;
|
||||||
|
|
||||||
err = token_init (ctrl, ctx, 1);
|
err = tkd_init (ctrl, ctx, 1);
|
||||||
return err;
|
return err;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -268,7 +268,7 @@ cmd_readkey (assuan_context_t ctx, char *line)
|
|||||||
if (strlen (keygrip) != 40)
|
if (strlen (keygrip) != 40)
|
||||||
err = gpg_error (GPG_ERR_INV_ID);
|
err = gpg_error (GPG_ERR_INV_ID);
|
||||||
|
|
||||||
err = token_readkey (ctrl, ctx, keygrip);
|
err = tkd_readkey (ctrl, ctx, keygrip);
|
||||||
|
|
||||||
return err;
|
return err;
|
||||||
}
|
}
|
||||||
@ -305,10 +305,10 @@ cmd_pksign (assuan_context_t ctx, char *line)
|
|||||||
if (strlen (keygrip) != 40)
|
if (strlen (keygrip) != 40)
|
||||||
err = gpg_error (GPG_ERR_INV_ID);
|
err = gpg_error (GPG_ERR_INV_ID);
|
||||||
|
|
||||||
err = token_sign (ctrl, ctx, keygrip, hash_algo, &outdata, &outdatalen);
|
err = tkd_sign (ctrl, ctx, keygrip, hash_algo, &outdata, &outdatalen);
|
||||||
if (err)
|
if (err)
|
||||||
{
|
{
|
||||||
log_error ("token_sign failed: %s\n", gpg_strerror (err));
|
log_error ("tkd_sign failed: %s\n", gpg_strerror (err));
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
@ -385,7 +385,7 @@ cmd_keyinfo (assuan_context_t ctx, char *line)
|
|||||||
else
|
else
|
||||||
keygrip = skip_options (line);
|
keygrip = skip_options (line);
|
||||||
|
|
||||||
err = token_keyinfo (ctrl, keygrip, opt_data, cap);
|
err = tkd_keyinfo (ctrl, ctx, keygrip, opt_data, cap);
|
||||||
|
|
||||||
return err;
|
return err;
|
||||||
}
|
}
|
||||||
|
182
tkd/pkcs11.c
182
tkd/pkcs11.c
@ -146,8 +146,7 @@ get_slot_list (struct cryptoki *ck,
|
|||||||
}
|
}
|
||||||
|
|
||||||
static long
|
static long
|
||||||
get_token_info (struct token *token,
|
get_token_info (struct token *token, struct ck_token_info *tk_info)
|
||||||
struct ck_token_info *tk_info)
|
|
||||||
{
|
{
|
||||||
unsigned long err = 0;
|
unsigned long err = 0;
|
||||||
struct cryptoki *ck = token->ck;
|
struct cryptoki *ck = token->ck;
|
||||||
@ -225,9 +224,6 @@ close_session (struct token *token)
|
|||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
|
|
||||||
token->session = 0;
|
|
||||||
token->num_keys = 0;
|
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -972,8 +968,8 @@ do_pksign (struct key *key, int hash_algo,
|
|||||||
return err;
|
return err;
|
||||||
}
|
}
|
||||||
|
|
||||||
gpg_error_t
|
static gpg_error_t
|
||||||
token_new (assuan_context_t ctx, struct cryptoki *ck, struct token *token,
|
token_open (assuan_context_t ctx, struct cryptoki *ck, struct token *token,
|
||||||
ck_slot_id_t slot_id)
|
ck_slot_id_t slot_id)
|
||||||
{
|
{
|
||||||
gpg_error_t err = 0;
|
gpg_error_t err = 0;
|
||||||
@ -1021,6 +1017,7 @@ token_new (assuan_context_t ctx, struct cryptoki *ck, struct token *token,
|
|||||||
if (err)
|
if (err)
|
||||||
{
|
{
|
||||||
close_session (token);
|
close_session (token);
|
||||||
|
token->session = 0;
|
||||||
return err;
|
return err;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -1032,9 +1029,79 @@ token_new (assuan_context_t ctx, struct cryptoki *ck, struct token *token,
|
|||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static gpg_error_t
|
||||||
|
token_close (struct token *token)
|
||||||
|
{
|
||||||
|
int j;
|
||||||
|
long r;
|
||||||
|
int num_keys = token->num_keys;
|
||||||
|
|
||||||
|
if (!token->valid)
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
if (token->login_required)
|
||||||
|
logout (token);
|
||||||
|
|
||||||
|
r = close_session (token);
|
||||||
|
if (r)
|
||||||
|
log_error ("Error at close_session: %ld\n", r);
|
||||||
|
|
||||||
|
token->ck = NULL;
|
||||||
|
token->slot_id = 0;
|
||||||
|
token->login_required = 0;
|
||||||
|
token->session = 0;
|
||||||
|
token->num_keys = 0;
|
||||||
|
|
||||||
|
for (j = 0; j < num_keys; j++)
|
||||||
|
{
|
||||||
|
struct key *k = &token->key_list[j];
|
||||||
|
|
||||||
|
if ((k->flags & KEY_FLAG_VALID))
|
||||||
|
{
|
||||||
|
gcry_sexp_release (k->pubkey);
|
||||||
|
k->pubkey = NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
k->token = NULL;
|
||||||
|
k->flags = 0;
|
||||||
|
k->key_type = 0;
|
||||||
|
k->label_len = 0;
|
||||||
|
k->id_len = 0;
|
||||||
|
k->p11_keyid = 0;
|
||||||
|
k->mechanism = 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
token->valid = 0;
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
static gpg_error_t
|
||||||
|
token_check (struct token *token)
|
||||||
|
{
|
||||||
|
struct ck_token_info tk_info;
|
||||||
|
|
||||||
|
if (get_token_info (token, &tk_info))
|
||||||
|
{
|
||||||
|
/* Possibly, invalidate the token and close session.
|
||||||
|
* Now, ingore the error. */
|
||||||
|
return gpg_error (GPG_ERR_INV_RESPONSE);
|
||||||
|
}
|
||||||
|
|
||||||
|
if ((tk_info.flags & CKF_TOKEN_INITIALIZED) == 0
|
||||||
|
|| (tk_info.flags & CKF_TOKEN_PRESENT) == 0
|
||||||
|
|| (tk_info.flags & CKF_USER_PIN_LOCKED) != 0)
|
||||||
|
{
|
||||||
|
token_close (token);
|
||||||
|
return gpg_error (GPG_ERR_CARD_NOT_PRESENT);
|
||||||
|
}
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
gpg_error_t
|
gpg_error_t
|
||||||
token_init (ctrl_t ctrl, assuan_context_t ctx, int rescan)
|
tkd_init (ctrl_t ctrl, assuan_context_t ctx, int rescan)
|
||||||
{
|
{
|
||||||
gpg_error_t err = 0;
|
gpg_error_t err = 0;
|
||||||
|
|
||||||
@ -1083,7 +1150,7 @@ token_init (ctrl_t ctrl, assuan_context_t ctx, int rescan)
|
|||||||
{
|
{
|
||||||
struct token *token = &ck->token_list[num_tokens]; /* Allocate one token in CK */
|
struct token *token = &ck->token_list[num_tokens]; /* Allocate one token in CK */
|
||||||
|
|
||||||
err = token_new (ctx, ck, token, slot_list[i]);
|
err = token_open (ctx, ck, token, slot_list[i]);
|
||||||
if (!err)
|
if (!err)
|
||||||
num_tokens++;
|
num_tokens++;
|
||||||
}
|
}
|
||||||
@ -1099,37 +1166,33 @@ token_init (ctrl_t ctrl, assuan_context_t ctx, int rescan)
|
|||||||
r = get_slot_list (ck, &num_slots, slot_list);
|
r = get_slot_list (ck, &num_slots, slot_list);
|
||||||
if (r)
|
if (r)
|
||||||
{
|
{
|
||||||
token_fini (ctrl, ctx);
|
tkd_fini (ctrl, ctx);
|
||||||
return gpg_error (GPG_ERR_INV_RESPONSE);
|
return gpg_error (GPG_ERR_INV_RESPONSE);
|
||||||
}
|
}
|
||||||
|
|
||||||
for (i = 0; i < num_slots; i++)
|
for (i = 0; i < num_slots; i++)
|
||||||
{
|
{
|
||||||
ck_slot_id_t slot_id = slot_list[i];
|
|
||||||
int j;
|
int j;
|
||||||
int found = 0;
|
ck_slot_id_t slot_id = slot_list[i];
|
||||||
|
struct token *token = NULL;
|
||||||
|
|
||||||
for (j = 0; j < ck->num_slots; j++)
|
for (j = 0; j < ck->num_slots; j++)
|
||||||
|
if (slot_id == ck->token_list[j].slot_id)
|
||||||
{
|
{
|
||||||
struct token *token = &ck->token_list[j];
|
token = &ck->token_list[j];
|
||||||
|
|
||||||
if (slot_id == token->slot_id)
|
|
||||||
{
|
|
||||||
found = 1;
|
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
|
||||||
if (found)
|
if (token)
|
||||||
{
|
{
|
||||||
/*XXX: rescan the keys??? */
|
err = token_check (token);
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
/* new token */
|
/* new token */
|
||||||
{
|
{
|
||||||
/* Allocate one token in CK */
|
/* Allocate one token in CK */
|
||||||
struct token *token = &ck->token_list[ck->num_slots];
|
token = &ck->token_list[ck->num_slots];
|
||||||
err = token_new (ctx, ck, token, slot_id);
|
err = token_open (ctx, ck, token, slot_id);
|
||||||
if (!err)
|
if (!err)
|
||||||
ck->num_slots++;
|
ck->num_slots++;
|
||||||
}
|
}
|
||||||
@ -1139,12 +1202,11 @@ token_init (ctrl_t ctrl, assuan_context_t ctx, int rescan)
|
|||||||
}
|
}
|
||||||
|
|
||||||
gpg_error_t
|
gpg_error_t
|
||||||
token_fini (ctrl_t ctrl, assuan_context_t ctx)
|
tkd_fini (ctrl_t ctrl, assuan_context_t ctx)
|
||||||
{
|
{
|
||||||
long r;
|
long r;
|
||||||
struct cryptoki *ck = ck_instance;
|
struct cryptoki *ck = ck_instance;
|
||||||
int i;
|
int i;
|
||||||
int j;
|
|
||||||
|
|
||||||
(void)ctrl;
|
(void)ctrl;
|
||||||
(void)ctx;
|
(void)ctx;
|
||||||
@ -1153,45 +1215,7 @@ token_fini (ctrl_t ctrl, assuan_context_t ctx)
|
|||||||
{
|
{
|
||||||
struct token *token = &ck->token_list[i];
|
struct token *token = &ck->token_list[i];
|
||||||
|
|
||||||
if (!token->valid)
|
token_close (token);
|
||||||
{
|
|
||||||
token->ck = NULL;
|
|
||||||
token->slot_id = 0;
|
|
||||||
token->login_required = 0;
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (token->login_required)
|
|
||||||
logout (token);
|
|
||||||
|
|
||||||
r = close_session (token);
|
|
||||||
if (r)
|
|
||||||
log_error ("Error at close_session: %ld\n", r);
|
|
||||||
|
|
||||||
token->ck = NULL;
|
|
||||||
token->slot_id = 0;
|
|
||||||
token->login_required = 0;
|
|
||||||
|
|
||||||
for (j = 0; j < token->num_keys; j++)
|
|
||||||
{
|
|
||||||
struct key *k = &token->key_list[i];
|
|
||||||
|
|
||||||
if ((k->flags & KEY_FLAG_VALID))
|
|
||||||
{
|
|
||||||
gcry_sexp_release (k->pubkey);
|
|
||||||
k->pubkey = NULL;
|
|
||||||
}
|
|
||||||
|
|
||||||
k->token = NULL;
|
|
||||||
k->flags = 0;
|
|
||||||
k->key_type = 0;
|
|
||||||
k->label_len = 0;
|
|
||||||
k->id_len = 0;
|
|
||||||
k->p11_keyid = 0;
|
|
||||||
k->mechanism = 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
token->valid = 0;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
ck->num_slots = 0;
|
ck->num_slots = 0;
|
||||||
@ -1212,10 +1236,9 @@ token_fini (ctrl_t ctrl, assuan_context_t ctx)
|
|||||||
|
|
||||||
|
|
||||||
gpg_error_t
|
gpg_error_t
|
||||||
token_sign (ctrl_t ctrl, assuan_context_t ctx,
|
tkd_sign (ctrl_t ctrl, assuan_context_t ctx,
|
||||||
const char *keygrip, int hash_algo,
|
const char *keygrip, int hash_algo,
|
||||||
unsigned char **r_outdata,
|
unsigned char **r_outdata, size_t *r_outdatalen)
|
||||||
size_t *r_outdatalen)
|
|
||||||
{
|
{
|
||||||
gpg_error_t err;
|
gpg_error_t err;
|
||||||
struct key *k;
|
struct key *k;
|
||||||
@ -1226,6 +1249,13 @@ token_sign (ctrl_t ctrl, assuan_context_t ctx,
|
|||||||
/* mismatch: size_t for GnuPG, unsigned long for PKCS#11 */
|
/* mismatch: size_t for GnuPG, unsigned long for PKCS#11 */
|
||||||
/* mismatch: application prepare buffer for PKCS#11 */
|
/* mismatch: application prepare buffer for PKCS#11 */
|
||||||
|
|
||||||
|
if (!ck->handle)
|
||||||
|
{
|
||||||
|
err = tkd_init (ctrl, ctx, 0);
|
||||||
|
if (err)
|
||||||
|
return err;
|
||||||
|
}
|
||||||
|
|
||||||
*r_outdata = NULL;
|
*r_outdata = NULL;
|
||||||
r = find_key (ck, keygrip, &k);
|
r = find_key (ck, keygrip, &k);
|
||||||
if (r)
|
if (r)
|
||||||
@ -1287,7 +1317,7 @@ get_usage_string (struct key *k)
|
|||||||
}
|
}
|
||||||
|
|
||||||
gpg_error_t
|
gpg_error_t
|
||||||
token_readkey (ctrl_t ctrl, assuan_context_t ctx, const char *keygrip)
|
tkd_readkey (ctrl_t ctrl, assuan_context_t ctx, const char *keygrip)
|
||||||
{
|
{
|
||||||
gpg_error_t err = 0;
|
gpg_error_t err = 0;
|
||||||
struct key *k;
|
struct key *k;
|
||||||
@ -1296,6 +1326,14 @@ token_readkey (ctrl_t ctrl, assuan_context_t ctx, const char *keygrip)
|
|||||||
|
|
||||||
(void)ctrl;
|
(void)ctrl;
|
||||||
(void)ctx;
|
(void)ctx;
|
||||||
|
|
||||||
|
if (!ck->handle)
|
||||||
|
{
|
||||||
|
err = tkd_init (ctrl, ctx, 0);
|
||||||
|
if (err)
|
||||||
|
return err;
|
||||||
|
}
|
||||||
|
|
||||||
r = find_key (ck, keygrip, &k);
|
r = find_key (ck, keygrip, &k);
|
||||||
if (r)
|
if (r)
|
||||||
return gpg_error (GPG_ERR_NO_SECKEY);
|
return gpg_error (GPG_ERR_NO_SECKEY);
|
||||||
@ -1304,13 +1342,21 @@ token_readkey (ctrl_t ctrl, assuan_context_t ctx, const char *keygrip)
|
|||||||
}
|
}
|
||||||
|
|
||||||
gpg_error_t
|
gpg_error_t
|
||||||
token_keyinfo (ctrl_t ctrl, const char *keygrip, int opt_data, int cap)
|
tkd_keyinfo (ctrl_t ctrl, assuan_context_t ctx, const char *keygrip,
|
||||||
|
int opt_data, int cap)
|
||||||
{
|
{
|
||||||
gpg_error_t err = 0;
|
gpg_error_t err = 0;
|
||||||
struct cryptoki *ck = ck_instance;
|
struct cryptoki *ck = ck_instance;
|
||||||
struct key *k;
|
struct key *k;
|
||||||
const char *usage;
|
const char *usage;
|
||||||
|
|
||||||
|
if (!ck->handle)
|
||||||
|
{
|
||||||
|
err = tkd_init (ctrl, ctx, 0);
|
||||||
|
if (err)
|
||||||
|
return err;
|
||||||
|
}
|
||||||
|
|
||||||
if (keygrip)
|
if (keygrip)
|
||||||
{
|
{
|
||||||
unsigned long r;
|
unsigned long r;
|
||||||
|
@ -110,16 +110,16 @@ void send_keyinfo (ctrl_t ctrl, int data, const char *keygrip_str,
|
|||||||
const char *usage);
|
const char *usage);
|
||||||
|
|
||||||
/*-- pkcs11.c --*/
|
/*-- pkcs11.c --*/
|
||||||
gpg_error_t token_init (ctrl_t ctrl, assuan_context_t ctx, int rescan);
|
gpg_error_t tkd_init (ctrl_t ctrl, assuan_context_t ctx, int rescan);
|
||||||
gpg_error_t token_fini (ctrl_t ctrl, assuan_context_t ctx);
|
gpg_error_t tkd_fini (ctrl_t ctrl, assuan_context_t ctx);
|
||||||
|
|
||||||
gpg_error_t token_sign (ctrl_t ctrl, assuan_context_t ctx,
|
gpg_error_t tkd_sign (ctrl_t ctrl, assuan_context_t ctx,
|
||||||
const char *keygrip, int hash_algo,
|
const char *keygrip, int hash_algo,
|
||||||
unsigned char **r_outdata,
|
unsigned char **r_outdata,
|
||||||
size_t *r_outdatalen);
|
size_t *r_outdatalen);
|
||||||
gpg_error_t token_readkey (ctrl_t ctrl, assuan_context_t ctx,
|
gpg_error_t tkd_readkey (ctrl_t ctrl, assuan_context_t ctx,
|
||||||
const char *keygrip);
|
const char *keygrip);
|
||||||
gpg_error_t token_keyinfo (ctrl_t ctrl,
|
gpg_error_t tkd_keyinfo (ctrl_t ctrl, assuan_context_t ctx,
|
||||||
const char *keygrip, int opt_data, int cap);
|
const char *keygrip, int opt_data, int cap);
|
||||||
|
|
||||||
|
|
||||||
|
Loading…
x
Reference in New Issue
Block a user