| /* |
| * Hid gadget driver daemon for FIDO2 |
| * |
| * Copyright (C) 2019 James.Bottomley@HansenPartnership.com |
| * |
| * SPDX-License-Identifier: GPL-2.0-only |
| */ |
| |
| |
| #include <fcntl.h> |
| #include <getopt.h> |
| #include <stdio.h> |
| #include <stdlib.h> |
| #include <string.h> |
| #include <unistd.h> |
| |
| #include <sys/types.h> |
| #include <sys/stat.h> |
| |
| #include "u2f.h" |
| #include "u2f_hid.h" |
| #include "hidgd-tpm.h" |
| |
| static int dev; |
| static int certd; |
| |
| static const uint32_t parent = 0x81000101; |
| |
| static struct option long_options[] = { |
| {"help", 0, 0, 'h'}, |
| {"version", 0, 0, 'v'}, |
| {0, 0, 0, 0,} |
| }; |
| |
| static void usage(char *argv0, FILE *f) |
| { |
| fprintf(f, "Usage: %s [options] <hidg device> <certificate file>\n\n" |
| "Options:\n" |
| "\t-h, --help print this help message\n" |
| "\t-v, --version print package version\n" |
| "\n", |
| argv0); |
| } |
| |
| /* |
| * The FIDO protocol has got to be one of the most screwed |
| * up on the planet: All the HID frame messages are little endian |
| * and all the U2F messages are big endian |
| */ |
| static void u2f_setbe(int val, uint8_t *buf) |
| { |
| buf[0] = (val >> 8) & 0xff; |
| buf[1] = val & 0xff; |
| } |
| |
| static int u2f_getbe(uint8_t *buf) |
| { |
| return (buf[0] << 8) + buf[1]; |
| } |
| |
| static void u2fhid_set_len(U2FHID_FRAME *frame, int len) |
| { |
| frame->init.bcntl = len & 0xff; |
| frame->init.bcnth = (len >> 8) & 0xff; |
| } |
| |
| static int get_apdu(uint8_t **ptr) |
| { |
| int len; |
| |
| if (**ptr == 0) { |
| len = u2f_getbe(*ptr + 1); |
| *ptr += 3; |
| } else { |
| len = *(*ptr)++; |
| } |
| |
| return len; |
| } |
| |
| static void process_error(uint32_t cid, int err) |
| { |
| char buf[HID_RPT_SIZE]; |
| U2FHID_FRAME *reply = (U2FHID_FRAME *)buf; |
| int count; |
| |
| memset(buf, 0, sizeof(buf)); |
| reply->cid = cid; |
| reply->init.cmd = U2FHID_ERROR; |
| reply->init.bcnth = 0; |
| reply->init.bcntl = sizeof(reply) + 1; |
| reply->init.data[0] = err; |
| |
| count = write(dev, buf, sizeof(buf)); |
| printf("wrote error frame %d\n", count); |
| } |
| |
| static int get_payload(U2FHID_FRAME *frame, uint8_t buf[HID_MAX_PAYLOAD]) |
| { |
| int len = MSG_LEN(*frame); |
| int count = sizeof(frame->init.data); |
| int seq = 0; |
| |
| memcpy(buf, frame->init.data, count); |
| |
| printf("looking for packet of len %d\n", len); |
| |
| while (count < len) { |
| int c = read(dev, frame, HID_RPT_SIZE); |
| |
| if (c != HID_RPT_SIZE) { |
| fprintf(stderr, "Got short read of sequence packet %d != %d\n", c, HID_RPT_SIZE); |
| process_error(frame->cid, ERR_INVALID_LEN); |
| return -ERR_INVALID_LEN; |
| } |
| |
| if (seq++ != frame->cont.seq) { |
| fprintf(stderr, "Invalid sequence %d != %d\n", seq-1, frame->cont.seq); |
| return -ERR_INVALID_SEQ; |
| } |
| memcpy(&buf[count], frame->cont.data, sizeof(frame->cont.data)); |
| count += sizeof(frame->cont.data); |
| } |
| return len; |
| } |
| |
| static void send_payload(uint8_t ctap[HID_MAX_PAYLOAD], int len, uint32_t cid, |
| int err) |
| { |
| uint8_t buf[HID_RPT_SIZE]; |
| U2FHID_FRAME *frame = (U2FHID_FRAME *)buf; |
| int count; |
| int seq = 0; |
| |
| /* response bytes at end */ |
| u2f_setbe(err, &ctap[len]); |
| /* account for response bytes */ |
| len += 2; |
| |
| frame->cid = cid; |
| frame->init.cmd = U2FHID_MSG; |
| u2fhid_set_len(frame, len); |
| count = sizeof(frame->init.data); |
| memcpy(frame->init.data, ctap, count); |
| write(dev, buf, sizeof(buf)); |
| |
| while (count < len) { |
| frame->cid = cid; |
| frame->cont.seq = seq++; |
| memcpy(frame->cont.data, &ctap[count], sizeof(frame->cont.data)); |
| count += sizeof(frame->cont.data); |
| write(dev, buf, sizeof(buf)); |
| } |
| } |
| |
| static void process_version(uint32_t cid) |
| { |
| uint8_t buf[HID_RPT_SIZE]; |
| U2FHID_FRAME *frame = (U2FHID_FRAME *)buf; |
| static const char *repl = "U2F_V2"; |
| const int len = strlen(repl); |
| |
| memset(buf, 0, sizeof(buf)); |
| frame->cid = cid; |
| frame->init.cmd = U2FHID_MSG; |
| frame->init.bcnth = 0; |
| frame->init.bcntl = len + 2; /* error at end */ |
| memcpy(frame->init.data, repl, len); |
| u2f_setbe(U2F_SW_NO_ERROR, &frame->init.data[len]); |
| write(dev, buf, sizeof(buf)); |
| } |
| |
| const char keystr[] = "This is a key handle"; |
| |
| static void process_register(uint32_t cid, uint8_t ctap[HID_MAX_PAYLOAD]) |
| { |
| uint8_t buf[HID_MAX_PAYLOAD]; |
| U2F_REGISTER_REQ *req; |
| U2F_REGISTER_RESP *resp = (U2F_REGISTER_RESP *)buf; |
| uint8_t *ptr = &ctap[4]; /* point to APDU lengths */ |
| int len; |
| |
| len = get_apdu(&ptr); |
| if (len != sizeof(U2F_REGISTER_REQ)) { |
| fprintf(stderr, "Wrong REGISTER REQ len %d != %ld\n", |
| len, sizeof(U2F_REGISTER_REQ)); |
| process_error(cid, ERR_INVALID_CMD); |
| return; |
| } |
| /* |
| * standard seems to require this but Mozilla doesn't transmit it |
| len = get_apdu(&ptr); |
| if (len < sizeof(U2F_REGISTER_RESP)) { |
| fprintf(stderr, "Wrong REGISTER RESP len %d < %ld\n", |
| len, sizeof(U2F_REGISTER_RESP)); |
| process_error(cid, ERR_INVALID_CMD); |
| return; |
| } |
| */ |
| req = (U2F_REGISTER_REQ *)ptr; |
| printf("chal[0] = %d, appId[0] = %d\n", req->chal[0], req->appId[0]); |
| memset(buf, 0, sizeof(buf)); |
| resp->registerId = U2F_REGISTER_ID; |
| tpm_get_public_point(parent, &resp->pubKey); |
| resp->keyHandleLen = sizeof(keystr); /* include trailing 0 */ |
| strcpy((char *)resp->keyHandleCertSig, keystr); |
| ptr = &resp->keyHandleCertSig[resp->keyHandleLen]; |
| /* place the DER encoded cert into the buffer */ |
| lseek(certd, 0, SEEK_SET); |
| len = read(certd, ptr, sizeof(buf) - (ptr - buf)); |
| if (len < 0) { |
| perror("Failed to load cert into reply"); |
| process_error(cid, ERR_INVALID_CMD); |
| return; |
| } |
| ptr += len; |
| ptr += tpm_fill_register_sig(parent, req, resp, ptr); |
| |
| send_payload(buf, ptr - buf, cid, U2F_SW_NO_ERROR); |
| } |
| |
| static void process_authenticate(uint32_t cid, uint8_t ctap[HID_MAX_PAYLOAD]) |
| { |
| uint8_t buf[HID_MAX_PAYLOAD]; |
| U2F_AUTHENTICATE_REQ *req; |
| U2F_AUTHENTICATE_RESP *resp = (U2F_AUTHENTICATE_RESP *)buf; |
| uint8_t *ptr = &ctap[4]; /* point to APDU lengths */ |
| int len; |
| |
| len = get_apdu(&ptr); |
| if (len > sizeof(U2F_AUTHENTICATE_REQ)) { |
| fprintf(stderr, "Wrong AUTHENTICATE REQ len %d > %ld\n", |
| len, sizeof(U2F_AUTHENTICATE_REQ)); |
| process_error(cid, ERR_INVALID_CMD); |
| return; |
| } |
| /* |
| * standard seems to require this but Mozilla doesn't transmit it |
| len = get_apdu(&ptr); |
| if (len < sizeof(U2F_AUTHENTICATE_RESP)) { |
| fprintf(stderr, "Wrong AUTHENTICATE RESP len %d < %ld\n", |
| len, sizeof(U2F_AUTHENTICATE_RESP)); |
| process_error(cid, ERR_INVALID_CMD); |
| return; |
| } |
| */ |
| req = (U2F_AUTHENTICATE_REQ *)ptr; |
| if (req->keyHandleLen != sizeof(keystr)) { |
| fprintf(stderr, "Wrong keystr len %d != %ld\n", |
| req->keyHandleLen, sizeof(keystr)); |
| process_error(cid, ERR_INVALID_CMD); |
| return; |
| } |
| if (strcmp((char *)req->keyHandle, keystr) != 0) { |
| fprintf(stderr, "Wrong keystr '%s' != '%s'\n", |
| req->keyHandle, keystr); |
| process_error(cid, ERR_INVALID_CMD); |
| return; |
| } |
| memset(buf, 0, sizeof(buf)); |
| resp->flags = U2F_AUTH_FLAG_TUP; /* pretend we have user presence */ |
| send_payload(buf, sizeof(U2F_AUTHENTICATE_RESP), cid, |
| ctap[2] == U2F_AUTH_CHECK_ONLY ? |
| U2F_SW_CONDITIONS_NOT_SATISFIED : U2F_SW_NO_ERROR); |
| } |
| |
| static void process_msg(U2FHID_FRAME *frame) |
| { |
| uint8_t ctap[HID_MAX_PAYLOAD]; |
| int len; |
| int ins; |
| |
| uint32_t cid = frame->cid; |
| |
| len = get_payload(frame, ctap); |
| if (len < 0) { |
| process_error(frame->cid, -len); |
| return; |
| } |
| ins = ctap[1]; |
| printf("Got CLA=0x%x, ins=0x%x\n", ctap[0], ins); |
| |
| if (ins == U2F_VERSION) { |
| printf("U2F VERSION\n"); |
| process_version(cid); |
| } else if (ins == U2F_REGISTER) { |
| printf("U2F REGISTER\n"); |
| process_register(cid, ctap); |
| } else if (ins == U2F_AUTHENTICATE) { |
| printf("U2F_AUTHENTICATE\n"); |
| process_authenticate(cid, ctap); |
| } else { |
| printf("Unrecognized command 0x%x\n", ins); |
| process_error(cid, ERR_INVALID_CMD); |
| } |
| } |
| |
| static void process_init(U2FHID_FRAME *frame) |
| { |
| char buf[HID_RPT_SIZE]; |
| U2FHID_FRAME *reply = (U2FHID_FRAME *)buf; |
| U2FHID_INIT_REQ *req = (U2FHID_INIT_REQ *)frame->init.data; |
| U2FHID_INIT_RESP *resp = (U2FHID_INIT_RESP *)reply->init.data; |
| int count; |
| |
| if (MSG_LEN(*frame) != sizeof(U2FHID_INIT_REQ)) { |
| fprintf(stderr, "INIT message wrong length %d != %ld\n", |
| MSG_LEN(*frame), |
| sizeof(U2FHID_INIT_REQ)); |
| process_error(frame->cid, ERR_INVALID_LEN); |
| return; |
| } |
| if (frame->cid != CID_BROADCAST) { |
| fprintf(stderr, "INIT message to wrong cid %x\n", frame->cid); |
| process_error(frame->cid, ERR_INVALID_CMD); |
| return; |
| } |
| memset(buf, 0, sizeof(buf)); |
| reply->cid = CID_BROADCAST; |
| reply->init.cmd = U2FHID_INIT; |
| reply->init.bcnth = 0; |
| reply->init.bcntl = sizeof(*resp); |
| printf("setting reply size to %ld\n", sizeof(*resp)); |
| memcpy(resp->nonce, req->nonce, sizeof(req->nonce)); |
| resp->cid = 1; |
| resp->versionInterface = U2FHID_IF_VERSION; |
| |
| count = write(dev, buf, sizeof(buf)); |
| printf("written %d bytes\n", count); |
| } |
| |
| static void command_loop(void) |
| { |
| int count; |
| uint8_t buf[HID_RPT_SIZE]; |
| U2FHID_FRAME *frame = (U2FHID_FRAME *)buf; |
| |
| count = read(dev, buf, sizeof(buf)); |
| printf("received %d bytes\n", count); |
| if (frame->init.cmd == U2FHID_INIT) { |
| printf("Got INIT\n"); |
| process_init(frame); |
| } else if (frame->init.cmd == U2FHID_MSG){ |
| printf("Got MSG\n"); |
| process_msg(frame); |
| } else { |
| printf("Got unknown command 0x%x\n", FRAME_CMD(*frame)); |
| process_error(frame->cid, ERR_INVALID_CMD); |
| } |
| } |
| |
| int main(int argc, char *argv[]) |
| { |
| const char *file, *cert; |
| |
| for (;;) { |
| int c, option_index; |
| |
| c = getopt_long(argc, argv, "hv", |
| long_options, &option_index); |
| if (c == -1) |
| break; |
| |
| switch (c) { |
| case 'h': |
| usage(argv[0], stdout); |
| exit(0); |
| case 'v': |
| fprintf(stdout, "%s " VERSION "\n" |
| "Copyright 2019 by James Bottomley\n" |
| "License GPL-2.0-only\n" |
| "Written by James Bottomley <James.Bottomley@HansenPartnership.com>\n", |
| argv[0]); |
| exit(0); |
| default: |
| usage(argv[0], stderr); |
| exit(1); |
| } |
| } |
| |
| if (optind > argc - 2) { |
| fprintf(stderr, "too few arguments\n"); |
| usage(argv[0], stderr); |
| exit(1); |
| } |
| if (optind < argc -2) { |
| fprintf(stderr, "too many arguments\n"); |
| usage(argv[0], stderr); |
| exit(1); |
| } |
| |
| cert = argv[argc - 1]; |
| file = argv[argc - 2]; |
| |
| dev = open(file, O_RDWR); |
| if (dev < 0) { |
| fprintf(stderr, "Failed to open %s: ", file); |
| perror(""); |
| exit(1); |
| } |
| |
| certd = open(cert, O_RDWR); |
| if (certd < 0) { |
| fprintf(stderr, "Failed to open %s: ", cert); |
| perror(""); |
| exit(1); |
| } |
| |
| for (;;) { |
| command_loop(); |
| } |
| } |