Statistics
| Branch: | Revision:

root / libcacard / vcard_emul_nss.c @ 4d5b97da

History | View | Annotate | Download (38 kB)

1
/*
2
 * This is the actual card emulator.
3
 *
4
 * These functions can be implemented in different ways on different platforms
5
 * using the underlying system primitives. For Linux it uses NSS, though direct
6
 * to PKCS #11, openssl+pkcs11, or even gnu crypto libraries+pkcs #11 could be
7
 * used. On Windows CAPI could be used.
8
 *
9
 * This work is licensed under the terms of the GNU LGPL, version 2.1 or later.
10
 * See the COPYING.LIB file in the top-level directory.
11
 */
12

    
13
/*
14
 * NSS headers
15
 */
16

    
17
/* avoid including prototypes.h that redefines uint32 */
18
#define NO_NSPR_10_SUPPORT
19

    
20
#include <nss.h>
21
#include <pk11pub.h>
22
#include <cert.h>
23
#include <key.h>
24
#include <secmod.h>
25
#include <prthread.h>
26
#include <secerr.h>
27

    
28
#include "qemu-common.h"
29

    
30
#include "vcard.h"
31
#include "card_7816t.h"
32
#include "vcard_emul.h"
33
#include "vreader.h"
34
#include "vevent.h"
35

    
36
typedef enum {
37
    VCardEmulUnknown = -1,
38
    VCardEmulFalse = 0,
39
    VCardEmulTrue = 1
40
} VCardEmulTriState;
41

    
42
struct VCardKeyStruct {
43
    CERTCertificate *cert;
44
    PK11SlotInfo *slot;
45
    SECKEYPrivateKey *key;
46
    VCardEmulTriState failedX509;
47
};
48

    
49

    
50
typedef struct VirtualReaderOptionsStruct VirtualReaderOptions;
51

    
52
struct VReaderEmulStruct {
53
    PK11SlotInfo *slot;
54
    VCardEmulType default_type;
55
    char *type_params;
56
    PRBool present;
57
    int     series;
58
    VCard *saved_vcard;
59
};
60

    
61
/*
62
 *  NSS Specific options
63
 */
64
struct VirtualReaderOptionsStruct {
65
    char *name;
66
    char *vname;
67
    VCardEmulType card_type;
68
    char *type_params;
69
    char **cert_name;
70
    int cert_count;
71
};
72

    
73
struct VCardEmulOptionsStruct {
74
    void *nss_db;
75
    VirtualReaderOptions *vreader;
76
    int vreader_count;
77
    VCardEmulType hw_card_type;
78
    const char *hw_type_params;
79
    PRBool use_hw;
80
};
81

    
82
static int nss_emul_init;
83

    
84
/* if we have more that just the slot, define
85
 * VCardEmulStruct here */
86

    
87
/*
88
 * allocate the set of arrays for certs, cert_len, key
89
 */
90
static PRBool
91
vcard_emul_alloc_arrays(unsigned char ***certsp, int **cert_lenp,
92
                        VCardKey ***keysp, int cert_count)
93
{
94
    *certsp = NULL;
95
    *cert_lenp = NULL;
96
    *keysp = NULL;
97
    *certsp = (unsigned char **)g_malloc(sizeof(unsigned char *)*cert_count);
98
    *cert_lenp = (int *)g_malloc(sizeof(int)*cert_count);
99
    *keysp = (VCardKey **)g_malloc(sizeof(VCardKey *)*cert_count);
100
    return PR_TRUE;
101
}
102

    
103
/*
104
 * Emulator specific card information
105
 */
106
typedef struct CardEmulCardStruct CardEmulPrivate;
107

    
108
static VCardEmul *
109
vcard_emul_new_card(PK11SlotInfo *slot)
110
{
111
    PK11_ReferenceSlot(slot);
112
    /* currently we don't need anything other than the slot */
113
    return (VCardEmul *)slot;
114
}
115

    
116
static void
117
vcard_emul_delete_card(VCardEmul *vcard_emul)
118
{
119
    PK11SlotInfo *slot = (PK11SlotInfo *)vcard_emul;
120
    if (slot == NULL) {
121
        return;
122
    }
123
    PK11_FreeSlot(slot);
124
}
125

    
126
static PK11SlotInfo *
127
vcard_emul_card_get_slot(VCard *card)
128
{
129
    /* note, the card is holding the reference, no need to get another one */
130
    return (PK11SlotInfo *)vcard_get_private(card);
131
}
132

    
133

    
134
/*
135
 * key functions
136
 */
137
/* private constructure */
138
static VCardKey *
139
vcard_emul_make_key(PK11SlotInfo *slot, CERTCertificate *cert)
140
{
141
    VCardKey *key;
142

    
143
    key = (VCardKey *)g_malloc(sizeof(VCardKey));
144
    key->slot = PK11_ReferenceSlot(slot);
145
    key->cert = CERT_DupCertificate(cert);
146
    /* NOTE: if we aren't logged into the token, this could return NULL */
147
    /* NOTE: the cert is a temp cert, not necessarily the cert in the token,
148
     * use the DER version of this function */
149
    key->key = PK11_FindKeyByDERCert(slot, cert, NULL);
150
    key->failedX509 = VCardEmulUnknown;
151
    return key;
152
}
153

    
154
/* destructor */
155
void
156
vcard_emul_delete_key(VCardKey *key)
157
{
158
    if (!nss_emul_init || (key == NULL)) {
159
        return;
160
    }
161
    if (key->key) {
162
        SECKEY_DestroyPrivateKey(key->key);
163
        key->key = NULL;
164
    }
165
    if (key->cert) {
166
        CERT_DestroyCertificate(key->cert);
167
    }
168
    if (key->slot) {
169
        PK11_FreeSlot(key->slot);
170
    }
171
}
172

    
173
/*
174
 * grab the nss key from a VCardKey. If it doesn't exist, try to look it up
175
 */
176
static SECKEYPrivateKey *
177
vcard_emul_get_nss_key(VCardKey *key)
178
{
179
    if (key->key) {
180
        return key->key;
181
    }
182
    /* NOTE: if we aren't logged into the token, this could return NULL */
183
    key->key = PK11_FindPrivateKeyFromCert(key->slot, key->cert, NULL);
184
    return key->key;
185
}
186

    
187
/*
188
 * Map NSS errors to 7816 errors
189
 */
190
static vcard_7816_status_t
191
vcard_emul_map_error(int error)
192
{
193
    switch (error) {
194
    case SEC_ERROR_TOKEN_NOT_LOGGED_IN:
195
        return VCARD7816_STATUS_ERROR_CONDITION_NOT_SATISFIED;
196
    case SEC_ERROR_BAD_DATA:
197
    case SEC_ERROR_OUTPUT_LEN:
198
    case SEC_ERROR_INPUT_LEN:
199
    case SEC_ERROR_INVALID_ARGS:
200
    case SEC_ERROR_INVALID_ALGORITHM:
201
    case SEC_ERROR_NO_KEY:
202
    case SEC_ERROR_INVALID_KEY:
203
    case SEC_ERROR_DECRYPTION_DISALLOWED:
204
        return VCARD7816_STATUS_ERROR_DATA_INVALID;
205
    case SEC_ERROR_NO_MEMORY:
206
        return VCARD7816_STATUS_EXC_ERROR_MEMORY_FAILURE;
207
    }
208
    return VCARD7816_STATUS_EXC_ERROR_CHANGE;
209
}
210

    
211
/* RSA sign/decrypt with the key, signature happens 'in place' */
212
vcard_7816_status_t
213
vcard_emul_rsa_op(VCard *card, VCardKey *key,
214
                  unsigned char *buffer, int buffer_size)
215
{
216
    SECKEYPrivateKey *priv_key;
217
    unsigned signature_len;
218
    PK11SlotInfo *slot;
219
    SECStatus rv;
220
    unsigned char buf[2048];
221
    unsigned char *bp = NULL;
222
    int pad_len;
223
    vcard_7816_status_t ret = VCARD7816_STATUS_SUCCESS;
224

    
225
    if ((!nss_emul_init) || (key == NULL)) {
226
        /* couldn't get the key, indicate that we aren't logged in */
227
        return VCARD7816_STATUS_ERROR_CONDITION_NOT_SATISFIED;
228
    }
229
    priv_key = vcard_emul_get_nss_key(key);
230
    if (priv_key == NULL) {
231
        /* couldn't get the key, indicate that we aren't logged in */
232
        return VCARD7816_STATUS_ERROR_CONDITION_NOT_SATISFIED;
233
    }
234
    slot = vcard_emul_card_get_slot(card);
235

    
236
    /*
237
     * this is only true of the rsa signature
238
     */
239
    signature_len = PK11_SignatureLen(priv_key);
240
    if (buffer_size != signature_len) {
241
        return  VCARD7816_STATUS_ERROR_DATA_INVALID;
242
    }
243
    /* be able to handle larger keys if necessariy */
244
    bp = &buf[0];
245
    if (sizeof(buf) < signature_len) {
246
        bp = g_malloc(signature_len);
247
    }
248

    
249
    /*
250
     * do the raw operations. Some tokens claim to do CKM_RSA_X_509, but then
251
     * choke when they try to do the actual operations. Try to detect
252
     * those cases and treat them as if the token didn't claim support for
253
     * X_509.
254
     */
255
    if (key->failedX509 != VCardEmulTrue
256
                              && PK11_DoesMechanism(slot, CKM_RSA_X_509)) {
257
        rv = PK11_PrivDecryptRaw(priv_key, bp, &signature_len, signature_len,
258
                                 buffer, buffer_size);
259
        if (rv == SECSuccess) {
260
            assert(buffer_size == signature_len);
261
            memcpy(buffer, bp, signature_len);
262
            key->failedX509 = VCardEmulFalse;
263
            goto cleanup;
264
        }
265
        /*
266
         * we've had a successful X509 operation, this failure must be
267
         * somethine else
268
         */
269
        if (key->failedX509 == VCardEmulFalse) {
270
            ret = vcard_emul_map_error(PORT_GetError());
271
            goto cleanup;
272
        }
273
        /*
274
         * key->failedX509 must be Unknown at this point, try the
275
         * non-x_509 case
276
         */
277
    }
278
    /* token does not support CKM_RSA_X509, emulate that with CKM_RSA_PKCS */
279
    /* is this a PKCS #1 formatted signature? */
280
    if ((buffer[0] == 0) && (buffer[1] == 1)) {
281
        int i;
282

    
283
        for (i = 2; i < buffer_size; i++) {
284
            /* rsa signature pad */
285
            if (buffer[i] != 0xff) {
286
                break;
287
            }
288
        }
289
        if ((i < buffer_size) && (buffer[i] == 0)) {
290
            /* yes, we have a properly formated PKCS #1 signature */
291
            /*
292
             * NOTE: even if we accidentally got an encrypt buffer, which
293
             * through shear luck started with 00, 01, ff, 00, it won't matter
294
             * because the resulting Sign operation will effectively decrypt
295
             * the real buffer.
296
             */
297
            SECItem signature;
298
            SECItem hash;
299

    
300
            i++;
301
            hash.data = &buffer[i];
302
            hash.len = buffer_size - i;
303
            signature.data = bp;
304
            signature.len = signature_len;
305
            rv = PK11_Sign(priv_key,  &signature, &hash);
306
            if (rv != SECSuccess) {
307
                ret = vcard_emul_map_error(PORT_GetError());
308
                goto cleanup;
309
            }
310
            assert(buffer_size == signature.len);
311
            memcpy(buffer, bp, signature.len);
312
            /*
313
             * we got here because either the X509 attempt failed, or the
314
             * token couldn't do the X509 operation, in either case stay
315
             * with the PKCS version for future operations on this key
316
             */
317
            key->failedX509 = VCardEmulTrue;
318
            goto cleanup;
319
        }
320
    }
321
    pad_len = buffer_size - signature_len;
322
    assert(pad_len < 4);
323
    /*
324
     * OK now we've decrypted the payload, package it up in PKCS #1 for the
325
     * upper layer.
326
     */
327
    buffer[0] = 0;
328
    buffer[1] = 2; /* RSA_encrypt  */
329
    pad_len -= 3; /* format is 0 || 2 || pad || 0 || data */
330
    /*
331
     * padding for PKCS #1 encrypted data is a string of random bytes. The
332
     * random butes protect against potential decryption attacks against RSA.
333
     * Since PrivDecrypt has already stripped those bytes, we can't reconstruct
334
     * them. This shouldn't matter to the upper level code which should just
335
     * strip this code out anyway, so We'll pad with a constant 3.
336
     */
337
    memset(&buffer[2], 0x03, pad_len);
338
    pad_len += 2; /* index to the end of the pad */
339
    buffer[pad_len] = 0;
340
    pad_len++; /* index to the start of the data */
341
    memcpy(&buffer[pad_len], bp, signature_len);
342
    /*
343
     * we got here because either the X509 attempt failed, or the
344
     * token couldn't do the X509 operation, in either case stay
345
     * with the PKCS version for future operations on this key
346
     */
347
    key->failedX509 = VCardEmulTrue;
348
cleanup:
349
    if (bp != buf) {
350
        g_free(bp);
351
    }
352
    return ret;
353
}
354

    
355
/*
356
 * Login functions
357
 */
358
/* return the number of login attempts still possible on the card. if unknown,
359
 * return -1 */
360
int
361
vcard_emul_get_login_count(VCard *card)
362
{
363
    return -1;
364
}
365

    
366
/* login into the card, return the 7816 status word (sw2 || sw1) */
367
vcard_7816_status_t
368
vcard_emul_login(VCard *card, unsigned char *pin, int pin_len)
369
{
370
    PK11SlotInfo *slot;
371
    unsigned char *pin_string = NULL;
372
    int i;
373
    SECStatus rv;
374

    
375
    if (!nss_emul_init) {
376
        return VCARD7816_STATUS_ERROR_CONDITION_NOT_SATISFIED;
377
    }
378
    slot = vcard_emul_card_get_slot(card);
379
     /* We depend on the PKCS #11 module internal login state here because we
380
      * create a separate process to handle each guest instance. If we needed
381
      * to handle multiple guests from one process, then we would need to keep
382
      * a lot of extra state in our card structure
383
      * */
384
    pin_string = g_malloc(pin_len+1);
385
    memcpy(pin_string, pin, pin_len);
386
    pin_string[pin_len] = 0;
387

    
388
    /* handle CAC expanded pins correctly */
389
    for (i = pin_len-1; i >= 0 && (pin_string[i] == 0xff); i--) {
390
        pin_string[i] = 0;
391
    }
392

    
393
    rv = PK11_Authenticate(slot, PR_FALSE, pin_string);
394
    memset(pin_string, 0, pin_len);  /* don't let the pin hang around in memory
395
                                        to be snooped */
396
    g_free(pin_string);
397
    if (rv == SECSuccess) {
398
        return VCARD7816_STATUS_SUCCESS;
399
    }
400
    /* map the error from port get error */
401
    return VCARD7816_STATUS_ERROR_CONDITION_NOT_SATISFIED;
402
}
403

    
404
void
405
vcard_emul_reset(VCard *card, VCardPower power)
406
{
407
    PK11SlotInfo *slot;
408

    
409
    if (!nss_emul_init) {
410
        return;
411
    }
412

    
413
    /*
414
     * if we reset the card (either power on or power off), we lose our login
415
     * state
416
     */
417
    /* TODO: we may also need to send insertion/removal events? */
418
    slot = vcard_emul_card_get_slot(card);
419
    PK11_Logout(slot); /* NOTE: ignoring SECStatus return value */
420
}
421

    
422

    
423
static VReader *
424
vcard_emul_find_vreader_from_slot(PK11SlotInfo *slot)
425
{
426
    VReaderList *reader_list = vreader_get_reader_list();
427
    VReaderListEntry *current_entry = NULL;
428

    
429
    if (reader_list == NULL) {
430
        return NULL;
431
    }
432
    for (current_entry = vreader_list_get_first(reader_list); current_entry;
433
                        current_entry = vreader_list_get_next(current_entry)) {
434
        VReader *reader = vreader_list_get_reader(current_entry);
435
        VReaderEmul *reader_emul = vreader_get_private(reader);
436
        if (reader_emul->slot == slot) {
437
            return reader;
438
        }
439
        vreader_free(reader);
440
    }
441

    
442
    return NULL;
443
}
444

    
445
/*
446
 * create a new reader emul
447
 */
448
static VReaderEmul *
449
vreader_emul_new(PK11SlotInfo *slot, VCardEmulType type, const char *params)
450
{
451
    VReaderEmul *new_reader_emul;
452

    
453
    new_reader_emul = (VReaderEmul *)g_malloc(sizeof(VReaderEmul));
454

    
455
    new_reader_emul->slot = PK11_ReferenceSlot(slot);
456
    new_reader_emul->default_type = type;
457
    new_reader_emul->type_params = strdup(params);
458
    new_reader_emul->present = PR_FALSE;
459
    new_reader_emul->series = 0;
460
    new_reader_emul->saved_vcard = NULL;
461
    return new_reader_emul;
462
}
463

    
464
static void
465
vreader_emul_delete(VReaderEmul *vreader_emul)
466
{
467
    if (vreader_emul == NULL) {
468
        return;
469
    }
470
    if (vreader_emul->slot) {
471
        PK11_FreeSlot(vreader_emul->slot);
472
    }
473
    if (vreader_emul->type_params) {
474
        g_free(vreader_emul->type_params);
475
    }
476
    g_free(vreader_emul);
477
}
478

    
479
/*
480
 *  TODO: move this to emulater non-specific file
481
 */
482
static VCardEmulType
483
vcard_emul_get_type(VReader *vreader)
484
{
485
    VReaderEmul *vreader_emul;
486

    
487
    vreader_emul = vreader_get_private(vreader);
488
    if (vreader_emul && vreader_emul->default_type != VCARD_EMUL_NONE) {
489
        return vreader_emul->default_type;
490
    }
491

    
492
    return vcard_emul_type_select(vreader);
493
}
494
/*
495
 *  TODO: move this to emulater non-specific file
496
 */
497
static const char *
498
vcard_emul_get_type_params(VReader *vreader)
499
{
500
    VReaderEmul *vreader_emul;
501

    
502
    vreader_emul = vreader_get_private(vreader);
503
    if (vreader_emul && vreader_emul->type_params) {
504
        return vreader_emul->type_params;
505
    }
506

    
507
    return "";
508
}
509

    
510
/* pull the slot out of the reader private data */
511
static PK11SlotInfo *
512
vcard_emul_reader_get_slot(VReader *vreader)
513
{
514
    VReaderEmul *vreader_emul = vreader_get_private(vreader);
515
    if (vreader_emul == NULL) {
516
        return NULL;
517
    }
518
    return vreader_emul->slot;
519
}
520

    
521
/*
522
 *  Card ATR's map to physical cards. VCARD_ATR_PREFIX will set appropriate
523
 *  historical bytes for any software emulated card. The remaining bytes can be
524
 *  used to indicate the actual emulator
525
 */
526
static const unsigned char nss_atr[] = { VCARD_ATR_PREFIX(3), 'N', 'S', 'S' };
527

    
528
void
529
vcard_emul_get_atr(VCard *card, unsigned char *atr, int *atr_len)
530
{
531
    int len = MIN(sizeof(nss_atr), *atr_len);
532
    assert(atr != NULL);
533

    
534
    memcpy(atr, nss_atr, len);
535
    *atr_len = len;
536
}
537

    
538
/*
539
 * create a new card from certs and keys
540
 */
541
static VCard *
542
vcard_emul_make_card(VReader *reader,
543
                     unsigned char * const *certs, int *cert_len,
544
                     VCardKey *keys[], int cert_count)
545
{
546
    VCardEmul *vcard_emul;
547
    VCard *vcard;
548
    PK11SlotInfo *slot;
549
    VCardEmulType type;
550
    const char *params;
551

    
552
    type = vcard_emul_get_type(reader);
553

    
554
    /* ignore the inserted card */
555
    if (type == VCARD_EMUL_NONE) {
556
        return NULL;
557
    }
558
    slot = vcard_emul_reader_get_slot(reader);
559
    if (slot == NULL) {
560
        return NULL;
561
    }
562

    
563
    params = vcard_emul_get_type_params(reader);
564
    /* params these can be NULL */
565

    
566
    vcard_emul = vcard_emul_new_card(slot);
567
    if (vcard_emul == NULL) {
568
        return NULL;
569
    }
570
    vcard = vcard_new(vcard_emul, vcard_emul_delete_card);
571
    if (vcard == NULL) {
572
        vcard_emul_delete_card(vcard_emul);
573
        return NULL;
574
    }
575
    vcard_init(reader, vcard, type, params, certs, cert_len, keys, cert_count);
576
    return vcard;
577
}
578

    
579

    
580
/*
581
 * 'clone' a physical card as a virtual card
582
 */
583
static VCard *
584
vcard_emul_mirror_card(VReader *vreader)
585
{
586
    /*
587
     * lookup certs using the C_FindObjects. The Stan Cert handle won't give
588
     * us the real certs until we log in.
589
     */
590
    PK11GenericObject *firstObj, *thisObj;
591
    int cert_count;
592
    unsigned char **certs;
593
    int *cert_len;
594
    VCardKey **keys;
595
    PK11SlotInfo *slot;
596
    PRBool ret;
597
    VCard *card;
598

    
599
    slot = vcard_emul_reader_get_slot(vreader);
600
    if (slot == NULL) {
601
        return NULL;
602
    }
603

    
604
    firstObj = PK11_FindGenericObjects(slot, CKO_CERTIFICATE);
605
    if (firstObj == NULL) {
606
        return NULL;
607
    }
608

    
609
    /* count the certs */
610
    cert_count = 0;
611
    for (thisObj = firstObj; thisObj;
612
                             thisObj = PK11_GetNextGenericObject(thisObj)) {
613
        cert_count++;
614
    }
615

    
616
    if (cert_count == 0) {
617
        PK11_DestroyGenericObjects(firstObj);
618
        return NULL;
619
    }
620

    
621
    /* allocate the arrays */
622
    ret = vcard_emul_alloc_arrays(&certs, &cert_len, &keys, cert_count);
623
    if (ret == PR_FALSE) {
624
        return NULL;
625
    }
626

    
627
    /* fill in the arrays */
628
    cert_count = 0;
629
    for (thisObj = firstObj; thisObj;
630
                             thisObj = PK11_GetNextGenericObject(thisObj)) {
631
        SECItem derCert;
632
        CERTCertificate *cert;
633
        SECStatus rv;
634

    
635
        rv = PK11_ReadRawAttribute(PK11_TypeGeneric, thisObj,
636
                                   CKA_VALUE, &derCert);
637
        if (rv != SECSuccess) {
638
            continue;
639
        }
640
        /* create floating temp cert. This gives us a cert structure even if
641
         * the token isn't logged in */
642
        cert = CERT_NewTempCertificate(CERT_GetDefaultCertDB(), &derCert,
643
                                       NULL, PR_FALSE, PR_TRUE);
644
        SECITEM_FreeItem(&derCert, PR_FALSE);
645
        if (cert == NULL) {
646
            continue;
647
        }
648

    
649
        certs[cert_count] = cert->derCert.data;
650
        cert_len[cert_count] = cert->derCert.len;
651
        keys[cert_count] = vcard_emul_make_key(slot, cert);
652
        cert_count++;
653
        CERT_DestroyCertificate(cert); /* key obj still has a reference */
654
    }
655

    
656
    /* now create the card */
657
    card = vcard_emul_make_card(vreader, certs, cert_len, keys, cert_count);
658
    g_free(certs);
659
    g_free(cert_len);
660
    g_free(keys);
661

    
662
    return card;
663
}
664

    
665
static VCardEmulType default_card_type = VCARD_EMUL_NONE;
666
static const char *default_type_params = "";
667

    
668
/*
669
 * This thread looks for card and reader insertions and puts events on the
670
 * event queue
671
 */
672
static void
673
vcard_emul_event_thread(void *arg)
674
{
675
    PK11SlotInfo *slot;
676
    VReader *vreader;
677
    VReaderEmul *vreader_emul;
678
    VCard *vcard;
679
    SECMODModule *module = (SECMODModule *)arg;
680

    
681
    do {
682
        /*
683
         * XXX - the latency value doesn't matter one bit. you only get no
684
         * blocking (flags |= CKF_DONT_BLOCK) or PKCS11_WAIT_LATENCY (==500),
685
         * hard coded in coolkey.  And it isn't coolkey's fault - the timeout
686
         * value we pass get's dropped on the floor before C_WaitForSlotEvent
687
         * is called.
688
         */
689
        slot = SECMOD_WaitForAnyTokenEvent(module, 0, 500);
690
        if (slot == NULL) {
691
            /* this could be just a no event indication */
692
            if (PORT_GetError() == SEC_ERROR_NO_EVENT) {
693
                continue;
694
            }
695
            break;
696
        }
697
        vreader = vcard_emul_find_vreader_from_slot(slot);
698
        if (vreader == NULL) {
699
            /* new vreader */
700
            vreader_emul = vreader_emul_new(slot, default_card_type,
701
                                            default_type_params);
702
            vreader = vreader_new(PK11_GetSlotName(slot), vreader_emul,
703
                                  vreader_emul_delete);
704
            PK11_FreeSlot(slot);
705
            slot = NULL;
706
            vreader_add_reader(vreader);
707
            vreader_free(vreader);
708
            continue;
709
        }
710
        /* card remove/insert */
711
        vreader_emul = vreader_get_private(vreader);
712
        if (PK11_IsPresent(slot)) {
713
            int series = PK11_GetSlotSeries(slot);
714
            if (series != vreader_emul->series) {
715
                if (vreader_emul->present) {
716
                    vreader_insert_card(vreader, NULL);
717
                }
718
                vcard = vcard_emul_mirror_card(vreader);
719
                vreader_insert_card(vreader, vcard);
720
                vcard_free(vcard);
721
            }
722
            vreader_emul->series = series;
723
            vreader_emul->present = 1;
724
            vreader_free(vreader);
725
            PK11_FreeSlot(slot);
726
            continue;
727
        }
728
        if (vreader_emul->present) {
729
            vreader_insert_card(vreader, NULL);
730
        }
731
        vreader_emul->series = 0;
732
        vreader_emul->present = 0;
733
        PK11_FreeSlot(slot);
734
        vreader_free(vreader);
735
    } while (1);
736
}
737

    
738
/* if the card is inserted when we start up, make sure our state is correct */
739
static void
740
vcard_emul_init_series(VReader *vreader, VCard *vcard)
741
{
742
    VReaderEmul *vreader_emul = vreader_get_private(vreader);
743
    PK11SlotInfo *slot = vreader_emul->slot;
744

    
745
    vreader_emul->present = PK11_IsPresent(slot);
746
    vreader_emul->series = PK11_GetSlotSeries(slot);
747
    if (vreader_emul->present == 0) {
748
        vreader_insert_card(vreader, NULL);
749
    }
750
}
751

    
752
/*
753
 * each module has a separate wait call, create a thread for each module that
754
 * we are using.
755
 */
756
static void
757
vcard_emul_new_event_thread(SECMODModule *module)
758
{
759
    PR_CreateThread(PR_SYSTEM_THREAD, vcard_emul_event_thread,
760
                     module, PR_PRIORITY_HIGH, PR_GLOBAL_THREAD,
761
                     PR_UNJOINABLE_THREAD, 0);
762
}
763

    
764
static const VCardEmulOptions default_options = {
765
    .nss_db = NULL,
766
    .vreader = NULL,
767
    .vreader_count = 0,
768
    .hw_card_type = VCARD_EMUL_CAC,
769
    .hw_type_params = "",
770
    .use_hw = PR_TRUE
771
};
772

    
773

    
774
/*
775
 *  NSS needs the app to supply a password prompt. In our case the only time
776
 *  the password is supplied is as part of the Login APDU. The actual password
777
 *  is passed in the pw_arg in that case. In all other cases pw_arg should be
778
 *  NULL.
779
 */
780
static char *
781
vcard_emul_get_password(PK11SlotInfo *slot, PRBool retries, void *pw_arg)
782
{
783
    /* if it didn't work the first time, don't keep trying */
784
    if (retries) {
785
        return NULL;
786
    }
787
    /* we are looking up a password when we don't have one in hand */
788
    if (pw_arg == NULL) {
789
        return NULL;
790
    }
791
    /* TODO: we really should verify that were are using the right slot */
792
    return PORT_Strdup(pw_arg);
793
}
794

    
795
/* Force a card removal even if the card is not physically removed */
796
VCardEmulError
797
vcard_emul_force_card_remove(VReader *vreader)
798
{
799
    if (!nss_emul_init || (vreader_card_is_present(vreader) != VREADER_OK)) {
800
        return VCARD_EMUL_FAIL; /* card is already removed */
801
    }
802

    
803
    /* OK, remove it */
804
    vreader_insert_card(vreader, NULL);
805
    return VCARD_EMUL_OK;
806
}
807

    
808
/* Re-insert of a card that has been removed by force removal */
809
VCardEmulError
810
vcard_emul_force_card_insert(VReader *vreader)
811
{
812
    VReaderEmul *vreader_emul;
813
    VCard *vcard;
814

    
815
    if (!nss_emul_init || (vreader_card_is_present(vreader) == VREADER_OK)) {
816
        return VCARD_EMUL_FAIL; /* card is already removed */
817
    }
818
    vreader_emul = vreader_get_private(vreader);
819

    
820
    /* if it's a softcard, get the saved vcard from the reader emul structure */
821
    if (vreader_emul->saved_vcard) {
822
        vcard = vcard_reference(vreader_emul->saved_vcard);
823
    } else {
824
        /* it must be a physical card, rebuild it */
825
        if (!PK11_IsPresent(vreader_emul->slot)) {
826
            /* physical card has been removed, not way to reinsert it */
827
            return VCARD_EMUL_FAIL;
828
        }
829
        vcard = vcard_emul_mirror_card(vreader);
830
    }
831
    vreader_insert_card(vreader, vcard);
832
    vcard_free(vcard);
833

    
834
    return VCARD_EMUL_OK;
835
}
836

    
837

    
838
static PRBool
839
module_has_removable_hw_slots(SECMODModule *mod)
840
{
841
    int i;
842
    PRBool ret = PR_FALSE;
843
    SECMODListLock *moduleLock = SECMOD_GetDefaultModuleListLock();
844

    
845
    if (!moduleLock) {
846
        PORT_SetError(SEC_ERROR_NOT_INITIALIZED);
847
        return ret;
848
    }
849
    SECMOD_GetReadLock(moduleLock);
850
    for (i = 0; i < mod->slotCount; i++) {
851
        PK11SlotInfo *slot = mod->slots[i];
852
        if (PK11_IsRemovable(slot) && PK11_IsHW(slot)) {
853
            ret = PR_TRUE;
854
            break;
855
        }
856
    }
857
    SECMOD_ReleaseReadLock(moduleLock);
858
    return ret;
859
}
860

    
861
/* Previously we returned FAIL if no readers found. This makes
862
 * no sense when using hardware, since there may be no readers connected
863
 * at the time vcard_emul_init is called, but they will be properly
864
 * recognized later. So Instead return FAIL only if no_hw==1 and no
865
 * vcards can be created (indicates error with certificates provided
866
 * or db), or if any other higher level error (NSS error, missing coolkey). */
867
static int vcard_emul_init_called;
868

    
869
VCardEmulError
870
vcard_emul_init(const VCardEmulOptions *options)
871
{
872
    SECStatus rv;
873
    PRBool ret, has_readers = PR_FALSE, need_coolkey_module;
874
    VReader *vreader;
875
    VReaderEmul *vreader_emul;
876
    SECMODListLock *module_lock;
877
    SECMODModuleList *module_list;
878
    SECMODModuleList *mlp;
879
    int i;
880

    
881
    if (vcard_emul_init_called) {
882
        return VCARD_EMUL_INIT_ALREADY_INITED;
883
    }
884
    vcard_emul_init_called = 1;
885
    vreader_init();
886
    vevent_queue_init();
887

    
888
    if (options == NULL) {
889
        options = &default_options;
890
    }
891

    
892
    /* first initialize NSS */
893
    if (options->nss_db) {
894
        rv = NSS_Init(options->nss_db);
895
    } else {
896
        rv = NSS_Init("sql:/etc/pki/nssdb");
897
    }
898
    if (rv != SECSuccess) {
899
        return VCARD_EMUL_FAIL;
900
    }
901
    /* Set password callback function */
902
    PK11_SetPasswordFunc(vcard_emul_get_password);
903

    
904
    /* set up soft cards emulated by software certs rather than physical cards
905
     * */
906
    for (i = 0; i < options->vreader_count; i++) {
907
        int j;
908
        int cert_count;
909
        unsigned char **certs;
910
        int *cert_len;
911
        VCardKey **keys;
912
        PK11SlotInfo *slot;
913

    
914
        slot = PK11_FindSlotByName(options->vreader[i].name);
915
        if (slot == NULL) {
916
            continue;
917
        }
918
        vreader_emul = vreader_emul_new(slot, options->vreader[i].card_type,
919
                                        options->vreader[i].type_params);
920
        vreader = vreader_new(options->vreader[i].vname, vreader_emul,
921
                              vreader_emul_delete);
922
        vreader_add_reader(vreader);
923
        cert_count = options->vreader[i].cert_count;
924

    
925
        ret = vcard_emul_alloc_arrays(&certs, &cert_len, &keys,
926
                                      options->vreader[i].cert_count);
927
        if (ret == PR_FALSE) {
928
            continue;
929
        }
930
        cert_count = 0;
931
        for (j = 0; j < options->vreader[i].cert_count; j++) {
932
            /* we should have a better way of identifying certs than by
933
             * nickname here */
934
            CERTCertificate *cert = PK11_FindCertFromNickname(
935
                                        options->vreader[i].cert_name[j],
936
                                        NULL);
937
            if (cert == NULL) {
938
                continue;
939
            }
940
            certs[cert_count] = cert->derCert.data;
941
            cert_len[cert_count] = cert->derCert.len;
942
            keys[cert_count] = vcard_emul_make_key(slot, cert);
943
            /* this is safe because the key is still holding a cert reference */
944
            CERT_DestroyCertificate(cert);
945
            cert_count++;
946
        }
947
        if (cert_count) {
948
            VCard *vcard = vcard_emul_make_card(vreader, certs, cert_len,
949
                                                keys, cert_count);
950
            vreader_insert_card(vreader, vcard);
951
            vcard_emul_init_series(vreader, vcard);
952
            /* allow insertion and removal of soft cards */
953
            vreader_emul->saved_vcard = vcard_reference(vcard);
954
            vcard_free(vcard);
955
            vreader_free(vreader);
956
            has_readers = PR_TRUE;
957
        }
958
        g_free(certs);
959
        g_free(cert_len);
960
        g_free(keys);
961
    }
962

    
963
    /* if we aren't suppose to use hw, skip looking up hardware tokens */
964
    if (!options->use_hw) {
965
        nss_emul_init = has_readers;
966
        return has_readers ? VCARD_EMUL_OK : VCARD_EMUL_FAIL;
967
    }
968

    
969
    /* make sure we have some PKCS #11 module loaded */
970
    module_lock = SECMOD_GetDefaultModuleListLock();
971
    module_list = SECMOD_GetDefaultModuleList();
972
    need_coolkey_module = !has_readers;
973
    SECMOD_GetReadLock(module_lock);
974
    for (mlp = module_list; mlp; mlp = mlp->next) {
975
        SECMODModule *module = mlp->module;
976
        if (module_has_removable_hw_slots(module)) {
977
            need_coolkey_module = PR_FALSE;
978
            break;
979
        }
980
    }
981
    SECMOD_ReleaseReadLock(module_lock);
982

    
983
    if (need_coolkey_module) {
984
        SECMODModule *module;
985
        module = SECMOD_LoadUserModule(
986
                    (char *)"library=libcoolkeypk11.so name=Coolkey",
987
                    NULL, PR_FALSE);
988
        if (module == NULL) {
989
            return VCARD_EMUL_FAIL;
990
        }
991
        SECMOD_DestroyModule(module); /* free our reference, Module will still
992
                                       * be on the list.
993
                                       * until we destroy it */
994
    }
995

    
996
    /* now examine all the slots, finding which should be readers */
997
    /* We should control this with options. For now we mirror out any
998
     * removable hardware slot */
999
    default_card_type = options->hw_card_type;
1000
    default_type_params = strdup(options->hw_type_params);
1001

    
1002
    SECMOD_GetReadLock(module_lock);
1003
    for (mlp = module_list; mlp; mlp = mlp->next) {
1004
        SECMODModule *module = mlp->module;
1005

    
1006
        /* Ignore the internal module */
1007
        if (module == NULL || module == SECMOD_GetInternalModule()) {
1008
            continue;
1009
        }
1010

    
1011
        for (i = 0; i < module->slotCount; i++) {
1012
            PK11SlotInfo *slot = module->slots[i];
1013

    
1014
            /* only map removable HW slots */
1015
            if (slot == NULL || !PK11_IsRemovable(slot) || !PK11_IsHW(slot)) {
1016
                continue;
1017
            }
1018
            if (strcmp("E-Gate 0 0", PK11_GetSlotName(slot)) == 0) {
1019
                /*
1020
                 * coolkey <= 1.1.0-20 emulates this reader if it can't find
1021
                 * any hardware readers. This causes problems, warn user of
1022
                 * problems.
1023
                 */
1024
                fprintf(stderr, "known bad coolkey version - see "
1025
                        "https://bugzilla.redhat.com/show_bug.cgi?id=802435\n");
1026
                continue;
1027
            }
1028
            vreader_emul = vreader_emul_new(slot, options->hw_card_type,
1029
                                            options->hw_type_params);
1030
            vreader = vreader_new(PK11_GetSlotName(slot), vreader_emul,
1031
                                  vreader_emul_delete);
1032
            vreader_add_reader(vreader);
1033

    
1034
            if (PK11_IsPresent(slot)) {
1035
                VCard *vcard;
1036
                vcard = vcard_emul_mirror_card(vreader);
1037
                vreader_insert_card(vreader, vcard);
1038
                vcard_emul_init_series(vreader, vcard);
1039
                vcard_free(vcard);
1040
            }
1041
        }
1042
        vcard_emul_new_event_thread(module);
1043
    }
1044
    SECMOD_ReleaseReadLock(module_lock);
1045
    nss_emul_init = PR_TRUE;
1046

    
1047
    return VCARD_EMUL_OK;
1048
}
1049

    
1050
/* Recreate card insert events for all readers (user should
1051
 * deduce implied reader insert. perhaps do a reader insert as well?)
1052
 */
1053
void
1054
vcard_emul_replay_insertion_events(void)
1055
{
1056
    VReaderListEntry *current_entry;
1057
    VReaderListEntry *next_entry = NULL;
1058
    VReaderList *list = vreader_get_reader_list();
1059

    
1060
    for (current_entry = vreader_list_get_first(list); current_entry;
1061
            current_entry = next_entry) {
1062
        VReader *vreader = vreader_list_get_reader(current_entry);
1063
        next_entry = vreader_list_get_next(current_entry);
1064
        vreader_queue_card_event(vreader);
1065
    }
1066
}
1067

    
1068
/*
1069
 *  Silly little functions to help parsing our argument string
1070
 */
1071
static int
1072
count_tokens(const char *str, char token, char token_end)
1073
{
1074
    int count = 0;
1075

    
1076
    for (; *str; str++) {
1077
        if (*str == token) {
1078
            count++;
1079
        }
1080
        if (*str == token_end) {
1081
            break;
1082
        }
1083
    }
1084
    return count;
1085
}
1086

    
1087
static const char *
1088
strip(const char *str)
1089
{
1090
    for (; *str && isspace(*str); str++) {
1091
    }
1092
    return str;
1093
}
1094

    
1095
static const char *
1096
find_blank(const char *str)
1097
{
1098
    for (; *str && !isspace(*str); str++) {
1099
    }
1100
    return str;
1101
}
1102

    
1103

    
1104
/*
1105
 *  We really want to use some existing argument parsing library here. That
1106
 *  would give us a consistent look */
1107
static VCardEmulOptions options;
1108
#define READER_STEP 4
1109

    
1110
/* Expects "args" to be at the beginning of a token (ie right after the ','
1111
 * ending the previous token), and puts the next token start in "token",
1112
 * and its length in "token_length". "token" will not be nul-terminated.
1113
 * After calling the macro, "args" will be advanced to the beginning of
1114
 * the next token.
1115
 * This macro may call continue or break.
1116
 */
1117
#define NEXT_TOKEN(token) \
1118
            (token) = args; \
1119
            args = strpbrk(args, ",)"); \
1120
            if (*args == 0) { \
1121
                break; \
1122
            } \
1123
            if (*args == ')') { \
1124
                args++; \
1125
                continue; \
1126
            } \
1127
            (token##_length) = args - (token); \
1128
            args = strip(args+1);
1129

    
1130
VCardEmulOptions *
1131
vcard_emul_options(const char *args)
1132
{
1133
    int reader_count = 0;
1134
    VCardEmulOptions *opts;
1135

    
1136
    /* Allow the future use of allocating the options structure on the fly */
1137
    memcpy(&options, &default_options, sizeof(options));
1138
    opts = &options;
1139

    
1140
    do {
1141
        args = strip(args); /* strip off the leading spaces */
1142
        if (*args == ',') {
1143
            continue;
1144
        }
1145
        /* soft=(slot_name,virt_name,emul_type,emul_flags,cert_1, (no eol)
1146
         *       cert_2,cert_3...) */
1147
        if (strncmp(args, "soft=", 5) == 0) {
1148
            const char *name;
1149
            size_t name_length;
1150
            const char *vname;
1151
            size_t vname_length;
1152
            const char *type_params;
1153
            size_t type_params_length;
1154
            char type_str[100];
1155
            VCardEmulType type;
1156
            int count, i;
1157
            VirtualReaderOptions *vreaderOpt = NULL;
1158

    
1159
            args = strip(args + 5);
1160
            if (*args != '(') {
1161
                continue;
1162
            }
1163
            args = strip(args+1);
1164

    
1165
            NEXT_TOKEN(name)
1166
            NEXT_TOKEN(vname)
1167
            NEXT_TOKEN(type_params)
1168
            type_params_length = MIN(type_params_length, sizeof(type_str)-1);
1169
            strncpy(type_str, type_params, type_params_length);
1170
            type_str[type_params_length] = 0;
1171
            type = vcard_emul_type_from_string(type_str);
1172

    
1173
            NEXT_TOKEN(type_params)
1174

    
1175
            if (*args == 0) {
1176
                break;
1177
            }
1178

    
1179
            if (opts->vreader_count >= reader_count) {
1180
                reader_count += READER_STEP;
1181
                vreaderOpt = realloc(opts->vreader,
1182
                                reader_count * sizeof(*vreaderOpt));
1183
                if (vreaderOpt == NULL) {
1184
                    return opts; /* we're done */
1185
                }
1186
            }
1187
            opts->vreader = vreaderOpt;
1188
            vreaderOpt = &vreaderOpt[opts->vreader_count];
1189
            vreaderOpt->name = g_strndup(name, name_length);
1190
            vreaderOpt->vname = g_strndup(vname, vname_length);
1191
            vreaderOpt->card_type = type;
1192
            vreaderOpt->type_params =
1193
                g_strndup(type_params, type_params_length);
1194
            count = count_tokens(args, ',', ')') + 1;
1195
            vreaderOpt->cert_count = count;
1196
            vreaderOpt->cert_name = (char **)g_malloc(count*sizeof(char *));
1197
            for (i = 0; i < count; i++) {
1198
                const char *cert = args;
1199
                args = strpbrk(args, ",)");
1200
                vreaderOpt->cert_name[i] = g_strndup(cert, args - cert);
1201
                args = strip(args+1);
1202
            }
1203
            if (*args == ')') {
1204
                args++;
1205
            }
1206
            opts->vreader_count++;
1207
        /* use_hw= */
1208
        } else if (strncmp(args, "use_hw=", 7) == 0) {
1209
            args = strip(args+7);
1210
            if (*args == '0' || *args == 'N' || *args == 'n' || *args == 'F') {
1211
                opts->use_hw = PR_FALSE;
1212
            } else {
1213
                opts->use_hw = PR_TRUE;
1214
            }
1215
            args = find_blank(args);
1216
        /* hw_type= */
1217
        } else if (strncmp(args, "hw_type=", 8) == 0) {
1218
            args = strip(args+8);
1219
            opts->hw_card_type = vcard_emul_type_from_string(args);
1220
            args = find_blank(args);
1221
        /* hw_params= */
1222
        } else if (strncmp(args, "hw_params=", 10) == 0) {
1223
            const char *params;
1224
            args = strip(args+10);
1225
            params = args;
1226
            args = find_blank(args);
1227
            opts->hw_type_params = g_strndup(params, args-params);
1228
        /* db="/data/base/path" */
1229
        } else if (strncmp(args, "db=", 3) == 0) {
1230
            const char *db;
1231
            args = strip(args+3);
1232
            if (*args != '"') {
1233
                continue;
1234
            }
1235
            args++;
1236
            db = args;
1237
            args = strpbrk(args, "\"\n");
1238
            opts->nss_db = g_strndup(db, args-db);
1239
            if (*args != 0) {
1240
                args++;
1241
            }
1242
        } else {
1243
            args = find_blank(args);
1244
        }
1245
    } while (*args != 0);
1246

    
1247
    return opts;
1248
}
1249

    
1250
void
1251
vcard_emul_usage(void)
1252
{
1253
   fprintf(stderr,
1254
"emul args: comma separated list of the following arguments\n"
1255
" db={nss_database}               (default sql:/etc/pki/nssdb)\n"
1256
" use_hw=[yes|no]                 (default yes)\n"
1257
" hw_type={card_type_to_emulate}  (default CAC)\n"
1258
" hw_param={param_for_card}       (default \"\")\n"
1259
" soft=({slot_name},{vreader_name},{card_type_to_emulate},{params_for_card},\n"
1260
"       {cert1},{cert2},{cert3}    (default none)\n"
1261
"\n"
1262
"  {nss_database}          The location of the NSS cert & key database\n"
1263
"  {card_type_to_emulate}  What card interface to present to the guest\n"
1264
"  {param_for_card}        Card interface specific parameters\n"
1265
"  {slot_name}             NSS slot that contains the certs\n"
1266
"  {vreader_name}          Virtual reader name to present to the guest\n"
1267
"  {certN}                 Nickname of the certificate n on the virtual card\n"
1268
"\n"
1269
"These parameters come as a single string separated by blanks or newlines."
1270
"\n"
1271
"Unless use_hw is set to no, all tokens that look like removable hardware\n"
1272
"tokens will be presented to the guest using the emulator specified by\n"
1273
"hw_type, and parameters of hw_param.\n"
1274
"\n"
1275
"If more one or more soft= parameters are specified, these readers will be\n"
1276
"presented to the guest\n");
1277
}