FreeCalypso > hg > fc-sim-sniff
view sw/sniff-dec/command.c @ 58:95ed46b5f8f1 default tip
doc/Sniffing-hw-setup: mv-sniffer is here
author | Mychaela Falconia <falcon@freecalypso.org> |
---|---|
date | Wed, 04 Oct 2023 05:55:09 +0000 |
parents | eb4274e7f4da |
children |
line wrap: on
line source
/* * Here we implement decoding of command APDU exchanges. */ #include <sys/types.h> #include <stdio.h> #include <stdlib.h> #include <string.h> #include <strings.h> #include "state.h" extern char linebuf[]; extern int lineno; extern unsigned rx_byte; extern int state; static char cmd_start_timestamp[18]; static int cmd_start_line; static char data_start_timestamp[18]; static int data_start_line; static char sw1_timestamp[18]; static int sw1_line; static u_char cmd_hdr[5]; static unsigned hdr_byte_count; static unsigned data_total, data_sofar, data_thistime; static u_char data_buf[256], sw1; void start_cmd_header() { strcpy(cmd_start_timestamp, linebuf); cmd_start_line = lineno; cmd_hdr[0] = rx_byte; hdr_byte_count = 1; state = STATE_CMD_HDR; } static void print_cmd_hdr() { unsigned n; printf("%s line %d: CMD", cmd_start_timestamp, cmd_start_line); for (n = 0; n < 5; n++) printf(" %02X", cmd_hdr[n]); putchar('\n'); } void cmd_hdr_byte_in() { cmd_hdr[hdr_byte_count++] = rx_byte; if (hdr_byte_count < 5) return; print_cmd_hdr(); if ((cmd_hdr[1] & 0xF0) == 0x60 || (cmd_hdr[1] & 0xF0) == 0x90) { printf(" ERROR: INS byte is invalid!\n"); state = STATE_ERROR; return; } decode_cmd_opcode(cmd_hdr[1]); if (cmd_hdr[4]) data_total = cmd_hdr[4]; else data_total = 256; data_sofar = 0; state = STATE_CMD_PROC; } static void print_data() { unsigned n; printf("%s line %d: DATA\n", data_start_timestamp, data_start_line); for (n = 0; n < data_sofar; n++) { printf(" %02X", data_buf[n]); if ((n & 15) == 15 || n == data_sofar - 1) putchar('\n'); } } static void handle_sw1() { strcpy(sw1_timestamp, linebuf); sw1_line = lineno; sw1 = rx_byte; if (data_sofar) print_data(); if (cmd_hdr[1] == 0xA4 && cmd_hdr[4] == 2 && data_sofar == 2) decode_file_id((data_buf[0] << 8) | data_buf[1]); state = STATE_CMD_SW; } void handle_ack(single) { if (data_sofar >= data_total) { printf("%s line %d: ERROR: ACK for more data than possible\n", linebuf, lineno); state = STATE_ERROR; return; } if (!data_sofar) { strcpy(data_start_timestamp, linebuf); data_start_line = lineno; } if (single) data_thistime = 1; else data_thistime = data_total - data_sofar; state = STATE_CMD_DATA; } void cmd_proc_byte_in() { if (rx_byte == 0x60) return; if ((rx_byte & 0xF0) == 0x60 || (rx_byte & 0xF0) == 0x90) { handle_sw1(); return; } if (rx_byte == cmd_hdr[1]) { handle_ack(0); return; } if (rx_byte == (cmd_hdr[1] ^ 0xFF)) { handle_ack(1); return; } printf("%s line %d: ERROR: invalid procedure byte\n", linebuf, lineno); state = STATE_ERROR; } void cmd_data_byte_in() { data_buf[data_sofar++] = rx_byte; data_thistime--; if (data_thistime) return; state = STATE_CMD_PROC; } void cmd_sw2_byte_in() { printf("%s line %d: SW %02X %02X\n", sw1_timestamp, sw1_line, sw1, rx_byte); state = STATE_READY_FOR_CMD; }