sixlowpan.rs 16 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413
  1. use super::*;
  2. #[rstest]
  3. #[case::ieee802154(Medium::Ieee802154)]
  4. #[cfg(feature = "medium-ieee802154")]
  5. fn ieee802154_wrong_pan_id(#[case] medium: Medium) {
  6. let data = [
  7. 0x41, 0xcc, 0x3b, 0xff, 0xbe, 0x42, 0x42, 0x42, 0x42, 0x42, 0x42, 0x0b, 0x1a, 0x62, 0x3a,
  8. 0xa6, 0x34, 0x57, 0x29, 0x1c, 0x26,
  9. ];
  10. let response = None;
  11. let (mut iface, mut sockets, _device) = setup(medium);
  12. assert_eq!(
  13. iface.inner.process_ieee802154(
  14. &mut sockets,
  15. PacketMeta::default(),
  16. &data[..],
  17. &mut iface.fragments
  18. ),
  19. response,
  20. );
  21. }
  22. #[rstest]
  23. #[case::ieee802154(Medium::Ieee802154)]
  24. #[cfg(feature = "medium-ieee802154")]
  25. fn icmp_echo_request(#[case] medium: Medium) {
  26. let data = [
  27. 0x41, 0xcc, 0x3b, 0xef, 0xbe, 0x42, 0x42, 0x42, 0x42, 0x42, 0x42, 0x0b, 0x1a, 0x62, 0x3a,
  28. 0xa6, 0x34, 0x57, 0x29, 0x1c, 0x26, 0x6a, 0x33, 0x0a, 0x62, 0x17, 0x3a, 0x80, 0x00, 0xb0,
  29. 0xe3, 0x00, 0x04, 0x00, 0x01, 0x82, 0xf2, 0x82, 0x64, 0x00, 0x00, 0x00, 0x00, 0x66, 0x23,
  30. 0x0c, 0x00, 0x00, 0x00, 0x00, 0x00, 0x10, 0x11, 0x12, 0x13, 0x14, 0x15, 0x16, 0x17, 0x18,
  31. 0x19, 0x1a, 0x1b, 0x1c, 0x1d, 0x1e, 0x1f, 0x20, 0x21, 0x22, 0x23, 0x24, 0x25, 0x26, 0x27,
  32. 0x28, 0x29, 0x2a, 0x2b, 0x2c, 0x2d, 0x2e, 0x2f, 0x30, 0x31, 0x32, 0x33, 0x34, 0x35, 0x36,
  33. 0x37,
  34. ];
  35. let response = Some(IpPacket::new_ipv6(
  36. Ipv6Repr {
  37. src_addr: Ipv6Address::from_parts(&[0xfe80, 0, 0, 0, 0x180b, 0x4242, 0x4242, 0x4242]),
  38. dst_addr: Ipv6Address::from_parts(&[0xfe80, 0, 0, 0, 0x241c, 0x2957, 0x34a6, 0x3a62]),
  39. hop_limit: 64,
  40. next_header: IpProtocol::Icmpv6,
  41. payload_len: 64,
  42. },
  43. IpPayload::Icmpv6(Icmpv6Repr::EchoReply {
  44. ident: 4,
  45. seq_no: 1,
  46. data: &[
  47. 0x82, 0xf2, 0x82, 0x64, 0x00, 0x00, 0x00, 0x00, 0x66, 0x23, 0x0c, 0x00, 0x00, 0x00,
  48. 0x00, 0x00, 0x10, 0x11, 0x12, 0x13, 0x14, 0x15, 0x16, 0x17, 0x18, 0x19, 0x1a, 0x1b,
  49. 0x1c, 0x1d, 0x1e, 0x1f, 0x20, 0x21, 0x22, 0x23, 0x24, 0x25, 0x26, 0x27, 0x28, 0x29,
  50. 0x2a, 0x2b, 0x2c, 0x2d, 0x2e, 0x2f, 0x30, 0x31, 0x32, 0x33, 0x34, 0x35, 0x36, 0x37,
  51. ],
  52. }),
  53. ));
  54. let (mut iface, mut sockets, _device) = setup(medium);
  55. assert_eq!(
  56. iface.inner.process_ieee802154(
  57. &mut sockets,
  58. PacketMeta::default(),
  59. &data[..],
  60. &mut iface.fragments
  61. ),
  62. response,
  63. );
  64. }
  65. #[test]
  66. #[cfg(feature = "proto-sixlowpan-fragmentation")]
  67. fn test_echo_request_sixlowpan_128_bytes() {
  68. use crate::phy::Checksum;
  69. let (mut iface, mut sockets, mut device) = setup(Medium::Ieee802154);
  70. // TODO: modify the example, such that we can also test if the checksum is correctly
  71. // computed.
  72. iface.inner.caps.checksum.icmpv6 = Checksum::None;
  73. assert_eq!(iface.inner.caps.medium, Medium::Ieee802154);
  74. let now = iface.inner.now();
  75. iface.inner.neighbor_cache.fill(
  76. Ipv6Address([0xfe, 0x80, 0, 0, 0, 0, 0, 0, 0x2, 0, 0, 0, 0, 0, 0, 0]).into(),
  77. HardwareAddress::Ieee802154(Ieee802154Address::default()),
  78. now,
  79. );
  80. let mut ieee802154_repr = Ieee802154Repr {
  81. frame_type: Ieee802154FrameType::Data,
  82. security_enabled: false,
  83. frame_pending: false,
  84. ack_request: false,
  85. sequence_number: Some(5),
  86. pan_id_compression: true,
  87. frame_version: Ieee802154FrameVersion::Ieee802154_2003,
  88. dst_pan_id: Some(Ieee802154Pan(0xbeef)),
  89. dst_addr: Some(Ieee802154Address::Extended([
  90. 0x90, 0xfc, 0x48, 0xc2, 0xa4, 0x41, 0xfc, 0x76,
  91. ])),
  92. src_pan_id: Some(Ieee802154Pan(0xbeef)),
  93. src_addr: Some(Ieee802154Address::Extended([
  94. 0x42, 0x42, 0x42, 0x42, 0x42, 0x42, 0x0b, 0x1a,
  95. ])),
  96. };
  97. // NOTE: this data is retrieved from tests with Contiki-NG
  98. let request_first_part_packet = SixlowpanFragPacket::new_checked(&[
  99. 0xc0, 0xb0, 0x00, 0x8e, 0x6a, 0x33, 0x05, 0x25, 0x2c, 0x3a, 0x80, 0x00, 0xe0, 0x71, 0x00,
  100. 0x27, 0x00, 0x02, 0xa2, 0xc2, 0x2d, 0x63, 0x00, 0x00, 0x00, 0x00, 0xd9, 0x5e, 0x0c, 0x00,
  101. 0x00, 0x00, 0x00, 0x00, 0x10, 0x11, 0x12, 0x13, 0x14, 0x15, 0x16, 0x17, 0x18, 0x19, 0x1a,
  102. 0x1b, 0x1c, 0x1d, 0x1e, 0x1f, 0x20, 0x21, 0x22, 0x23, 0x24, 0x25, 0x26, 0x27, 0x28, 0x29,
  103. 0x2a, 0x2b, 0x2c, 0x2d, 0x2e, 0x2f, 0x30, 0x31, 0x32, 0x33, 0x34, 0x35, 0x36, 0x37, 0x38,
  104. 0x39, 0x3a, 0x3b, 0x3c, 0x3d, 0x3e, 0x3f, 0x40, 0x41, 0x42, 0x43, 0x44, 0x45, 0x46, 0x47,
  105. 0x48, 0x49, 0x4a, 0x4b, 0x4c, 0x4d, 0x4e, 0x4f,
  106. ])
  107. .unwrap();
  108. let request_first_part_iphc_packet =
  109. SixlowpanIphcPacket::new_checked(request_first_part_packet.payload()).unwrap();
  110. let request_first_part_iphc_repr = SixlowpanIphcRepr::parse(
  111. &request_first_part_iphc_packet,
  112. ieee802154_repr.src_addr,
  113. ieee802154_repr.dst_addr,
  114. &iface.inner.sixlowpan_address_context,
  115. )
  116. .unwrap();
  117. assert_eq!(
  118. request_first_part_iphc_repr.src_addr,
  119. Ipv6Address([
  120. 0xfe, 0x80, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x40, 0x42, 0x42, 0x42, 0x42, 0x42, 0xb,
  121. 0x1a,
  122. ]),
  123. );
  124. assert_eq!(
  125. request_first_part_iphc_repr.dst_addr,
  126. Ipv6Address([
  127. 0xfe, 0x80, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x92, 0xfc, 0x48, 0xc2, 0xa4, 0x41, 0xfc,
  128. 0x76,
  129. ]),
  130. );
  131. let request_second_part = [
  132. 0xe0, 0xb0, 0x00, 0x8e, 0x10, 0x50, 0x51, 0x52, 0x53, 0x54, 0x55, 0x56, 0x57, 0x58, 0x59,
  133. 0x5a, 0x5b, 0x5c, 0x5d, 0x5e, 0x5f, 0x60, 0x61, 0x62, 0x63, 0x64, 0x65, 0x66, 0x67, 0x68,
  134. 0x69, 0x6a, 0x6b, 0x6c, 0x6d, 0x6e, 0x6f, 0x70, 0x71, 0x72, 0x73, 0x74, 0x75, 0x76, 0x77,
  135. 0x78, 0x79, 0x7a, 0x7b, 0x7c, 0x7d, 0x7e, 0x7f,
  136. ];
  137. assert_eq!(
  138. iface.inner.process_sixlowpan(
  139. &mut sockets,
  140. PacketMeta::default(),
  141. &ieee802154_repr,
  142. &request_first_part_packet.into_inner()[..],
  143. &mut iface.fragments
  144. ),
  145. None
  146. );
  147. ieee802154_repr.sequence_number = Some(6);
  148. // data that was generated when using `ping -s 128`
  149. let data = &[
  150. 0xa2, 0xc2, 0x2d, 0x63, 0x00, 0x00, 0x00, 0x00, 0xd9, 0x5e, 0x0c, 0x00, 0x00, 0x00, 0x00,
  151. 0x00, 0x10, 0x11, 0x12, 0x13, 0x14, 0x15, 0x16, 0x17, 0x18, 0x19, 0x1a, 0x1b, 0x1c, 0x1d,
  152. 0x1e, 0x1f, 0x20, 0x21, 0x22, 0x23, 0x24, 0x25, 0x26, 0x27, 0x28, 0x29, 0x2a, 0x2b, 0x2c,
  153. 0x2d, 0x2e, 0x2f, 0x30, 0x31, 0x32, 0x33, 0x34, 0x35, 0x36, 0x37, 0x38, 0x39, 0x3a, 0x3b,
  154. 0x3c, 0x3d, 0x3e, 0x3f, 0x40, 0x41, 0x42, 0x43, 0x44, 0x45, 0x46, 0x47, 0x48, 0x49, 0x4a,
  155. 0x4b, 0x4c, 0x4d, 0x4e, 0x4f, 0x50, 0x51, 0x52, 0x53, 0x54, 0x55, 0x56, 0x57, 0x58, 0x59,
  156. 0x5a, 0x5b, 0x5c, 0x5d, 0x5e, 0x5f, 0x60, 0x61, 0x62, 0x63, 0x64, 0x65, 0x66, 0x67, 0x68,
  157. 0x69, 0x6a, 0x6b, 0x6c, 0x6d, 0x6e, 0x6f, 0x70, 0x71, 0x72, 0x73, 0x74, 0x75, 0x76, 0x77,
  158. 0x78, 0x79, 0x7a, 0x7b, 0x7c, 0x7d, 0x7e, 0x7f,
  159. ];
  160. let result = iface.inner.process_sixlowpan(
  161. &mut sockets,
  162. PacketMeta::default(),
  163. &ieee802154_repr,
  164. &request_second_part,
  165. &mut iface.fragments,
  166. );
  167. assert_eq!(
  168. result,
  169. Some(IpPacket::new_ipv6(
  170. Ipv6Repr {
  171. src_addr: Ipv6Address([
  172. 0xfe, 0x80, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x92, 0xfc, 0x48, 0xc2, 0xa4, 0x41,
  173. 0xfc, 0x76,
  174. ]),
  175. dst_addr: Ipv6Address([
  176. 0xfe, 0x80, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x40, 0x42, 0x42, 0x42, 0x42, 0x42,
  177. 0xb, 0x1a,
  178. ]),
  179. next_header: IpProtocol::Icmpv6,
  180. payload_len: 136,
  181. hop_limit: 64,
  182. },
  183. IpPayload::Icmpv6(Icmpv6Repr::EchoReply {
  184. ident: 39,
  185. seq_no: 2,
  186. data,
  187. })
  188. ))
  189. );
  190. iface.inner.neighbor_cache.fill(
  191. IpAddress::Ipv6(Ipv6Address([
  192. 0xfe, 0x80, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x40, 0x42, 0x42, 0x42, 0x42, 0x42, 0xb, 0x1a,
  193. ])),
  194. HardwareAddress::Ieee802154(Ieee802154Address::default()),
  195. Instant::now(),
  196. );
  197. let tx_token = device.transmit(Instant::now()).unwrap();
  198. iface.inner.dispatch_ieee802154(
  199. Ieee802154Address::default(),
  200. tx_token,
  201. PacketMeta::default(),
  202. result.unwrap(),
  203. &mut iface.fragmenter,
  204. );
  205. assert_eq!(
  206. device.queue[0],
  207. &[
  208. 0x41, 0xcc, 0x3, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0,
  209. 0x0, 0x0, 0x0, 0x0, 0xc0, 0xb0, 0x5, 0x4e, 0x7a, 0x11, 0x3a, 0x92, 0xfc, 0x48, 0xc2,
  210. 0xa4, 0x41, 0xfc, 0x76, 0x40, 0x42, 0x42, 0x42, 0x42, 0x42, 0xb, 0x1a, 0x81, 0x0, 0x0,
  211. 0x0, 0x0, 0x27, 0x0, 0x2, 0xa2, 0xc2, 0x2d, 0x63, 0x0, 0x0, 0x0, 0x0, 0xd9, 0x5e, 0xc,
  212. 0x0, 0x0, 0x0, 0x0, 0x0, 0x10, 0x11, 0x12, 0x13, 0x14, 0x15, 0x16, 0x17, 0x18, 0x19,
  213. 0x1a, 0x1b, 0x1c, 0x1d, 0x1e, 0x1f, 0x20, 0x21, 0x22, 0x23, 0x24, 0x25, 0x26, 0x27,
  214. 0x28, 0x29, 0x2a, 0x2b, 0x2c, 0x2d, 0x2e, 0x2f, 0x30, 0x31, 0x32, 0x33, 0x34, 0x35,
  215. 0x36, 0x37, 0x38, 0x39, 0x3a, 0x3b, 0x3c, 0x3d, 0x3e, 0x3f, 0x40, 0x41, 0x42, 0x43,
  216. 0x44, 0x45, 0x46, 0x47,
  217. ]
  218. );
  219. iface.poll(Instant::now(), &mut device, &mut sockets);
  220. assert_eq!(
  221. device.queue[1],
  222. &[
  223. 0x41, 0xcc, 0x4, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0,
  224. 0x0, 0x0, 0x0, 0x0, 0xe0, 0xb0, 0x5, 0x4e, 0xf, 0x48, 0x49, 0x4a, 0x4b, 0x4c, 0x4d,
  225. 0x4e, 0x4f, 0x50, 0x51, 0x52, 0x53, 0x54, 0x55, 0x56, 0x57, 0x58, 0x59, 0x5a, 0x5b,
  226. 0x5c, 0x5d, 0x5e, 0x5f, 0x60, 0x61, 0x62, 0x63, 0x64, 0x65, 0x66, 0x67, 0x68, 0x69,
  227. 0x6a, 0x6b, 0x6c, 0x6d, 0x6e, 0x6f, 0x70, 0x71, 0x72, 0x73, 0x74, 0x75, 0x76, 0x77,
  228. 0x78, 0x79, 0x7a, 0x7b, 0x7c, 0x7d, 0x7e, 0x7f,
  229. ]
  230. );
  231. }
  232. #[test]
  233. #[cfg(feature = "proto-sixlowpan-fragmentation")]
  234. fn test_sixlowpan_udp_with_fragmentation() {
  235. use crate::phy::Checksum;
  236. let mut ieee802154_repr = Ieee802154Repr {
  237. frame_type: Ieee802154FrameType::Data,
  238. security_enabled: false,
  239. frame_pending: false,
  240. ack_request: false,
  241. sequence_number: Some(5),
  242. pan_id_compression: true,
  243. frame_version: Ieee802154FrameVersion::Ieee802154_2003,
  244. dst_pan_id: Some(Ieee802154Pan(0xbeef)),
  245. dst_addr: Some(Ieee802154Address::Extended([
  246. 0x90, 0xfc, 0x48, 0xc2, 0xa4, 0x41, 0xfc, 0x76,
  247. ])),
  248. src_pan_id: Some(Ieee802154Pan(0xbeef)),
  249. src_addr: Some(Ieee802154Address::Extended([
  250. 0x42, 0x42, 0x42, 0x42, 0x42, 0x42, 0x0b, 0x1a,
  251. ])),
  252. };
  253. let (mut iface, mut sockets, mut device) = setup(Medium::Ieee802154);
  254. iface.inner.caps.checksum.udp = Checksum::None;
  255. let udp_rx_buffer = udp::PacketBuffer::new(vec![udp::PacketMetadata::EMPTY], vec![0; 1024 * 4]);
  256. let udp_tx_buffer = udp::PacketBuffer::new(vec![udp::PacketMetadata::EMPTY], vec![0; 1024 * 4]);
  257. let udp_socket = udp::Socket::new(udp_rx_buffer, udp_tx_buffer);
  258. let udp_socket_handle = sockets.add(udp_socket);
  259. {
  260. let socket = sockets.get_mut::<udp::Socket>(udp_socket_handle);
  261. assert_eq!(socket.bind(6969), Ok(()));
  262. assert!(!socket.can_recv());
  263. assert!(socket.can_send());
  264. }
  265. let udp_first_part = &[
  266. 0xc0, 0xbc, 0x00, 0x92, 0x6e, 0x33, 0x07, 0xe7, 0xdc, 0xf0, 0xd3, 0xc9, 0x1b, 0x39, 0xbf,
  267. 0xa0, 0x4c, 0x6f, 0x72, 0x65, 0x6d, 0x20, 0x69, 0x70, 0x73, 0x75, 0x6d, 0x20, 0x64, 0x6f,
  268. 0x6c, 0x6f, 0x72, 0x20, 0x73, 0x69, 0x74, 0x20, 0x61, 0x6d, 0x65, 0x74, 0x2c, 0x20, 0x63,
  269. 0x6f, 0x6e, 0x73, 0x65, 0x63, 0x74, 0x65, 0x74, 0x75, 0x72, 0x20, 0x61, 0x64, 0x69, 0x70,
  270. 0x69, 0x73, 0x63, 0x69, 0x6e, 0x67, 0x20, 0x65, 0x6c, 0x69, 0x74, 0x2e, 0x20, 0x49, 0x6e,
  271. 0x20, 0x61, 0x74, 0x20, 0x72, 0x68, 0x6f, 0x6e, 0x63, 0x75, 0x73, 0x20, 0x74, 0x6f, 0x72,
  272. 0x74, 0x6f, 0x72, 0x2e, 0x20, 0x43, 0x72, 0x61, 0x73, 0x20, 0x62, 0x6c, 0x61, 0x6e,
  273. ];
  274. assert_eq!(
  275. iface.inner.process_sixlowpan(
  276. &mut sockets,
  277. PacketMeta::default(),
  278. &ieee802154_repr,
  279. udp_first_part,
  280. &mut iface.fragments
  281. ),
  282. None
  283. );
  284. ieee802154_repr.sequence_number = Some(6);
  285. let udp_second_part = &[
  286. 0xe0, 0xbc, 0x00, 0x92, 0x11, 0x64, 0x69, 0x74, 0x20, 0x74, 0x65, 0x6c, 0x6c, 0x75, 0x73,
  287. 0x20, 0x64, 0x69, 0x61, 0x6d, 0x2c, 0x20, 0x76, 0x61, 0x72, 0x69, 0x75, 0x73, 0x20, 0x76,
  288. 0x65, 0x73, 0x74, 0x69, 0x62, 0x75, 0x6c, 0x75, 0x6d, 0x20, 0x6e, 0x69, 0x62, 0x68, 0x20,
  289. 0x63, 0x6f, 0x6d, 0x6d, 0x6f, 0x64, 0x6f, 0x20, 0x6e, 0x65, 0x63, 0x2e,
  290. ];
  291. assert_eq!(
  292. iface.inner.process_sixlowpan(
  293. &mut sockets,
  294. PacketMeta::default(),
  295. &ieee802154_repr,
  296. udp_second_part,
  297. &mut iface.fragments
  298. ),
  299. None
  300. );
  301. let socket = sockets.get_mut::<udp::Socket>(udp_socket_handle);
  302. let udp_data = b"Lorem ipsum dolor sit amet, consectetur adipiscing elit. \
  303. In at rhoncus tortor. Cras blandit tellus diam, varius vestibulum nibh commodo nec.";
  304. assert_eq!(
  305. socket.recv(),
  306. Ok((
  307. &udp_data[..],
  308. IpEndpoint {
  309. addr: IpAddress::Ipv6(Ipv6Address([
  310. 0xfe, 0x80, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x40, 0x42, 0x42, 0x42, 0x42, 0x42,
  311. 0xb, 0x1a,
  312. ])),
  313. port: 54217,
  314. }
  315. .into()
  316. ))
  317. );
  318. let tx_token = device.transmit(Instant::now()).unwrap();
  319. iface.inner.dispatch_ieee802154(
  320. Ieee802154Address::default(),
  321. tx_token,
  322. PacketMeta::default(),
  323. IpPacket::new_ipv6(
  324. Ipv6Repr {
  325. src_addr: Ipv6Address::default(),
  326. dst_addr: Ipv6Address::default(),
  327. next_header: IpProtocol::Udp,
  328. payload_len: udp_data.len(),
  329. hop_limit: 64,
  330. },
  331. IpPayload::Udp(
  332. UdpRepr {
  333. src_port: 1234,
  334. dst_port: 1234,
  335. },
  336. udp_data,
  337. ),
  338. ),
  339. &mut iface.fragmenter,
  340. );
  341. iface.poll(Instant::now(), &mut device, &mut sockets);
  342. assert_eq!(
  343. device.queue[0],
  344. &[
  345. 0x41, 0xcc, 0x3, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0,
  346. 0x0, 0x0, 0x0, 0x0, 0xc0, 0xb4, 0x5, 0x4e, 0x7e, 0x40, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0,
  347. 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0xf0, 0x4, 0xd2, 0x4, 0xd2, 0xf6,
  348. 0x4d, 0x4c, 0x6f, 0x72, 0x65, 0x6d, 0x20, 0x69, 0x70, 0x73, 0x75, 0x6d, 0x20, 0x64,
  349. 0x6f, 0x6c, 0x6f, 0x72, 0x20, 0x73, 0x69, 0x74, 0x20, 0x61, 0x6d, 0x65, 0x74, 0x2c,
  350. 0x20, 0x63, 0x6f, 0x6e, 0x73, 0x65, 0x63, 0x74, 0x65, 0x74, 0x75, 0x72, 0x20, 0x61,
  351. 0x64, 0x69, 0x70, 0x69, 0x73, 0x63, 0x69, 0x6e, 0x67, 0x20, 0x65, 0x6c, 0x69, 0x74,
  352. 0x2e, 0x20, 0x49, 0x6e, 0x20, 0x61, 0x74, 0x20, 0x72, 0x68, 0x6f, 0x6e, 0x63, 0x75,
  353. 0x73, 0x20, 0x74,
  354. ]
  355. );
  356. assert_eq!(
  357. device.queue[1],
  358. &[
  359. 0x41, 0xcc, 0x4, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0,
  360. 0x0, 0x0, 0x0, 0x0, 0xe0, 0xb4, 0x5, 0x4e, 0xf, 0x6f, 0x72, 0x74, 0x6f, 0x72, 0x2e,
  361. 0x20, 0x43, 0x72, 0x61, 0x73, 0x20, 0x62, 0x6c, 0x61, 0x6e, 0x64, 0x69, 0x74, 0x20,
  362. 0x74, 0x65, 0x6c, 0x6c, 0x75, 0x73, 0x20, 0x64, 0x69, 0x61, 0x6d, 0x2c, 0x20, 0x76,
  363. 0x61, 0x72, 0x69, 0x75, 0x73, 0x20, 0x76, 0x65, 0x73, 0x74, 0x69, 0x62, 0x75, 0x6c,
  364. 0x75, 0x6d, 0x20, 0x6e, 0x69, 0x62, 0x68, 0x20, 0x63, 0x6f, 0x6d, 0x6d, 0x6f, 0x64,
  365. 0x6f, 0x20, 0x6e, 0x65, 0x63, 0x2e,
  366. ]
  367. );
  368. }