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
|
#include <isotp/send.h>
#define PCI_START_BIT 0
#define PCI_WIDTH 4
#define PAYLOAD_LENGTH_START_BIT PCI_START_BIT + PCI_WIDTH
#define PAYLOAD_LENGTH_WIDTH 4
#define PAYLOAD_START_BIT PAYLOAD_LENGTH_START_BIT + PAYLOAD_LENGTH_WIDTH
void isotp_complete_send(IsoTpHandler* handler, IsoTpMessage* message,
bool status) {
handler->message_sent_callback(message, status);
}
bool isotp_send_single_frame(IsoTpHandler* handler, IsoTpMessage* message) {
uint64_t data = 0;
setBitField(&data, PCI_SINGLE, PCI_START_BIT, PCI_WIDTH);
setBitField(&data, message->size, PAYLOAD_LENGTH_START_BIT, PAYLOAD_LENGTH_WIDTH);
// TODO need a better bitfield API to support this - use byte array instead
// of uint64_t and specify desired total width
for(int i = 0; i < message->size; i++) {
setBitField(&data, message->payload[i], PAYLOAD_START_BIT + i * 8, 8);
}
uint8_t data_array[message->size + 1];
for(int i = 0; i < sizeof(data_array); i++) {
// TODO need getByte(x) function
data_array[i] = getBitField(data, i * 8, 8, false);
}
handler->shims->send_can_message(message->arbitration_id, data_array, sizeof(data_array));
isotp_complete_send(handler, message, true);
return true;
}
bool isotp_send_multi_frame(IsoTpHandler* handler, IsoTpMessage* message) {
// TODO make sure to copy payload into a local buffer
handler->shims->log("Only single frame messages are supported");
return false;
}
bool isotp_send(IsoTpHandler* handler, const uint8_t* payload,
uint16_t size) {
// we determine if it's single/multi frame and start the send
IsoTpMessage message = {
arbitration_id: handler->arbitration_id,
payload: payload,
size: size
};
if(size < 8) {
return isotp_send_single_frame(handler, &message);
} else {
return isotp_send_multi_frame(handler, &message);
}
}
|