scapy/test/can.uts

344 lines
17 KiB
Plaintext

% Regression tests for the CAN layer
# More information at http://www.secdev.org/projects/UTscapy/
############
############
+ Basic operations
= Load module
load_layer("can")
= Build a packet
pkt = CAN(flags="error", identifier=1234, data="test")
= Dissect & parse
pkt = CAN(raw(pkt))
pkt.flags == "error" and pkt.identifier == 1234 and pkt.length == 4 and pkt.data == b"test"
= Check flags values
pkt = CAN(flags="remote_transmission_request")
pkt.flags == 0x2
pkt = CAN(flags="extended")
pkt.flags == 0x4
############
############
+ Example PCAP file
= Read PCAP file
* From https://wiki.wireshark.org/SampleCaptures?action=AttachFile&do=get&target=CANopen.pca
from io import BytesIO
pcap_fd = BytesIO(b'\xd4\xc3\xb2\xa1\x02\x00\x04\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x04\x00\xe3\x00\x00\x00\xe2\xf3mT\x93\x8c\x03\x00\t\x00\x00\x00\t\x00\x00\x00\x00\x00\x073\x01\x00\x00\x00\x00\xe2\xf3mT\xae\x8c\x03\x00\n\x00\x00\x00\n\x00\x00\x00\x00\x00\x00\x00\x02\x7f\x00\x00\x81\x00\xe2\xf3mTI\x8f\x03\x00\t\x00\x00\x00\t\x00\x00\x00\x00\x00\x07B\x01\x00\x00\x00\x00\xe2\xf3mTM\x8f\x03\x00\t\x00\x00\x00\t\x00\x00\x00\x00\x00\x07c\x01\x00\x00\x00\x00\xe2\xf3mTN\x8f\x03\x00\t\x00\x00\x00\t\x00\x00\x00\x00\x00\x07!\x01\x00\x00\x00\x00\xf8\xf3mTv\x98\x04\x00\x10\x00\x00\x00\x10\x00\x00\x00\x00\x00\x06B\x08\x7f\x00\x00@\x08\x10\x00\x00\x00\x00\x00\xf8\xf3mT\x96\x98\x04\x00\x10\x00\x00\x00\x10\x00\x00\x00\x00\x00\x05\xc2\x08\x7f\x00\x00A\x08\x10\x00\x15\x00\x00\x00\xf8\xf3mT\xd4\x98\x04\x00\x10\x00\x00\x00\x10\x00\x00\x00\x00\x00\x06B\x08\x00\x00\x00`\x00\x00\x00\x00\x00\x00\x00\xf8\xf3mT\x12\x99\x04\x00\x10\x00\x00\x00\x10\x00\x00\x00\x00\x00\x06B\x08\x00\x00\x00\x80\x00\x00\x00!\x00\x00\x08\xf8\xf3mTC\x99\x04\x00\x10\x00\x00\x00\x10\x00\x00\x00\x00\x00\x05\xc2\x08\x7f\x00\x00\x00UltraHi\xf8\xf3mTx\x99\x04\x00\x10\x00\x00\x00\x10\x00\x00\x00\x00\x00\x06B\x08\x00\x00\x00\x80\x00\x00\x00!\x00\x00\x08\xf8\xf3mT\xce\x99\x04\x00\x10\x00\x00\x00\x10\x00\x00\x00\x00\x00\x06B\x08\x00\x00\x00p\x00\x00\x00\x00\x00\x00\x00\xf8\xf3mT\xe0\x99\x04\x00\x10\x00\x00\x00\x10\x00\x00\x00\x00\x00\x06B\x08\x00\x00\x00\x80\x00\x00\x00!\x00\x00\x08\xf8\xf3mT \x9a\x04\x00\x10\x00\x00\x00\x10\x00\x00\x00\x00\x00\x06B\x08\x00\x00\x00\x80\x00\x00\x00!\x00\x00\x08\xf8\xf3mTo\x9a\x04\x00\x10\x00\x00\x00\x10\x00\x00\x00\x00\x00\x05\xc2\x08\x00\x00\x00\x80\x00\x00\x00!\x00\x00\x083\xf4mTw\xbe\t\x00\x10\x00\x00\x00\x10\x00\x00\x00\x00\x00\x06B\x08\x7f\x00\x00@\x08\x10*\x00\x00\x00\x003\xf4mT4\xc0\t\x00\x10\x00\x00\x00\x10\x00\x00\x00\x00\x00\x05\xc2\x08\x00\x00\x00\x80\x08\x10*\x11\x00\t\x06i\xf4mT\xb0\x88\x0c\x00\x10\x00\x00\x00\x10\x00\x00\x00\x00\x00\x07\xe5\x08\x7f\x00\x00L\x00\x00\x00\x00\x00\x00\x00i\xf4mT+\x89\x0c\x00\x10\x00\x00\x00\x10\x00\x00\x00\x00\x00\x07\xe4\x08\x7f\x00\x00P\x00\x00\x00\x00\x00\x00\x00i\xf4mT-\x89\x0c\x00\x10\x00\x00\x00\x10\x00\x00\x00\x00\x00\x07\xe4\x08\x7f\x00\x00P\x00\x00\x00\x00\x00\x00\x00i\xf4mTS\x89\x0c\x00\x10\x00\x00\x00\x10\x00\x00\x00\x00\x00\x07\xe4\x08\x7f\x00\x00P\x00\x00\x00\x00\x00\x00\x00i\xf4mT\x99\x89\x0c\x00\x10\x00\x00\x00\x10\x00\x00\x00\x00\x00\x07\xe4\x08\x00\x00\x00P\x00\x00\x00\x00\x00\x00\x00\x8e\xf4mT\x86\xc4\x04\x00\n\x00\x00\x00\n\x00\x00\x00\x00\x00\x00\x00\x02\x00\x00\x00\x01B\x92\xf4mT\xae\xf0\x07\x00\n\x00\x00\x00\n\x00\x00\x00\x00\x00\x00\x00\x02\x00\x00\x00\x01\x00\xba\xf4mT%\xaa\x0b\x00\n\x00\x00\x00\n\x00\x00\x00\x00\x00\x00\x00\x02\x00\x00\x00\x02c\xe8\xf4mT\xbc\x0f\x06\x00\x10\x00\x00\x00\x10\x00\x00\x00\x00\x00\x06B\x08\x7f\x00\x00#\x00b\x01asdf\xe8\xf4mT\x07\x10\x06\x00\x10\x00\x00\x00\x10\x00\x00\x00\x00\x00\x05\xc2\x08\x00\x00\x00\x80\x00b\x01\x00\x00\x02\x06\x0f\xf5mT\x1c\x81\x00\x00\x10\x00\x00\x00\x10\x00\x00\x00\x00\x00\x06B\x08\x7f\x00\x00@\x00b\x01\x00\x00\x00\x00\x0f\xf5mT\xfe\x81\x00\x00\x10\x00\x00\x00\x10\x00\x00\x00\x00\x00\x05\xc2\x08\x00\x00\x00\x80\x00b\x01\x00\x00\x02\x068\xf5mT\x19\xc3\x00\x00\x10\x00\x00\x00\x10\x00\x00\x00\x00\x00\x06B\x08\x7f\x00\x00\xa0\x08\x10\x00\x10\x00\x00\x008\xf5mTg\xc3\x00\x00\x10\x00\x00\x00\x10\x00\x00\x00\x00\x00\x05\xc2\x08\x7f\x00\x00\xc2\x08\x10\x00\x15\x00\x00\x008\xf5mT\xd8\xc3\x00\x00\x10\x00\x00\x00\x10\x00\x00\x00\x00\x00\x06B\x08\x00\x00\x00\x80\x00\x00\x00!\x00\x00\x088\xf5mT\x17\xc4\x00\x00\x10\x00\x00\x00\x10\x00\x00\x00\x00\x00\x06B\x08\x7f\x00\x00\xa3\x00\x00\x00\x00\x00\x00\x008\xf5mT\xca\xc4\x00\x00\x10\x00\x00\x00\x10\x00\x00\x00\x00\x00\x05\xc2\x08\x00\x00\x00\x80\x00\x00\x00!\x00\x00\x08')
packets = rdpcap(pcap_fd)
= Check if parsing worked: each packet has a CAN layer
all(CAN in pkt for pkt in packets)
= Check if parsing worked: no packet has a Raw or Padding layer
not any(Raw in pkt or Padding in pkt for pkt in packets)
= Identifiers
set(pkt.identifier for pkt in packets) == {0, 1474, 1602, 1825, 1843, 1858, 1891, 2020, 2021}
= Flags
set(pkt.flags for pkt in packets) == {0}
= Data length
set(pkt.length for pkt in packets) == {1, 2, 8}
############
############
+ swap-bytes functionality (for PF_CAN socket interactions)
= read PCAP of a CookedLinux/SocketCAN capture (CAN standard and extended)
conf.contribs['CAN']['swap-bytes'] = True
pcap_fd_can_a = BytesIO(b'\xd4\xc3\xb2\xa1\x02\x00\x04\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x04\x00q\x00\x00\x00\x15f`Zv\xde\n\x00 \x00\x00\x00 \x00\x00\x00\x00\x01\x01\x18\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x0c\xdf\x07\x00\x00\x03\x00\x00\x00\x02\x01\r\x00\x00\x00\x00\x00')
packets_can_a = rdpcap(pcap_fd_can_a)
pcap_fd_can_b = BytesIO(b'\xd4\xc3\xb2\xa1\x02\x00\x04\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x04\x00q\x00\x00\x00\xf4i`Z\xf3\x99\x07\x00 \x00\x00\x00 \x00\x00\x00\x00\x01\x01\x18\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x0c\xf13\xdb\x98\x03\x00\x00\x00\x02\x01\r\x00\x00\x00\x00\x00')
packets_can_b = rdpcap(pcap_fd_can_b)
= check CAN is detected over CookedLinux (each packet has both layers)
all(CAN in pkt for pkt in packets_can_a)
all(CAN in pkt for pkt in packets_can_b)
all(CookedLinux in pkt for pkt in packets_can_a)
all(CookedLinux in pkt for pkt in packets_can_b)
= Check if parsing worked: no packet has a Raw or Padding layer
not any(Raw in pkt or Padding in pkt for pkt in packets)
= Check byte swap for dissection
packets_can_a[0].identifier == 0x7df
packets_can_a[0].flags == 0x0
packets_can_b[0].identifier == 0x18db33f1
packets_can_b[0].flags == "extended"
= Check byte swap-back for building
raw(packets_can_a[0]) == b'\x00\x01\x01\x18\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x0c\xdf\x07\x00\x00\x03\x00\x00\x00\x02\x01\r\x00\x00\x00\x00\x00'
raw(packets_can_b[0]) == b'\x00\x01\x01\x18\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x0c\xf13\xdb\x98\x03\x00\x00\x00\x02\x01\r\x00\x00\x00\x00\x00'
= Check building CAN packet with not padded data field
* check building
p = CAN(flags='error', identifier=1234, data=b'')
bytes(p)
p = CAN(flags='error', identifier=1234, data=b'\x0a\x0b')
bytes(p)
* check padding handling
p_too_much_data = CAN(flags='error', length=1, identifier=1234, data=b'\x01\x02')
p = CAN(bytes(p_too_much_data))
p.haslayer('Padding') and p['Padding'].load == b'\x02'
= Check rdcandump default
* default reading
pcap_fd = BytesIO(b'''(1539191392.761779) vcan0 123#11223344
(1539191470.820239) vcan0 123#11223344
(1539191471.503168) vcan0 123#11223344
(1539191471.891423) vcan0 123#11223344
(1539191492.026403) vcan0 1F334455#1122334455667788
(1539191494.084177) vcan0 1F334455#1122334455667788
(1539191494.724228) vcan0 1F334455#1122334455667788
(1539191495.148182) vcan0 1F334455#1122334455667788
(1539191495.563320) vcan0 1F334455#1122334455667788''')
packets = rdcandump(pcap_fd)
assert len(packets) == 9
assert packets[0].identifier == 0x123
assert packets[8].identifier == 0x1F334455
assert packets[8].flags == 0b100
assert packets[0].length == 4
assert packets[8].length == 8
assert packets[0].data == b'\x11\x22\x33\x44'
assert packets[8].data == b'\x11\x22\x33\x44\x55\x66\x77\x88'
= Check rdcandump filter
* interface filter 1
pcap_fd = BytesIO(b'''(1539191392.761779) vcan0 123#11223344
(1539191470.820239) vcan1 123#11223344
(1539191471.503168) vcan1 123#11223344
(1539191471.891423) vcan0 123#11223344
(1539191492.026403) vcan0 1F334455#1122334455667788
(1539191494.084177) vcan1 1F334455#1122334455667788
(1539191494.724228) vcan1 1F334455#1122334455667788
(1539191495.148182) vcan0 1F334455#1122334455667788
(1539191495.563320) vcan1 1F334455#1122334455667788''')
packets = rdcandump(pcap_fd, interface="vcan0")
assert len(packets) == 4
assert packets[0].identifier == 0x123
assert packets[-1].identifier == 0x1F334455
assert packets[-1].flags == 0b100
assert packets[0].length == 4
assert packets[-1].length == 8
assert packets[0].data == b'\x11\x22\x33\x44'
assert packets[-1].data == b'\x11\x22\x33\x44\x55\x66\x77\x88'
* interface filter 2
pcap_fd = BytesIO(b'''(1539191392.761779) vcan0 123#11223344
(1539191470.820239) vcan0 123#11223344
(1539191471.503168) vcan0 123#11223344
(1539191471.891423) vcan0 123#11223344
(1539191492.026403) vcan1 1F334455#1122334455667788
(1539191494.084177) vcan1 1F334455#1122334455667788
(1539191494.724228) vcan1 1F334455#1122334455667788
(1539191495.148182) vcan1 1F334455#1122334455667788
(1539191495.563320) vcan1 1F334455#1122334455667788''')
packets = rdcandump(pcap_fd, interface="vcan0")
assert len(packets) == 4
assert packets[0].identifier == 0x123
assert packets[0].length == 4
assert packets[0].data == b'\x11\x22\x33\x44'
* interface filter 3
pcap_fd = BytesIO(b'''(1539191392.761779) vcan0 123#11223344
(1539191470.820239) vcan0 123#11223344
(1539191471.503168) vcan0 123#11223344
(1539191471.891423) vcan0 123#11223344
(1539191492.026403) vcan1 1F334455#1122334455667788
(1539191494.084177) vcan1 1F334455#1122334455667788
(1539191494.724228) vcan1 1F334455#1122334455667788
(1539191495.148182) vcan1 1F334455#1122334455667788
(1539191495.563320) vcan1 1F334455#1122334455667788''')
packets = rdcandump(pcap_fd, interface="vcan1")
assert len(packets) == 5
assert packets[-1].identifier == 0x1F334455
assert packets[-1].flags == 0b100
assert packets[-1].length == 8
assert packets[-1].data == b'\x11\x22\x33\x44\x55\x66\x77\x88'
* interface filter 4
pcap_fd = BytesIO(b'''(1539191392.761779) vcan2 123#11223344
(1539191470.820239) vcan0 123#11223344
(1539191471.503168) vcan2 123#11223344
(1539191471.891423) vcan0 123#11223344
(1539191492.026403) vcan1 1F334455#1122334455667788
(1539191494.084177) vcan1 1F334455#1122334455667788
(1539191494.724228) vcan2 1F334455#1122334455667788
(1539191495.148182) vcan1 1F334455#1122334455667788
(1539191495.563320) vcan2 1F334455#1122334455667788''')
packets = rdcandump(pcap_fd, interface=["vcan1", "vcan0"])
assert len(packets) == 5
assert packets[0].identifier == 0x123
assert packets[-1].identifier == 0x1F334455
assert packets[-1].flags == 0b100
assert packets[0].length == 4
assert packets[-1].length == 8
assert packets[0].data == b'\x11\x22\x33\x44'
assert packets[-1].data == b'\x11\x22\x33\x44\x55\x66\x77\x88'
= Check rdcandump not log file format
* interface not log file format
pcap_fd = BytesIO(b''' vcan0 1F334455 [8] 11 22 33 44 55 66 77 88
vcan0 1F3 [8] 11 22 33 44 55 66 77 88
vcan0 1F3 [8] 11 22 33 44 55 66 77 88
vcan0 1F334455 [8] 11 22 33 44 55 66 77 88
vcan0 1F3 [8] 11 22 33 44 55 66 77 88
vcan0 1F334455 [8] 11 22 33 44 55 66 77 88
vcan0 1F334455 [4] 11 22 33 44
vcan0 1F3 [4] 11 22 33 44''')
packets = rdcandump(pcap_fd, is_not_log_file_format=True)
assert len(packets) == 8
packets[-1].show()
assert packets[-1].identifier == 0x1F3
assert packets[1].identifier == 0x1F3
assert packets[0].identifier == 0x1F334455
assert packets[0].flags == 0b100
assert packets[-1].length == 4
assert packets[0].length == 8
assert packets[1].length == 8
assert packets[-1].data == b'\x11\x22\x33\x44'
assert packets[0].data == b'\x11\x22\x33\x44\x55\x66\x77\x88'
assert packets[1].data == b'\x11\x22\x33\x44\x55\x66\x77\x88'
* interface not log file format filtered 1
pcap_fd = BytesIO(b''' vcan0 1F334455 [8] 11 22 33 44 55 66 77 88
vcan1 1F3 [8] 11 22 33 44 55 66 77 88
vcan1 1F3 [8] 11 22 33 44 55 66 77 88
vcan0 1F334455 [8] 11 22 33 44 55 66 77 88
vcan0 1F3 [8] 11 22 33 44 55 66 77 88
vcan1 1F334455 [8] 11 22 33 44 55 66 77 88
vcan1 1F334455 [4] 11 22 33 44
vcan0 1F3 [4] 11 22 33 44
''')
packets = rdcandump(pcap_fd, is_not_log_file_format=True, interface="vcan0")
assert len(packets) == 4
assert packets[-1].identifier == 0x1F3
assert packets[2].identifier == 0x1F3
assert packets[0].identifier == 0x1F334455
assert packets[0].flags == 0b100
assert packets[-1].length == 4
assert packets[0].length == 8
assert packets[2].length == 8
assert packets[-1].data == b'\x11\x22\x33\x44'
assert packets[0].data == b'\x11\x22\x33\x44\x55\x66\x77\x88'
assert packets[2].data == b'\x11\x22\x33\x44\x55\x66\x77\x88'
* interface not log file format filtered 2
pcap_fd = BytesIO(b''' vcan0 1F334455 [8] 11 22 33 44 55 66 77 88
vcan1 1F3 [8] 11 22 33 44 55 66 77 88
vcan2 1F3 [8] 11 22 33 44 55 66 77 88
vcan0 1F334455 [8] 11 22 33 44 55 66 77 88
vcan0 1F3 [8] 11 22 33 44 55 66 77 88
vcan1 1F334455 [8] 11 22 33 44 55 66 77 88
vcan2 1F334455 [4] 11 22 33 44
vcan0 1F3 [4] 11 22 33 44
''')
packets = rdcandump(pcap_fd, is_not_log_file_format=True,
interface=["vcan0", "vcan1"])
assert len(packets) == 6
assert packets[-1].identifier == 0x1F3
assert packets[1].identifier == 0x1F3
assert packets[0].identifier == 0x1F334455
assert packets[0].flags == 0b100
assert packets[-1].length == 4
assert packets[0].length == 8
assert packets[1].length == 8
assert packets[-1].data == b'\x11\x22\x33\x44'
assert packets[0].data == b'\x11\x22\x33\x44\x55\x66\x77\x88'
assert packets[1].data == b'\x11\x22\x33\x44\x55\x66\x77\x88'
= Check rdcandump count
* interface not log file format filtered 2 count 1
pcap_fd = BytesIO(b''' vcan0 1F334455 [8] 11 22 33 44 55 66 77 88
vcan1 1F3 [8] 11 22 33 44 55 66 77 88
vcan2 1F3 [8] 11 22 33 44 55 66 77 88
vcan0 1F334455 [8] 11 22 33 44 55 66 77 88
vcan0 1F3 [8] 11 22 33 44 55 66 77 88
vcan2 1F334455 [8] 11 22 33 44 55 66 77 88
vcan2 1F334455 [4] 11 22 33 44
vcan0 1F3 [4] 11 22 33 44
''')
packets = rdcandump(pcap_fd, is_not_log_file_format=True,
interface=["vcan2"],
count=2)
assert len(packets) == 2
assert packets[0].identifier == 0x1F3
assert packets[-1].identifier == 0x1F334455
assert packets[-1].flags == 0b100
assert packets[-1].length == 8
assert packets[0].length == 8
assert packets[1].length == 8
assert packets[0].data == b'\x11\x22\x33\x44\x55\x66\x77\x88'
assert packets[1].data == b'\x11\x22\x33\x44\x55\x66\x77\x88'
* interface not log file format filtered 2 count 2
pcap_fd = BytesIO(b''' vcan0 1F334455 [8] 11 22 33 44 55 66 77 88
vcan1 1F3 [8] 11 22 33 44 55 66 77 88
vcan2 1F3 [8] 11 22 33 44 55 66 77 88
vcan0 1F334455 [8] 11 22 33 44 55 66 77 88
vcan0 1F3 [8] 11 22 33 44 55 66 77 88
vcan2 1F334455 [8] 11 22 33 44 55 66 77 88
vcan2 1F334455 [4] 11 22 33 44
vcan0 1F3 [4] 11 22 33 44
''')
packets = rdcandump(pcap_fd, is_not_log_file_format=True,
count=2)
assert len(packets) == 2
assert packets[1].identifier == 0x1F3
assert packets[0].identifier == 0x1F334455
assert packets[0].flags == 0b100
assert packets[-1].length == 8
assert packets[0].length == 8
assert packets[1].length == 8
assert packets[0].data == b'\x11\x22\x33\x44\x55\x66\x77\x88'
assert packets[1].data == b'\x11\x22\x33\x44\x55\x66\x77\x88'
* default reading
pcap_fd = BytesIO(b'''(1539191392.761779) vcan0 123#11223344
(1539191470.820239) vcan0 123#11223344
(1539191471.503168) vcan0 123#11223344
(1539191471.891423) vcan0 123#11223344
(1539191492.026403) vcan0 1F334455#1122334455667788
(1539191494.084177) vcan0 1F334455#1122334455667788
(1539191494.724228) vcan0 1F334455#1122334455667788
(1539191495.148182) vcan0 1F334455#1122334455667788
(1539191495.563320) vcan0 1F334455#1122334455667788''')
packets = rdcandump(pcap_fd, count=5)
assert len(packets) == 5
assert packets[0].identifier == 0x123
assert packets[-1].identifier == 0x1F334455
assert packets[-1].flags == 0b100
assert packets[0].length == 4
assert packets[-1].length == 8
assert packets[0].data == b'\x11\x22\x33\x44'
assert packets[-1].data == b'\x11\x22\x33\x44\x55\x66\x77\x88'