-
Notifications
You must be signed in to change notification settings - Fork 9
/
session.c
192 lines (179 loc) · 6.19 KB
/
session.c
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
#include "session.h"
#include <stdarg.h>
#include <stdio.h>
#include <string.h>
#define NO_UPDATE (0xFFFF)
// From ISO/IEC 7816-3:2006 Section 8.3 Table 7
static unsigned clock_conversion(unsigned char speed) {
switch (speed >> 4) {
case 2: return 558;
case 3: return 744;
case 4: return 1116;
case 5: return 1488;
case 6: return 1860;
case 9: return 512;
case 0xA: return 768;
case 0xB: return 1024;
case 0xC: return 1536;
case 0xD: return 2048;
default: return 372;
}
}
// From ISO/IEC 7816-3:2006 Section 8.3 Table 8
static unsigned baud_divisor(unsigned char speed) {
switch (speed & 0x0F) {
case 2: return 2;
case 3: return 4;
case 4: return 8;
case 5: return 16;
case 6: return 32;
case 7: return 64;
case 8: return 12;
case 9: return 20;
default: return 1;
}
}
#define BASE_ETU (372)
static void send_packet(struct session *session, enum result result) {
struct current_session *curr = &session->curr;
struct packet packet;
packet.data = curr->buf;
packet.data_length = curr->buf_index;
packet.result = result;
packet.time = curr->buf_time;
session->completed_packet(&packet);
}
static void session_log(struct session *session, const char *fmt, ...) {
char buf[SESSION_BUFLEN];
int size = sizeof(buf);
int len;
va_list args;
va_start(args, fmt);
len = vsnprintf(buf, size, fmt, args);
va_end(args);
if (len >= SESSION_BUFLEN) buf[SESSION_BUFLEN-1] = 0;
if (len > 0) session->log_msg(buf);
}
void session_reset(struct session *session) {
struct current_session *curr = &session->curr;
if (curr->buf_index > 0) {
// Incomplete packet in buffer, consider it noise
send_packet(session, NOISE);
curr->buf_index = 0;
}
memset(&session->curr, 0, sizeof(struct current_session));
atr_init(&curr->atr);
pps_init(&curr->pps);
data_init(&curr->data);
curr->baudrate = session->base_baudrate;
session->set_baudrate(session->serial_fd, session->base_baudrate);
}
static void update_speed(struct session *session, unsigned speed, char *phase) {
unsigned new_etu = clock_conversion(speed) / baud_divisor(speed);
unsigned baudrate = session->base_baudrate * BASE_ETU / ((float) new_etu);
if (baudrate == session->curr.baudrate) return;
session_log(session, "Switching to %d ticks per ETU (%d baud) after %s",
new_etu, baudrate, phase);
session->curr.baudrate = baudrate;
session->set_baudrate(session->serial_fd, baudrate);
}
void session_init(struct session *session, completed_packet_fn completed_packet,
set_baudrate_fn set_baudrate, log_msg_fn log_msg, int fd,
unsigned baudrate) {
memset(session, 0, sizeof(struct session));
session->set_baudrate = set_baudrate;
session->serial_fd = fd;
session->base_baudrate = baudrate;
session->completed_packet = completed_packet;
session->log_msg = log_msg;
session_reset(session);
}
static enum result analyze_byte(struct current_session *curr,
unsigned char data, unsigned *complete) {
switch (curr->state) {
case INIT:
// Ignore early noise
if (data == 0x00 || data == 0xFF) return NOISE;
curr->state = ATR;
// Fallthrough
case ATR:
return atr_analyze(&curr->atr, data, complete);
case IDLE:
if (data == 0xFF) {
// PPS start byte
curr->state = PPS;
return pps_analyze(&curr->pps, data, complete);
}
if (curr->protocol_version == 0) {
curr->state = T0_DATA;
return data_t0_analyze(&curr->data, data);
}
if (curr->protocol_version == 1) {
curr->state = T1_DATA;
return data_t1_analyze(&curr->data, data);
}
return STATE_ERROR;
case PPS:
return pps_analyze(&curr->pps, data, complete);
case T0_DATA:
return data_t0_analyze(&curr->data, data);
case T1_DATA:
return data_t1_analyze(&curr->data, data);
}
return STATE_ERROR;
}
unsigned char convert_from_inverse(unsigned char data) {
unsigned i;
unsigned char out = 0;
for (i = 0; i < 8; i++) {
if (((data >> i) & 0x01) == 0) {
out |= 1 << (7-i);
}
}
return out;
}
void session_add_byte(struct session *session, unsigned char data) {
struct current_session *curr = &session->curr;
enum result res = STATE_ERROR;
unsigned phase_complete = 0;
if (curr->state == INIT && convert_from_inverse(data) == ATR_TS_INVERSE) {
// Inverse convention signaled in start of ATR
curr->inverse_convention = 1;
session_log(session, "Switching to inverse convention");
}
if (curr->buf_index < SESSION_BUFLEN) {
if (curr->inverse_convention) data = convert_from_inverse(data);
curr->buf[curr->buf_index++] = data;
res = analyze_byte(curr, data, &phase_complete);
}
if (curr->buf_index == 1) {
// Record time of first byte
gettimeofday(&curr->buf_time, NULL);
}
if (res) {
send_packet(session, res);
memset(curr->buf, 0, SESSION_BUFLEN);
curr->buf_index = 0;
if (phase_complete) {
unsigned proto = NO_UPDATE;
unsigned speed = NO_UPDATE;
char *phase = "?";
if (curr->state == ATR) {
atr_result(&curr->atr, &proto, &speed);
phase = "ATR";
} else if (curr->state == PPS) {
pps_result(&curr->pps, &proto, &speed);
phase = "PPS";
}
if (proto != NO_UPDATE && proto != curr->protocol_version) {
curr->protocol_version = proto;
session_log(session, "Switching to protocol T=%d after %s",
curr->protocol_version, phase);
}
if (speed != NO_UPDATE) {
update_speed(session, speed, phase);
}
curr->state = IDLE;
}
}
}