Compare commits
	
		
			1247 Commits
		
	
	
		
			embassy-fu
			...
			stm32-ring
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|  | b49bc449d7 | ||
|  | a21a2901ac | ||
|  | 374c92a4f0 | ||
|  | 433422b9f2 | ||
|  | a85b34c1fe | ||
|  | 1078f6f4e7 | ||
|  | 2bb6e93e86 | ||
|  | 2afa08c923 | ||
|  | a61701b756 | ||
|  | 7a36072a15 | ||
|  | a167c77d39 | ||
|  | 8839f3f62a | ||
|  | ac111f40d8 | ||
|  | 3229b5e809 | ||
|  | b2047c4351 | ||
|  | 849011b826 | ||
|  | 6cec6fa09b | ||
|  | 0d224a00e1 | ||
|  | 47ae9b7981 | ||
|  | 8e22d57447 | ||
|  | 5f99ccf54c | ||
|  | 54e695b1b2 | ||
|  | 8fc92fdf62 | ||
|  | c6424fdc11 | ||
|  | 3c31236c10 | ||
|  | 6096f0cf4b | ||
|  | a1d45303c3 | ||
|  | 7601779693 | ||
|  | 1806422763 | ||
|  | 00cde67abe | ||
|  | 96e8a7ddb9 | ||
|  | 25864ae4dc | ||
|  | 14e0090cb1 | ||
|  | 45843034ec | ||
|  | 7757405908 | ||
|  | fc268df6f5 | ||
|  | 4ea6662e55 | ||
|  | 49455792cb | ||
|  | 855c0d1423 | ||
|  | 05c36e05f9 | ||
|  | b58b9ff390 | ||
|  | 1d5adb8974 | ||
|  | be66e0f7ce | ||
|  | 861f49cfd4 | ||
|  | 7ab9fe0522 | ||
|  | 1c8492bab2 | ||
|  | 19588a9e6f | ||
|  | 1d2f6667df | ||
|  | ac0ea406f9 | ||
|  | 7336b8cd88 | ||
|  | bcbe3040a1 | ||
|  | f4ade6af8b | ||
|  | fa1ec29ae6 | ||
|  | 58e727d3b9 | ||
|  | 4cd5ed81aa | ||
|  | 4618b79b22 | ||
|  | db16b6ff3f | ||
|  | a9074fd09b | ||
|  | f2469776f4 | ||
|  | a10850a6da | ||
|  | ce04b732d1 | ||
|  | ff6748a0d8 | ||
|  | 7646f18836 | ||
|  | 41fe718ea8 | ||
|  | 94c6727b3f | ||
|  | b77794c9a7 | ||
|  | ba886b45b8 | ||
|  | 2119b8e1ca | ||
|  | 49bed094a3 | ||
|  | 49ecd8d7c5 | ||
|  | 29cc661dca | ||
|  | 3e730aa8b0 | ||
|  | 28a3454846 | ||
|  | 91cddd50f6 | ||
|  | 9d610c6866 | ||
|  | 03d6363d5a | ||
|  | d960bf344a | ||
|  | 31b54e0fbd | ||
|  | 3ba73b5ff4 | ||
|  | 8c733c29cc | ||
|  | 278818395e | ||
|  | 42a8f1671d | ||
|  | 1cf26f0eb3 | ||
|  | d91c37dae3 | ||
|  | 759d911b50 | ||
|  | a277deeaa5 | ||
|  | cb00fb18cb | ||
|  | f8b359dc5a | ||
|  | edef790e1a | ||
|  | 18af9f304a | ||
|  | deadf40c85 | ||
|  | 405649ddc7 | ||
|  | e2410cbb6a | ||
|  | 52decfb16c | ||
|  | 91047c61b9 | ||
|  | 054ca17f66 | ||
|  | 0d82ebea29 | ||
|  | f729d2d060 | ||
|  | 73f25093c7 | ||
|  | a3f727e2e1 | ||
|  | 0dea7b02d6 | ||
|  | a4866ad278 | ||
|  | d78edba0d4 | ||
|  | b283f213d9 | ||
|  | ba47fe9c41 | ||
|  | 8285263fc2 | ||
|  | cc5bca8e83 | ||
|  | 0a2f7b4661 | ||
|  | 02c86bca52 | ||
|  | fb27594b2e | ||
|  | a73f9474a0 | ||
|  | 3bf41e9a06 | ||
|  | 0e01b28d5e | ||
|  | 8aca324c2d | ||
|  | 7ee9e8322c | ||
|  | 43c20dbe65 | ||
|  | 8cd117fd5d | ||
|  | f64d1131b6 | ||
|  | b153a5b0d7 | ||
|  | 9b51c8f4d4 | ||
|  | 510ae7e3dc | ||
|  | f67eb84ec7 | ||
|  | 5de6bb3adf | ||
|  | 54fe50c685 | ||
|  | 837cdacd16 | ||
|  | 41e90e22e2 | ||
|  | 64b80c2e4d | ||
|  | 26f4d7d283 | ||
|  | fdd6e08ed6 | ||
|  | 7d64de153f | ||
|  | e24421a393 | ||
|  | 4de4039417 | ||
|  | f589247c1f | ||
|  | 37181c79d9 | ||
|  | a2ac1eed1b | ||
|  | bfa3cbaf30 | ||
|  | 216b120f15 | ||
|  | 08d9e5981e | ||
|  | 8a9136e4e4 | ||
|  | 3260f6b2af | ||
|  | 2080d8bb6d | ||
|  | a673b9aa29 | ||
|  | 38c5b97df0 | ||
|  | 21ea98810a | ||
|  | fbd6eeb748 | ||
|  | dbded7a6ce | ||
|  | efc70debb3 | ||
|  | 173c65b543 | ||
|  | a86a100879 | ||
|  | bba8b0ded5 | ||
|  | 095f5ef279 | ||
|  | 1c68c62ebd | ||
|  | f5216624bb | ||
|  | 46227bec1e | ||
|  | e63a34ba21 | ||
|  | 82dd7a5f8c | ||
|  | 0dfa192992 | ||
|  | e14fa11fc3 | ||
|  | df7ef1d98f | ||
|  | 9202dbf32a | ||
|  | 4044d728a6 | ||
|  | 6acc361109 | ||
|  | 27ec29e2c5 | ||
|  | 90c1422381 | ||
|  | 9e1ddeac86 | ||
|  | 99dcbf00c4 | ||
|  | 776e001b5b | ||
|  | 6ba2bb1a7f | ||
|  | a258e15c23 | ||
|  | 7a682ec02a | ||
|  | e9ede443bc | ||
|  | bc550cbfda | ||
|  | 8da9c07a65 | ||
|  | bd6bb2d248 | ||
|  | 81f10e136a | ||
|  | 1fdce6e52a | ||
|  | f395ec44e8 | ||
|  | 63941432e3 | ||
|  | be0f93ff37 | ||
|  | 224eaaf797 | ||
|  | f681b9d4e5 | ||
|  | 650589ab3f | ||
|  | b9fc2a6b33 | ||
|  | 3002ee0dcf | ||
|  | ce0e1a5db3 | ||
|  | a3ecf5caf6 | ||
|  | a7629299f4 | ||
|  | e7ff759f1c | ||
|  | 62ecd97350 | ||
|  | 577f060d24 | ||
|  | 6a6c673c5f | ||
|  | c1d5f86871 | ||
|  | 0289630fe4 | ||
|  | 5a03b2e9e8 | ||
|  | 4863f88d02 | ||
|  | ba8cafb20c | ||
|  | f3699e67b9 | ||
|  | 201a038134 | ||
|  | 8fd8ef9ca7 | ||
|  | 9a677ab618 | ||
|  | 5c42ca13bd | ||
|  | f426c47747 | ||
|  | 6e947c83b6 | ||
|  | e183801957 | ||
|  | 00258bca43 | ||
|  | 813bba200f | ||
|  | 32836129f6 | ||
|  | f06554cf1d | ||
|  | f5df567619 | ||
|  | 1b86570cfd | ||
|  | f51cbebffd | ||
|  | c309797488 | ||
|  | 35a34afbc2 | ||
|  | b150b9506b | ||
|  | ab6179fb07 | ||
|  | 636a3d05c2 | ||
|  | d3ce64254a | ||
|  | d8c92c53d6 | ||
|  | 1f25d2ba83 | ||
|  | cae683cd41 | ||
|  | 53c60df997 | ||
|  | 6760258ec3 | ||
|  | dbfd28130f | ||
|  | 4ef8e008e8 | ||
|  | 44b7fe45e2 | ||
|  | df17a88448 | ||
|  | dee8c71a2d | ||
|  | 1fbb8f0b32 | ||
|  | 047ea9066f | ||
|  | f38899728c | ||
|  | 8469a2409c | ||
|  | dee1d51ad3 | ||
|  | da8258b767 | ||
|  | 60809edf2c | ||
|  | be37eee13d | ||
|  | c9b7dbc541 | ||
|  | e2516bba09 | ||
|  | f3ec6080bf | ||
|  | 89279dcdc9 | ||
|  | d9696bd212 | ||
|  | 611d023829 | ||
|  | 9f28d80977 | ||
|  | 52cab3a9f4 | ||
|  | 31ef783ac1 | ||
|  | 9f1dac3f5d | ||
|  | 7677268319 | ||
|  | 0ef419bee4 | ||
|  | 92e96bd601 | ||
|  | 7c53ebd576 | ||
|  | 20e7b5e296 | ||
|  | 37d8f2e512 | ||
|  | efd9e18321 | ||
|  | f588105429 | ||
|  | 28b8ac4b62 | ||
|  | 511a951246 | ||
|  | a0b6096610 | ||
|  | 7e9e628eb9 | ||
|  | 5d9ae3dbdb | ||
|  | 8290236ed6 | ||
|  | eed2b12325 | ||
|  | 2a49e11cb0 | ||
|  | 57d3d4d581 | ||
|  | 95b31cf2db | ||
|  | 05b2b2fb5f | ||
|  | 7e5ead78fe | ||
|  | 2deb2c624c | ||
|  | d8e2f82569 | ||
|  | 043b3072c4 | ||
|  | 991b22b6a1 | ||
|  | e2e15e436a | ||
|  | 3deb65bc87 | ||
|  | 064ec9581e | ||
|  | 84bfe9b8c9 | ||
|  | a77ce1088d | ||
|  | e962fe794c | ||
|  | 78e6b4d261 | ||
|  | 53efb02900 | ||
|  | 6c93309df4 | ||
|  | 25577e0eaf | ||
|  | 9242ad89d4 | ||
|  | 8256ac1044 | ||
|  | 54b82d9966 | ||
|  | 5923e143e3 | ||
|  | 143105eeb6 | ||
|  | 3ede5667d4 | ||
|  | 7140e97202 | ||
|  | 803c09c300 | ||
|  | 5e19fb6fb9 | ||
|  | c38eb9660b | ||
|  | 7c6936a2e3 | ||
|  | c94f1e1450 | ||
|  | df3a1e1b9d | ||
|  | 36ad82a52b | ||
|  | 117fca84ea | ||
|  | ef361d2e88 | ||
|  | ae26a08026 | ||
|  | 1349dabe1a | ||
|  | 932b80ca8a | ||
|  | 4ce1c5f27d | ||
|  | 7c11d85e1e | ||
|  | 8aaffe82e7 | ||
|  | b1e2195b49 | ||
|  | 0909a6cd3f | ||
|  | bfebf7a436 | ||
|  | 5504fc54fe | ||
|  | d3c4e4a20a | ||
|  | 94890e544e | ||
|  | b41ee47115 | ||
|  | 08f911d25e | ||
|  | 7ef6a3cfb2 | ||
|  | cd2ed065dc | ||
|  | dd88775871 | ||
|  | e11eebfa57 | ||
|  | 268e29b153 | ||
|  | f78aa4f936 | ||
|  | 0e13fe9925 | ||
|  | 472dc6b7d1 | ||
|  | 50b0fb1a37 | ||
|  | cfbe93c280 | ||
|  | d9d6fd6d70 | ||
|  | 42931b51f2 | ||
|  | e3efda2249 | ||
|  | 373760a56b | ||
|  | 5955d81374 | ||
|  | 80972f1e0e | ||
|  | f3dcb5eb22 | ||
|  | 6b2aaacf83 | ||
|  | a78e10e003 | ||
|  | 02caec9482 | ||
|  | 760d4a72cb | ||
|  | e7129371d0 | ||
|  | e3c4e00be0 | ||
|  | 91d8afd371 | ||
|  | 89129babf9 | ||
|  | def576ac46 | ||
|  | ef1890e911 | ||
|  | 754bb802ba | ||
|  | 68c260edeb | ||
|  | fc8c83e00a | ||
|  | 87898501a2 | ||
|  | a0d089536a | ||
|  | 15e1747220 | ||
|  | 0bbc3a3d81 | ||
|  | 5a12fd6c75 | ||
|  | b7dfc8de10 | ||
|  | ddbd509865 | ||
|  | 69944675a3 | ||
|  | 4ee3d15519 | ||
|  | 6806bb9692 | ||
|  | d6ce1c4325 | ||
|  | 7a841b58d1 | ||
|  | 14f6bc88ea | ||
|  | 2d7f35cf57 | ||
|  | cf179f3076 | ||
|  | 20aa86d63e | ||
|  | a77fdefd7c | ||
|  | a6cef4baf2 | ||
|  | 6a802c4708 | ||
|  | 732614579b | ||
|  | a33774ec51 | ||
|  | 8a3a7c65a8 | ||
|  | 21400da073 | ||
|  | 805bca1f5a | ||
|  | 7186e03801 | ||
|  | 2c45b5c519 | ||
|  | 9c7b9b7848 | ||
|  | 7be63b3468 | ||
|  | e9a5b31fa8 | ||
|  | e8fc7a66a3 | ||
|  | bc69eb596e | ||
|  | 245147634b | ||
|  | 73ccc04231 | ||
|  | 47d5f127bb | ||
|  | 47e07584ca | ||
|  | c848bd9c9c | ||
|  | a8567f0617 | ||
|  | 99c4346579 | ||
|  | 7edd72f8f5 | ||
|  | 6c73b23f38 | ||
|  | 6b44027eab | ||
|  | cccceb88f2 | ||
|  | d8b265856f | ||
|  | cd2f28d2ab | ||
|  | 9939d43800 | ||
|  | 299689dfa2 | ||
|  | 88483b5abe | ||
|  | 04f90e3a9d | ||
|  | ba9afbc26d | ||
|  | aa77a06d58 | ||
|  | 425a35bf8a | ||
|  | ce7bd6955f | ||
|  | 41d558a5f4 | ||
|  | 7a4db1da26 | ||
|  | 0b49b588a2 | ||
|  | b6663a013f | ||
|  | 3e541c43e7 | ||
|  | fcd24adba9 | ||
|  | 4bfe624893 | ||
|  | 7be385dbb1 | ||
|  | f9c0c53e12 | ||
|  | 13f0c64a8c | ||
|  | 472df3fad6 | ||
|  | 2c9f289f40 | ||
|  | 46b437dea0 | ||
|  | e9a161b462 | ||
|  | 43462947ed | ||
|  | e73c6c9d90 | ||
|  | 89a371d10c | ||
|  | bce1ce7dcb | ||
|  | 12d6e37b3f | ||
|  | 055597063f | ||
|  | 79061021f9 | ||
|  | c9d89f391b | ||
|  | e484cb1b87 | ||
|  | 8fd30e407c | ||
|  | b2c6dc45e3 | ||
|  | 969e85150c | ||
|  | e7a19a9725 | ||
|  | a614e697d0 | ||
|  | 993875e11f | ||
|  | 2087561003 | ||
|  | b62e3e1d47 | ||
|  | 468c4266c8 | ||
|  | 18646c579c | ||
|  | b1bc034a7e | ||
|  | fded9fa52a | ||
|  | 6bf8d090a1 | ||
|  | 4054fb8779 | ||
|  | 935633c90b | ||
|  | bd4c4209af | ||
|  | 27e989afa9 | ||
|  | b0e26440ee | ||
|  | bc0cb43307 | ||
|  | c22218c72e | ||
|  | 18fe398673 | ||
|  | f5e09a8f4a | ||
|  | 5249996d28 | ||
|  | 5913553cb1 | ||
|  | 36319fc121 | ||
|  | 9e58d9274c | ||
|  | 9f5762d365 | ||
|  | a32e82029a | ||
|  | 2dc5608203 | ||
|  | d113fcfe32 | ||
|  | 96788ac93a | ||
|  | c66b28e759 | ||
|  | f8f1d3bcf0 | ||
|  | 34563b74aa | ||
|  | 63b75eaf64 | ||
|  | 9cf000ef4e | ||
|  | 42c13c8c3d | ||
|  | a054891263 | ||
|  | 403a83e08d | ||
|  | c88bbaa5ec | ||
|  | 6dfda69cc4 | ||
|  | bf013be9ba | ||
|  | d91efe3e62 | ||
|  | f7dfc49c5c | ||
|  | 8eb8ea6174 | ||
|  | 75f69803af | ||
|  | 1955a225e8 | ||
|  | 9eb65b11cb | ||
|  | 7650fea5f2 | ||
|  | 916f94b366 | ||
|  | ccc224c81f | ||
|  | bef559307c | ||
|  | 7b9075130e | ||
|  | 51478caad8 | ||
|  | 4314b823aa | ||
|  | 78d733fc73 | ||
|  | 7bdb3abad7 | ||
|  | 14ed0b90b8 | ||
|  | 64003cdfd6 | ||
|  | 8bd8fbd131 | ||
|  | 78974dfeb4 | ||
|  | 4ac257adb9 | ||
|  | f95aafc90e | ||
|  | c4f4aa10f9 | ||
|  | 206b4b597e | ||
|  | c0e40b887b | ||
|  | 351e4407ef | ||
|  | aabc275186 | ||
|  | 66ca57312f | ||
|  | 6dbb631f1e | ||
|  | 4dfa32b1e0 | ||
|  | 711ce10145 | ||
|  | b16b3b0dbb | ||
|  | 3c601bf8d2 | ||
|  | 485bb76e46 | ||
|  | d719f8bc03 | ||
|  | 4e212c7a0b | ||
|  | c1e93c0904 | ||
|  | 90f2939bf6 | ||
|  | 5cb0c8cc01 | ||
|  | 28b695e7c9 | ||
|  | 2331d58aa6 | ||
|  | 482ba835c4 | ||
|  | 7172dfd083 | ||
|  | 8fb380b180 | ||
|  | bc71230cd0 | ||
|  | 5367baa845 | ||
|  | d5f88e578c | ||
|  | 64247ae456 | ||
|  | 7be4337de9 | ||
|  | 2209bef4f2 | ||
|  | 43a4409405 | ||
|  | 3255e0a172 | ||
|  | 73ef85b765 | ||
|  | f0f92909c1 | ||
|  | 896764bb85 | ||
|  | 42462681bd | ||
|  | 4e884ee2d2 | ||
|  | dda5a4cc9d | ||
|  | 464faa2a04 | ||
|  | 035de6f3ff | ||
|  | f1a4db44c4 | ||
|  | ada3d5be7c | ||
|  | b05cd77a62 | ||
|  | 7fa478358a | ||
|  | 3f88bf6f9b | ||
|  | a2bd37be40 | ||
|  | 13328c58d3 | ||
|  | 1567e724f9 | ||
|  | 4ad255b34b | ||
|  | 4fd59f26fb | ||
|  | 272982ee54 | ||
|  | a53f525f51 | ||
|  | 7783e0ebb1 | ||
|  | e641db1f75 | ||
|  | e3f8020c3b | ||
|  | dfc58ad3a2 | ||
|  | 1626a4a74b | ||
|  | 5e74926907 | ||
|  | 218b44652c | ||
|  | 41a563aae3 | ||
|  | 363054de98 | ||
|  | 06abde8676 | ||
|  | 4e15043fc2 | ||
|  | 951f208915 | ||
|  | 1e36c91bf8 | ||
|  | 80b7c3cf69 | ||
|  | d159a6c62d | ||
|  | d21643c060 | ||
|  | e1eac15c42 | ||
|  | 76642b3a3c | ||
|  | 20c1dd112c | ||
|  | 4c4e923e05 | ||
|  | a509af4bc0 | ||
|  | a2b8921ff3 | ||
|  | 128a453163 | ||
|  | a7d3ef9122 | ||
|  | 32c3725631 | ||
|  | 48dff04d64 | ||
|  | 472473d8c1 | ||
|  | 6e68353a93 | ||
|  | 7ae47cb1d8 | ||
|  | cd440a49d6 | ||
|  | 614740a1b2 | ||
|  | c203cefe01 | ||
|  | 9cfea693ed | ||
|  | 023b0d5b22 | ||
|  | bd7b3bd455 | ||
|  | a4371e9544 | ||
|  | e1a0df7d46 | ||
|  | 2b6654541d | ||
|  | 43d018b67f | ||
|  | ab4b3fa96d | ||
|  | 26474ce6eb | ||
|  | 5edb3052e6 | ||
|  | 13666c9f12 | ||
|  | ac3e225988 | ||
|  | bab4277a86 | ||
|  | da6b1e8399 | ||
|  | 9d637070a5 | ||
|  | 86487db5d1 | ||
|  | 3af991ab63 | ||
|  | 1d841cc8ac | ||
|  | 4a224efe75 | ||
|  | c4a2c62096 | ||
|  | 366fab5b87 | ||
|  | aa21aebb0b | ||
|  | 9f9230ae7a | ||
|  | b9ecdb72bb | ||
|  | 617b0a03b9 | ||
|  | f5ff3c4ac3 | ||
|  | a7fa7d0de2 | ||
|  | 102b2e52cb | ||
|  | 7b11e339bd | ||
|  | dadd6aafe9 | ||
|  | 1b6aae9dde | ||
|  | e4dc473e04 | ||
|  | 562432ad8b | ||
|  | 494a76a0f1 | ||
|  | 36ca18132d | ||
|  | 218f8e0490 | ||
|  | ba18656e94 | ||
|  | c8a7b74bc2 | ||
|  | e3174d7a99 | ||
|  | fda36fd81b | ||
|  | 64ebb9b7fe | ||
|  | 7d8e6649b7 | ||
|  | 662a02a557 | ||
|  | 0bb6000e5c | ||
|  | a432d91d82 | ||
|  | 9af25c3396 | ||
|  | ef4a20f67b | ||
|  | c4cbb89fcd | ||
|  | 95cff35a91 | ||
|  | cb88dd285d | ||
|  | 791fbb3ca0 | ||
|  | 4a8e9cf4d9 | ||
|  | fb1946be7f | ||
|  | a697f1517a | ||
|  | 465e4c8b19 | ||
|  | 594969f281 | ||
|  | b5cf332cc0 | ||
|  | ca10fe7135 | ||
|  | 4c19464548 | ||
|  | 768fe699cf | ||
|  | c21cc21c62 | ||
|  | b6ca6d699a | ||
|  | 48e1aab762 | ||
|  | 7e251a2550 | ||
|  | e453334870 | ||
|  | c9e2cd6dd4 | ||
|  | 7ec15f2def | ||
|  | 1e60c60afd | ||
|  | 34b67fe137 | ||
|  | ffa75e1e39 | ||
|  | 2a0ea52878 | ||
|  | 5ee8626d72 | ||
|  | 32bdc54ccb | ||
|  | f38d54a6a6 | ||
|  | 4dadfb41ea | ||
|  | 64e610fef7 | ||
|  | f98ba4ebac | ||
|  | 5e3c33b777 | ||
|  | 6ad2bcf97a | ||
|  | 0412d1922c | ||
|  | 825b67101b | ||
|  | f604153f05 | ||
|  | 539a8107e2 | ||
|  | 78c2c1709b | ||
|  | fe15a7beee | ||
|  | 570ffab670 | ||
|  | 8f4fae9b36 | ||
|  | 2eae12b7f1 | ||
|  | 65ab714fae | ||
|  | 83af513424 | ||
|  | f0ae1f9133 | ||
|  | db8e9efe73 | ||
|  | 6ab4ecaf83 | ||
|  | b1203bf036 | ||
|  | 7d34f4f538 | ||
|  | f07e59b24a | ||
|  | d2f2b451d0 | ||
|  | 15e3f42b7c | ||
|  | 355761fd68 | ||
|  | 2a349afea7 | ||
|  | aea5a0fd96 | ||
|  | 634704bff4 | ||
|  | 868d01889b | ||
|  | 16590732f8 | ||
|  | 816b214403 | ||
|  | b6c8505697 | ||
|  | 7ecb05ff77 | ||
|  | b0c8c688c7 | ||
|  | b0529bc943 | ||
|  | 88fd521b01 | ||
|  | 1af102a1aa | ||
|  | 041531c829 | ||
|  | 0feecd5cde | ||
|  | 065a0a1ee7 | ||
|  | ce842fe28c | ||
|  | 96b97c4711 | ||
|  | dbf7493708 | ||
|  | 2baebabf4d | ||
|  | 0a27b6cedb | ||
|  | 401185b1d9 | ||
|  | f8afc3c882 | ||
|  | 4c4b47f78a | ||
|  | 539f97da53 | ||
|  | 1096a9746c | ||
|  | 3fbedd7c09 | ||
|  | 3c537a9fae | ||
|  | b72da125eb | ||
|  | 0ecc54f58c | ||
|  | f339e8518f | ||
|  | 6d4c6e0481 | ||
|  | 840a75674b | ||
|  | a24037edf9 | ||
|  | 68c186309f | ||
|  | 2332d8cd23 | ||
|  | 5aa59e9737 | ||
|  | bf4c0de16a | ||
|  | 8497f98de2 | ||
|  | 6e6c3cbebc | ||
|  | 35afb60dd4 | ||
|  | 651eec0242 | ||
|  | 413f339489 | ||
|  | 9428c40c8d | ||
|  | 0aa2a9ac27 | ||
|  | a6b52bde58 | ||
|  | 42f1b3ac74 | ||
|  | 72bb4f8798 | ||
|  | 3478004b4d | ||
|  | 2c8080b0ae | ||
|  | 7add0eafb8 | ||
|  | e4f457646f | ||
|  | 3afb62d8d6 | ||
|  | 771806be79 | ||
|  | 4a4b593694 | ||
|  | 4297eb27ff | ||
|  | 41d6316984 | ||
|  | 4e0d563997 | ||
|  | 97f9f248f4 | ||
|  | 147609d3bd | ||
|  | 007246b160 | ||
|  | 1f033d509a | ||
|  | 639b3f1d5b | ||
|  | 5655c6093f | ||
|  | 72bb9b53a2 | ||
|  | c29657f95a | ||
|  | f2fb9a2ca6 | ||
|  | 056eac998a | ||
|  | d1dd66cfca | ||
|  | e090ab1915 | ||
|  | eaad0cc1dc | ||
|  | 67a6e5accf | ||
|  | 787745188c | ||
|  | 74fdd4c03c | ||
|  | 10c9cc31b1 | ||
|  | cd9a65ba39 | ||
|  | 40ef66cdfb | ||
|  | 47a0769fc2 | ||
|  | e9a2c4a9e3 | ||
|  | 662bb5797f | ||
|  | 2457fcaa35 | ||
|  | da9ee83756 | ||
|  | aa92ce6dc7 | ||
|  | 1bd6c954c2 | ||
|  | 55d9af71e3 | ||
|  | 395b5fed64 | ||
|  | 0db3837dcc | ||
|  | c0f3610581 | ||
|  | 63122f6d7e | ||
|  | 46fd82d184 | ||
|  | 5ae91ed3b6 | ||
|  | 849a0e174f | ||
|  | e221d91330 | ||
|  | 5b72410828 | ||
|  | feaeb533fb | ||
|  | ebc735008f | ||
|  | bffa5be2f4 | ||
|  | 5b65b0e843 | ||
|  | 790e4e1594 | ||
|  | 5eae295c8a | ||
|  | 3005ee0178 | ||
|  | 8f30652109 | ||
|  | e9219405ca | ||
|  | aaaf5f23a8 | ||
|  | ac74613b5a | ||
|  | 47747d3b73 | ||
|  | c4d8f3579e | ||
|  | 731eb3c6e3 | ||
|  | 3d68c0400b | ||
|  | 13d9d8fde1 | ||
|  | 36639e5262 | ||
|  | 4fbd03a908 | ||
|  | aea28c8aa0 | ||
|  | 95fdc7c552 | ||
|  | eb1d2e1295 | ||
|  | 96d6c7243b | ||
|  | d8821cfd41 | ||
|  | cc0248d83a | ||
|  | 34eaade14f | ||
|  | 1ee58492fb | ||
|  | 5d4f09156a | ||
|  | 488e322478 | ||
|  | cd59046e6c | ||
|  | 35db6e639b | ||
|  | e94ca0efd6 | ||
|  | 236d104844 | ||
|  | f22297e3d6 | ||
|  | 1d2f97b4e2 | ||
|  | 58ab829049 | ||
|  | 5fdd521a76 | ||
|  | 1b8c0733e6 | ||
|  | 94010d3362 | ||
|  | f7fe0c1441 | ||
|  | 7bda01ec24 | ||
|  | 40f0272dd0 | ||
|  | 54c153673d | ||
|  | 7cbc3aefe6 | ||
|  | ef2b83cc03 | ||
|  | 5e94b8060b | ||
|  | a2712caab1 | ||
|  | 02abe00439 | ||
|  | bb89a2341c | ||
|  | f109e73c6d | ||
|  | 9f854110f2 | ||
|  | d8ea297d6a | ||
|  | e1d7d8d841 | ||
|  | eb010fbe33 | ||
|  | 71df28e269 | ||
|  | 645fb66a51 | ||
|  | e0ea5dfdb2 | ||
|  | 8436c6180f | ||
|  | 1dcb0ea1f5 | ||
|  | 902586a019 | ||
|  | 3135ad016d | ||
|  | 199504be56 | ||
|  | 4cc0463123 | ||
|  | 2a35a09444 | ||
|  | cea29d7de3 | ||
|  | 787e5d4907 | ||
|  | aedcc472c9 | ||
|  | 4d84b5469e | ||
|  | 3ca14ba4e9 | ||
|  | d438d1b685 | ||
|  | 6b8ab32536 | ||
|  | 805b885de6 | ||
|  | 7b838d0336 | ||
|  | fa37452359 | ||
|  | 1e2fb0459d | ||
|  | 83c2f8f416 | ||
|  | 09077f133d | ||
|  | c7be481190 | ||
|  | 89821846d7 | ||
|  | 758f5d7ea2 | ||
|  | f0ba22fc17 | ||
|  | db7e153fc0 | ||
|  | a4f9e7cbcc | ||
|  | de95ab264d | ||
|  | 04a7d97673 | ||
|  | 5aad2129ef | ||
|  | 50c5cc5db6 | ||
|  | b76631bebe | ||
|  | eae67d0be8 | ||
|  | 28991d7794 | ||
|  | 2fa2c1a6fe | ||
|  | 83b199a874 | ||
|  | cf900a8a3f | ||
|  | 4f2f375777 | ||
|  | e7c876d744 | ||
|  | 64c2e1b9b6 | ||
|  | 61be0e75c8 | ||
|  | a074cd0625 | ||
|  | ca4f615b25 | ||
|  | 536b6a2de5 | ||
|  | 51233c0357 | ||
|  | 5c52d6c217 | ||
|  | f474817872 | ||
|  | 97cb95bbf4 | ||
|  | 99c561a749 | ||
|  | f09745dfe1 | ||
|  | da9f82f507 | ||
|  | f13639e78c | ||
|  | 908eef2775 | ||
|  | 633ffe46ae | ||
|  | e6b9722a31 | ||
|  | b8f51c6496 | ||
|  | 33ee48b9e8 | ||
|  | a6d941fac3 | ||
|  | 15b4ed2c67 | ||
|  | bbfb786139 | ||
|  | 81dc532d2d | ||
|  | 06fb3e4251 | ||
|  | 4943dec1a7 | ||
|  | 15a93246d6 | ||
|  | f5391efe22 | ||
|  | 64e8cfef8e | ||
|  | 16838f8a66 | ||
|  | 6b88057aef | ||
|  | a444a65ebf | ||
|  | 1ed260b105 | ||
|  | 9f870a5edf | ||
|  | eb149a0bd4 | ||
|  | 551b54ddcb | ||
|  | 2528f45138 | ||
|  | 9505a6f752 | ||
|  | ea61c19280 | ||
|  | bcec55464f | ||
|  | 0b066b22d1 | ||
|  | 3a1ddd66c6 | ||
|  | 8d2d5a30a5 | ||
|  | 43c1afb6a6 | ||
|  | eba42cb5f4 | ||
|  | 4fe834db2f | ||
|  | 5cfad3f853 | ||
|  | 17857bc18f | ||
|  | dca11095e2 | ||
|  | d2e8794f29 | ||
|  | 122a31d208 | ||
|  | e70ae71ecc | ||
|  | 10e3c3f2ec | ||
|  | d05979c708 | ||
|  | 4a2e810485 | ||
|  | 6e1120e17e | ||
|  | 99682d313b | ||
|  | cbc97758e3 | ||
|  | dbe97b4098 | ||
|  | f22f36f51b | ||
|  | 5a64bf651c | ||
|  | 356beabc3b | ||
|  | 3760b60db3 | ||
|  | cecd77938c | ||
|  | 059610a8de | ||
|  | a3a58e8e4a | ||
|  | bd5ef80bec | ||
|  | c53614f057 | ||
|  | 1365ce6ab8 | ||
|  | 14a2d15240 | ||
|  | af34fc4ccc | ||
|  | aecfce1159 | ||
|  | 207fa19551 | ||
|  | 7da18e194a | ||
|  | a3e8a6bc3a | ||
|  | 1920e90dcd | ||
|  | b99533607c | ||
|  | ea4d08b6cf | ||
|  | 05968bf0f3 | ||
|  | fc086fd4ba | ||
|  | ea702b3719 | ||
|  | 97d18c5ffb | ||
|  | eed34f945c | ||
|  | 0b2d6996e8 | ||
|  | e7fdd500d8 | ||
|  | 1f246d0e37 | ||
|  | a7d5c87049 | ||
|  | 49e1091309 | ||
|  | 79b49c6fae | ||
|  | f053bf742c | ||
|  | 9423987ac5 | ||
|  | bc21b6efaf | ||
|  | a7b90c7fb6 | ||
|  | 4e61d83555 | ||
|  | c871fe0848 | ||
|  | 3c6c382465 | ||
|  | 171b764d82 | ||
|  | 08c8022583 | ||
|  | 4f2dcca34b | ||
|  | 9c30d565b9 | ||
|  | f363f6ce92 | ||
|  | 6bf24b4d1a | ||
|  | 88bbc238b7 | ||
|  | 2cfe2439c9 | ||
|  | 7b38b95e10 | ||
|  | 5142674786 | ||
|  | a5b1d2237f | ||
|  | 61560e740d | ||
|  | d2246ae693 | ||
|  | 7f499f3edc | ||
|  | 01e23bf9dd | ||
|  | e5097a8866 | ||
|  | f9da6271ce | ||
|  | 4976cbbe60 | ||
|  | 9b86de770b | ||
|  | 9cac649fcf | ||
|  | ff76fde299 | ||
|  | 560eecdb73 | ||
|  | ac6995f9e6 | ||
|  | eeb072d9cb | ||
|  | 1669e39565 | ||
|  | 80e58426fc | ||
|  | 1b249ca72d | ||
|  | 66611a80ca | ||
|  | d1eee52625 | ||
|  | 71cc6833e1 | ||
|  | 0c9ec8dc36 | ||
|  | ea868920e6 | ||
|  | 7a6732adcf | ||
|  | 52c03cf0a4 | ||
|  | ac61e0ee9f | ||
|  | 33f75419e5 | ||
|  | 6062978d58 | ||
|  | ca8afacfd0 | ||
|  | 9ad7e85288 | ||
|  | ad0eb3f4bd | ||
|  | 8d809c96ec | ||
|  | 9b209ffe1c | ||
|  | 5f02bee388 | ||
|  | 1bed02296c | ||
|  | 516f4ce946 | ||
|  | 545cc9326b | ||
|  | d99841fea9 | ||
|  | e3cf4255c6 | ||
|  | 4ce4131f8b | ||
|  | f78c706b89 | ||
|  | 4d5550070f | ||
|  | 53608a87ac | ||
|  | ba6e452cc5 | ||
|  | c2404ee8ca | ||
|  | ce1cba761c | ||
|  | 495ca6108c | ||
|  | 73d06dd67b | ||
|  | 866a42f3ae | ||
|  | f45d34ce7c | ||
|  | bf0ad38640 | ||
|  | de103a5f4f | ||
|  | 8c42b26fc6 | ||
|  | fa495b8e88 | ||
|  | d9c773f475 | ||
|  | a669f4cfd8 | ||
|  | 9d2641f2f5 | ||
|  | 6c5d81ada5 | ||
|  | 18453ee64c | ||
|  | 02a3cdb507 | ||
|  | e4c2b2aa9a | ||
|  | a4afab4640 | ||
|  | b7d0944265 | ||
|  | 1559374a19 | ||
|  | 5846b4ff7d | ||
|  | 83fcc360fe | ||
|  | 86113e199f | ||
|  | 9223b67306 | ||
|  | 71a56292d6 | ||
|  | 4da6320e63 | ||
|  | aff265a7f5 | ||
|  | 79cee74151 | ||
|  | 327d3cf0df | ||
|  | 79ba20d315 | ||
|  | 9d5b524bb0 | ||
|  | ef533e6df4 | ||
|  | e1faf88607 | ||
|  | 3d0ba58b2d | ||
|  | f554962f54 | ||
|  | aa8ba2115c | ||
|  | 322cfafed3 | ||
|  | df7174ecb0 | ||
|  | f8fd6ab208 | ||
|  | 6718ca3a94 | ||
|  | 9dca368c3d | ||
|  | d49d1b6b1c | ||
|  | 1b9479197d | ||
|  | 530182d668 | ||
|  | 94606833aa | ||
|  | 59765590e0 | ||
|  | 4fd831e4a8 | ||
|  | cae8499179 | ||
|  | f075e62444 | ||
|  | e8bb8faa23 | ||
|  | 63f5602111 | ||
|  | 753781a263 | ||
|  | 73208d5248 | ||
|  | 09afece93d | ||
|  | 1ee4bb22de | ||
|  | 5e2c52ee5b | ||
|  | 72b645b0c9 | ||
|  | 8d38eacae4 | ||
|  | 90d392205f | ||
|  | c96581879c | ||
|  | d5abd32da2 | ||
|  | 9f77dbf5ae | ||
|  | aabc02506b | ||
|  | 62c0b18f10 | ||
|  | 99284b8304 | ||
|  | a283c47557 | ||
|  | a7fdeac560 | ||
|  | 88a3c360e8 | ||
|  | 1d6f5493e7 | ||
|  | dab1762709 | ||
|  | ebf5a92ab2 | ||
|  | 874384826d | ||
|  | f4ebc36b63 | ||
|  | 38faae26e5 | ||
|  | a77e2c3512 | ||
|  | 8b9f4ad259 | ||
|  | 72c2e985bb | ||
|  | 7152031229 | ||
|  | 7ee7109508 | ||
|  | f9c62d4f1d | ||
|  | dc90006982 | ||
|  | a83560c6b1 | ||
|  | bb84d7a0ae | ||
|  | 526e90d3f3 | ||
|  | 77ece3f903 | ||
|  | d7f7614b22 | ||
|  | 823bd714fb | ||
|  | a89a0c2f12 | ||
|  | 9bb43ffe9a | ||
|  | bf1da0497c | ||
|  | 44c46e3c93 | ||
|  | b0d91e9f31 | ||
|  | 53c34ccc39 | ||
|  | be68d8ebb7 | ||
|  | 603513e76e | ||
|  | bcd3ab4ba1 | ||
|  | 820e6462b6 | ||
|  | 5c882cf4fa | ||
|  | 17d8d11f73 | ||
|  | 82d4360756 | ||
|  | e129a97d48 | ||
|  | 93354b812c | ||
|  | 65907204d6 | ||
|  | cd539ba3a0 | ||
|  | 86fd480672 | ||
|  | de0070948c | ||
|  | 7bbb4c22a1 | ||
|  | 5bf6564e95 | ||
|  | c863acd24f | ||
|  | f76444bdc4 | ||
|  | b3dfd06dd6 | ||
|  | 1db9e464ff | ||
|  | d6af0f6286 | ||
|  | f2239d34cc | ||
|  | ee76831f93 | ||
|  | 75e93cc142 | ||
|  | 049c31613b | ||
|  | 1e95c4fcff | ||
|  | daf2744716 | ||
|  | 49070c75b6 | ||
|  | f27a47a37b | ||
|  | f4f5824972 | ||
|  | 7f7c14b7bc | ||
|  | dc376a2390 | ||
|  | fa7781c48d | ||
|  | a9efbf18c6 | ||
|  | 3c06a18b94 | ||
|  | 6fa74b0c02 | ||
|  | b2a327a858 | ||
|  | 7f16b1cd23 | ||
|  | b743d9f48c | ||
|  | 3c24ad2db6 | ||
|  | a226e86503 | ||
|  | c5ce02b30e | ||
|  | 8536666148 | ||
|  | ca92302d03 | ||
|  | a45fb2d718 | ||
|  | eeb1515e9f | ||
|  | b4f2c2a05e | ||
|  | 18dc0dea63 | ||
|  | 9d674f0212 | ||
|  | 816778e3fa | ||
|  | 4f33cc5d1a | ||
|  | 2fed9f949a | ||
|  | 7412a859fd | ||
|  | 0db1332da8 | ||
|  | 334dfcdb65 | ||
|  | 54ba472540 | ||
|  | 4322293f63 | ||
|  | c14527486d | ||
|  | 81298394b5 | ||
|  | 5d1576ea73 | ||
|  | f46b838746 | ||
|  | 3f672c8a93 | ||
|  | e5af4c4bce | ||
|  | 7bb9620b1a | ||
|  | 2e7916c5fe | ||
|  | dacbc9acd5 | ||
|  | 10d1ad2343 | ||
|  | a0487380da | ||
|  | c1e25067da | ||
|  | 897b72c872 | ||
|  | 5914d80968 | ||
|  | 5f7e0eb2ae | ||
|  | 15b4f9db90 | ||
|  | 0f55f5a73d | ||
|  | 3d708a459c | ||
|  | 3b58ac1bf8 | ||
|  | 1d3e41f970 | ||
|  | 44d7a84e47 | ||
|  | b418c0e4d6 | ||
|  | 11da25800b | ||
|  | d0fe654c82 | ||
|  | 6663390224 | ||
|  | ac13675f3a | ||
|  | 0c6933fefb | ||
|  | 295cc997ae | ||
|  | ab1a6889a6 | ||
|  | 336ebe54c0 | ||
|  | 1c657d2d55 | ||
|  | c495c765df | ||
|  | feb840c503 | ||
|  | feead3ae89 | ||
|  | f7267d493f | ||
|  | ec10460547 | ||
|  | 79654510b7 | ||
|  | 70a3b85acc | ||
|  | 9794bc59cc | ||
|  | c4d5c047d7 | ||
|  | 809a4a127b | ||
|  | ea5f2c71e0 | ||
|  | b2d0f8d590 | ||
|  | 31d85da78a | ||
|  | 9611e7c9f2 | ||
|  | 573c433f64 | ||
|  | 34ed3441ce | ||
|  | 5886679006 | ||
|  | 7004b095c3 | ||
|  | 22c32b5d5c | ||
|  | 107bb0946a | ||
|  | f66f20b1ce | ||
|  | 6264fe39a5 | ||
|  | 7d5c1fcebf | ||
|  | 6cdff72d6d | ||
|  | 96eb669b34 | ||
|  | 506e5a4493 | ||
|  | d6e8a11ea7 | ||
|  | 50af13d470 | ||
|  | 3aa0c13ba5 | ||
|  | 8b464d2668 | ||
|  | 3ca7314476 | ||
|  | 5327b9c289 | ||
|  | 835b69456d | ||
|  | efe456ab14 | ||
|  | 9ff5c50774 | ||
|  | 71c130488b | ||
|  | e2181cb439 | ||
|  | b934f3f12e | ||
|  | 3fce6ec649 | ||
|  | 27905f1be1 | ||
|  | 7954cbc4e7 | ||
|  | 99dd2a9386 | ||
|  | 6d347af9fa | ||
|  | c8ecc55710 | ||
|  | 44150c4830 | ||
|  | 07c64d902e | ||
|  | e7d4bf258a | ||
|  | 838f3065ea | ||
|  | 1fb6bfbec9 | ||
|  | 8ba421f324 | ||
|  | 38900a7fb0 | ||
|  | fe08bdf0d8 | ||
|  | 30641d0564 | ||
|  | 464ae67108 | ||
|  | 171077bacf | ||
|  | b2720117c4 | ||
|  | 4781feafc4 | ||
|  | dcd8c62169 | ||
|  | c0b7fd910e | ||
|  | 47069dfbe1 | ||
|  | c30b38586a | ||
|  | 2636a8dc2e | ||
|  | 6dab322c58 | ||
|  | 5d114479ff | ||
|  | 1f36da5ca6 | ||
|  | af845b7d44 | ||
|  | 308ca4b8e3 | ||
|  | 60ca5e8479 | ||
|  | 84240d49ea | ||
|  | f31116cafa | ||
|  | 69d80c086d | ||
|  | 6ee29ff0bd | ||
|  | 8e8106ef55 | ||
|  | 61c666212f | ||
|  | 9a873d1dbf | 
							
								
								
									
										87
									
								
								.github/workflows/doc.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										87
									
								
								.github/workflows/doc.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,87 @@ | ||||
| name: Docs | ||||
|  | ||||
| on: | ||||
|   push: | ||||
|     branches: [master] | ||||
|  | ||||
| env: | ||||
|   BUILDER_THREADS: '1' | ||||
|  | ||||
| jobs: | ||||
|   doc: | ||||
|     runs-on: ubuntu-latest | ||||
|  | ||||
|     # Since stm32 crates take SO LONG to build, we split them | ||||
|     # into a separate job. This way it doesn't slow down updating | ||||
|     # the rest. | ||||
|     strategy: | ||||
|       matrix: | ||||
|         crates: | ||||
|           #- stm32  # runs out of disk space... | ||||
|           - rest | ||||
|  | ||||
|     # This will ensure at most one doc build job is running at a time | ||||
|     # (for stm32 and non-stm32 independently). | ||||
|     # If another job is already running, the new job will wait. | ||||
|     # If another job is already waiting, it'll be canceled. | ||||
|     # This means some commits will be skipped, but that's fine because | ||||
|     # we only care that the latest gets built. | ||||
|     concurrency: doc-${{ matrix.crates }} | ||||
|  | ||||
|     steps: | ||||
|       - uses: actions/checkout@v2 | ||||
|         with: | ||||
|           submodules: true | ||||
|       - name: Install Rust targets | ||||
|         run: | | ||||
|           rustup target add x86_64-unknown-linux-gnu | ||||
|           rustup target add wasm32-unknown-unknown | ||||
|           rustup target add thumbv6m-none-eabi | ||||
|           rustup target add thumbv7m-none-eabi | ||||
|           rustup target add thumbv7em-none-eabi | ||||
|           rustup target add thumbv7em-none-eabihf | ||||
|           rustup target add thumbv8m.base-none-eabi | ||||
|           rustup target add thumbv8m.main-none-eabi | ||||
|           rustup target add thumbv8m.main-none-eabihf | ||||
|  | ||||
|       - name: Install docserver | ||||
|         run: | | ||||
|           wget -q -O /usr/local/bin/builder "https://github.com/embassy-rs/docserver/releases/download/v0.4/builder" | ||||
|           chmod +x /usr/local/bin/builder | ||||
|  | ||||
|       - name: build-stm32 | ||||
|         if: ${{ matrix.crates=='stm32' }} | ||||
|         run: | | ||||
|           mkdir crates | ||||
|           builder ./embassy-stm32 crates/embassy-stm32/git.zup | ||||
|  | ||||
|       - name: build-rest | ||||
|         if: ${{ matrix.crates=='rest' }} | ||||
|         run: | | ||||
|           mkdir crates | ||||
|           builder ./embassy-boot/boot crates/embassy-boot/git.zup | ||||
|           builder ./embassy-boot/nrf crates/embassy-boot-nrf/git.zup | ||||
|           builder ./embassy-boot/rp crates/embassy-boot-rp/git.zup | ||||
|           builder ./embassy-boot/stm32 crates/embassy-boot-stm32/git.zup | ||||
|           builder ./embassy-cortex-m crates/embassy-cortex-m/git.zup | ||||
|           builder ./embassy-embedded-hal crates/embassy-embedded-hal/git.zup | ||||
|           builder ./embassy-executor crates/embassy-executor/git.zup | ||||
|           builder ./embassy-futures crates/embassy-futures/git.zup | ||||
|           builder ./embassy-lora crates/embassy-lora/git.zup | ||||
|           builder ./embassy-net crates/embassy-net/git.zup | ||||
|           builder ./embassy-net-driver crates/embassy-net-driver/git.zup | ||||
|           builder ./embassy-net-driver-channel crates/embassy-net-driver-channel/git.zup | ||||
|           builder ./embassy-nrf crates/embassy-nrf/git.zup | ||||
|           builder ./embassy-rp crates/embassy-rp/git.zup | ||||
|           builder ./embassy-sync crates/embassy-sync/git.zup | ||||
|           builder ./embassy-time crates/embassy-time/git.zup | ||||
|           builder ./embassy-usb crates/embassy-usb/git.zup | ||||
|           builder ./embassy-usb-driver crates/embassy-usb-driver/git.zup | ||||
|           builder ./embassy-usb-logger crates/embassy-usb-logger/git.zup | ||||
|  | ||||
|       - name: upload | ||||
|         run: | | ||||
|           mkdir -p ~/.kube | ||||
|           echo "${{secrets.KUBECONFIG}}" > ~/.kube/config | ||||
|           POD=$(kubectl -n embassy get po -l app=docserver -o jsonpath={.items[0].metadata.name}) | ||||
|           kubectl cp crates $POD:/data | ||||
							
								
								
									
										12
									
								
								.github/workflows/rust.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										12
									
								
								.github/workflows/rust.yml
									
									
									
									
										vendored
									
									
								
							| @@ -11,7 +11,7 @@ env: | ||||
|  | ||||
| jobs: | ||||
|   all: | ||||
|     runs-on: ubuntu-20.04 | ||||
|     runs-on: ubuntu-latest | ||||
|     needs: [build-nightly, build-stable, test] | ||||
|     steps: | ||||
|       - name: Done | ||||
| @@ -68,5 +68,11 @@ jobs: | ||||
|     runs-on: ubuntu-latest | ||||
|     steps: | ||||
|       - uses: actions/checkout@v2 | ||||
|       - name: Test | ||||
|         run: cd embassy-sync && cargo test | ||||
|  | ||||
|       - name: Test boot | ||||
|         working-directory: ./embassy-boot/boot | ||||
|         run: cargo test && cargo test --features nightly && cargo test --features "ed25519-dalek,nightly" && cargo test --features "ed25519-salty,nightly" | ||||
|  | ||||
|       - name: Test sync | ||||
|         working-directory: ./embassy-sync | ||||
|         run: cargo test | ||||
|   | ||||
							
								
								
									
										4
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										4
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							| @@ -4,8 +4,4 @@ target_ci_stable | ||||
| Cargo.lock | ||||
| third_party | ||||
| /Cargo.toml | ||||
| stm32-metapac-gen/out/ | ||||
| stm32-metapac-backup | ||||
| stm32-metapac/src/chips | ||||
| stm32-metapac/src/peripherals | ||||
| out/ | ||||
|   | ||||
							
								
								
									
										3
									
								
								.gitmodules
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										3
									
								
								.gitmodules
									
									
									
									
										vendored
									
									
								
							| @@ -1,3 +0,0 @@ | ||||
| [submodule "stm32-data"] | ||||
| 	path = stm32-data | ||||
| 	url = https://github.com/embassy-rs/stm32-data.git | ||||
							
								
								
									
										11
									
								
								.vscode/extensions.json
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										11
									
								
								.vscode/extensions.json
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,11 @@ | ||||
| { | ||||
| 	// See https://go.microsoft.com/fwlink/?LinkId=827846 to learn about workspace recommendations. | ||||
| 	// Extension identifier format: ${publisher}.${name}. Example: vscode.csharp | ||||
| 	// List of extensions which should be recommended for users of this workspace. | ||||
| 	"recommendations": [ | ||||
| 		"rust-lang.rust-analyzer", | ||||
| 		"tamasfe.even-better-toml", | ||||
| 	], | ||||
| 	// List of extensions recommended by VS Code that should not be recommended for users of this workspace. | ||||
| 	"unwantedRecommendations": [] | ||||
| } | ||||
							
								
								
									
										37
									
								
								.vscode/settings.json
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										37
									
								
								.vscode/settings.json
									
									
									
									
										vendored
									
									
								
							| @@ -1,44 +1,43 @@ | ||||
| { | ||||
|   "editor.formatOnSave": true, | ||||
|   "rust-analyzer.checkOnSave.allTargets": false, | ||||
|   "rust-analyzer.checkOnSave.noDefaultFeatures": true, | ||||
|   "[toml]": { | ||||
|     "editor.formatOnSave": false | ||||
|   }, | ||||
|   "rust-analyzer.check.allTargets": false, | ||||
|   "rust-analyzer.check.noDefaultFeatures": true, | ||||
|   "rust-analyzer.cargo.noDefaultFeatures": true, | ||||
|   "rust-analyzer.procMacro.enable": true, | ||||
|   "rust-analyzer.cargo.target": "thumbv7em-none-eabi", | ||||
|   //"rust-analyzer.cargo.target": "thumbv8m.main-none-eabihf", | ||||
|   "rust-analyzer.cargo.features": [ | ||||
|     // These are needed to prevent embassy-net from failing to build | ||||
|     //"embassy-net/medium-ethernet", | ||||
|     //"embassy-net/tcp", | ||||
|     //"embassy-net/pool-16", | ||||
|     //"time-tick-16mhz", | ||||
|     //"defmt-timestamp-uptime", | ||||
|     "nightly", | ||||
|     //"unstable-traits", | ||||
|   ], | ||||
|   "rust-analyzer.linkedProjects": [ | ||||
|     // Declare for the target you wish to develop | ||||
|     //"embassy-executor/Cargo.toml", | ||||
|     //"embassy-sync/Cargo.toml", | ||||
|     "examples/nrf/Cargo.toml", | ||||
|     // "embassy-executor/Cargo.toml", | ||||
|     // "embassy-sync/Cargo.toml", | ||||
|     "examples/nrf52840/Cargo.toml", | ||||
|     // "examples/nrf5340/Cargo.toml", | ||||
|     // "examples/nrf-rtos-trace/Cargo.toml", | ||||
|     // "examples/rp/Cargo.toml", | ||||
|     // "examples/std/Cargo.toml", | ||||
|     // "examples/stm32c0/Cargo.toml", | ||||
|     // "examples/stm32f0/Cargo.toml", | ||||
|     // "examples/stm32f1/Cargo.toml", | ||||
|     // "examples/stm32f2/Cargo.toml", | ||||
|     // "examples/stm32f3/Cargo.toml", | ||||
|     // "examples/stm32f4/Cargo.toml", | ||||
|     // "examples/stm32f7/Cargo.toml", | ||||
|     // "examples/stm32g0/Cargo.toml", | ||||
|     // "examples/stm32g4/Cargo.toml", | ||||
|     // "examples/stm32h5/Cargo.toml", | ||||
|     // "examples/stm32h7/Cargo.toml", | ||||
|     // "examples/stm32l0/Cargo.toml", | ||||
|     // "examples/stm32l1/Cargo.toml", | ||||
|     // "examples/stm32l4/Cargo.toml", | ||||
|     // "examples/stm32l5/Cargo.toml", | ||||
|     // "examples/stm32u5/Cargo.toml", | ||||
|     // "examples/stm32wb55/Cargo.toml", | ||||
|     // "examples/stm32wl55/Cargo.toml", | ||||
|     // "examples/stm32wb/Cargo.toml", | ||||
|     // "examples/stm32wl/Cargo.toml", | ||||
|     // "examples/wasm/Cargo.toml", | ||||
|   ], | ||||
|   "rust-analyzer.imports.granularity.enforce": true, | ||||
|   "rust-analyzer.imports.granularity.group": "module", | ||||
|   "rust-analyzer.cargo.buildScripts.enable": true, | ||||
|   "rust-analyzer.procMacro.attributes.enable": false, | ||||
| } | ||||
							
								
								
									
										19
									
								
								README.md
									
									
									
									
									
								
							
							
						
						
									
										19
									
								
								README.md
									
									
									
									
									
								
							| @@ -2,7 +2,7 @@ | ||||
|  | ||||
| Embassy is the next-generation framework for embedded applications. Write safe, correct and energy-efficient embedded code faster, using the Rust programming language, its async facilities, and the Embassy libraries. | ||||
|  | ||||
| ## <a href="https://embassy.dev/embassy/dev/index.html">Documentation</a> - <a href="https://docs.embassy.dev/">API reference</a> - <a href="https://embassy.dev/">Website</a> - <a href="https://matrix.to/#/#embassy-rs:matrix.org">Chat</a> | ||||
| ## <a href="https://embassy.dev/dev/index.html">Documentation</a> - <a href="https://docs.embassy.dev/">API reference</a> - <a href="https://embassy.dev/">Website</a> - <a href="https://matrix.to/#/#embassy-rs:matrix.org">Chat</a> | ||||
| ## Rust + async ❤️ embedded | ||||
|  | ||||
| The Rust programming language is blazingly fast and memory-efficient, with no runtime, garbage collector or OS. It catches a wide variety of bugs at compile time, thanks to its full memory- and thread-safety, and expressive type system.  | ||||
| @@ -14,12 +14,16 @@ Rust's <a href="https://rust-lang.github.io/async-book/">async/await</a> allows | ||||
| - **Hardware Abstraction Layers** - HALs implement safe, idiomatic Rust APIs to use the hardware capabilities, so raw register manipulation is not needed. The Embassy project maintains HALs for select hardware, but you can still use HALs from other projects with Embassy. | ||||
|   - <a href="https://docs.embassy.dev/embassy-stm32/">embassy-stm32</a>, for all STM32 microcontroller families. | ||||
|   - <a href="https://docs.embassy.dev/embassy-nrf/">embassy-nrf</a>, for the Nordic Semiconductor nRF52, nRF53, nRF91 series. | ||||
|   - <a href="https://docs.embassy.dev/embassy-rp/">embassy-rp</a>, for the Raspberry Pi RP2040 microcontroller. | ||||
|   - <a href="https://github.com/esp-rs">esp-rs</a>, for the Espressif Systems ESP32 series of chips. | ||||
|     - Embassy HAL support for Espressif chips is being developed in the [esp-rs/esp-hal](https://github.com/esp-rs/esp-hal) repository. | ||||
|     - Async WiFi, Bluetooth and ESP-NOW is being developed in the [esp-rs/esp-wifi](https://github.com/esp-rs/esp-wifi) repository. | ||||
|  | ||||
| - **Time that Just Works** -  | ||||
| No more messing with hardware timers. <a href="https://docs.embassy.dev/embassy-time">embassy_time</a> provides Instant, Duration and Timer types that are globally available and never overflow. | ||||
|  | ||||
| - **Real-time ready** -  | ||||
| Tasks on the same async executor run cooperatively, but you can create multiple executors with different priorities, so that higher priority tasks preempt lower priority ones. See the <a href="https://github.com/embassy-rs/embassy/blob/master/examples/nrf/src/bin/multiprio.rs">example</a>. | ||||
| Tasks on the same async executor run cooperatively, but you can create multiple executors with different priorities, so that higher priority tasks preempt lower priority ones. See the <a href="https://github.com/embassy-rs/embassy/blob/master/examples/nrf52840/src/bin/multiprio.rs">example</a>. | ||||
|  | ||||
| - **Low-power ready** -  | ||||
| Easily build devices with years of battery life. The async executor automatically puts the core to sleep when there's no work to do. Tasks are woken by interrupts, there is no busy-loop polling while waiting. | ||||
| @@ -31,7 +35,7 @@ The <a href="https://docs.embassy.dev/embassy-net/">embassy-net</a> network stac | ||||
| The <a href="https://github.com/embassy-rs/nrf-softdevice">nrf-softdevice</a> crate provides Bluetooth Low Energy 4.x and 5.x support for nRF52 microcontrollers. | ||||
|  | ||||
| - **LoRa** -  | ||||
| <a href="https://docs.embassy.dev/embassy-lora/">embassy-lora</a> supports LoRa networking on STM32WL wireless microcontrollers and Semtech SX127x transceivers. | ||||
| <a href="https://docs.embassy.dev/embassy-lora/">embassy-lora</a> supports LoRa networking. | ||||
|  | ||||
| - **USB** -  | ||||
| <a href="https://docs.embassy.dev/embassy-usb/">embassy-usb</a> implements a device-side USB stack. Implementations for common classes such as USB serial (CDC ACM) and USB HID are available, and a rich builder API allows building your own. | ||||
| @@ -87,7 +91,8 @@ async fn main(spawner: Spawner) { | ||||
|  | ||||
| Examples are found in the `examples/` folder seperated by the chip manufacturer they are designed to run on. For example: | ||||
|  | ||||
| *   `examples/nrf` run on the `nrf52840-dk` board (PCA10056) but should be easily adaptable to other nRF52 chips and boards. | ||||
| *   `examples/nrf52840` run on the `nrf52840-dk` board (PCA10056) but should be easily adaptable to other nRF52 chips and boards. | ||||
| *   `examples/nrf5340` run on the `nrf5340-dk` board (PCA10095). | ||||
| *   `examples/stm32xx` for the various STM32 families. | ||||
| *   `examples/rp` are for the RP2040 chip. | ||||
| *   `examples/std` are designed to run locally on your PC. | ||||
| @@ -101,16 +106,16 @@ git submodule init | ||||
| git submodule update | ||||
| ``` | ||||
|  | ||||
| - Install `probe-run` with defmt support. | ||||
| - Install `probe-rs-cli` with defmt support. | ||||
|  | ||||
| ```bash | ||||
| cargo install probe-run | ||||
| cargo install probe-rs-cli | ||||
| ``` | ||||
|  | ||||
| - Change directory to the sample's base directory. For example: | ||||
|  | ||||
| ```bash | ||||
| cd examples/nrf | ||||
| cd examples/nrf52840 | ||||
| ``` | ||||
|  | ||||
| - Run the example | ||||
|   | ||||
							
								
								
									
										96
									
								
								ci.sh
									
									
									
									
									
								
							
							
						
						
									
										96
									
								
								ci.sh
									
									
									
									
									
								
							| @@ -8,40 +8,31 @@ export DEFMT_LOG=trace | ||||
|  | ||||
| TARGET=$(rustc -vV | sed -n 's|host: ||p') | ||||
|  | ||||
| BUILD_EXTRA="" | ||||
| if [ $TARGET = "x86_64-unknown-linux-gnu" ]; then | ||||
|     BUILD_EXTRA="--- build --release --manifest-path examples/std/Cargo.toml --target $TARGET --out-dir out/examples/std" | ||||
| else | ||||
|     BUILD_EXTRA="" | ||||
| fi | ||||
|  | ||||
| find . -name '*.rs' -not -path '*target*' -not -path '*stm32-metapac-gen/out/*' -not -path '*stm32-metapac/src/*' | xargs rustfmt --check  --skip-children --unstable-features --edition 2018 | ||||
|  | ||||
| # Generate stm32-metapac | ||||
| if [ ! -d "stm32-metapac-backup" ] | ||||
| then | ||||
|     cp -r stm32-metapac stm32-metapac-backup | ||||
| fi | ||||
|  | ||||
| rm -rf stm32-metapac | ||||
| cp -r stm32-metapac-backup stm32-metapac | ||||
|  | ||||
| # for some reason Cargo stomps the cache if we don't specify --target. | ||||
| # This happens with vanilla Cargo, not just cargo-batch. Bug? | ||||
| (cd stm32-metapac-gen; cargo run --release --target $TARGET) | ||||
| rm -rf stm32-metapac | ||||
| mv stm32-metapac-gen/out stm32-metapac | ||||
| find . -name '*.rs' -not -path '*target*' | xargs rustfmt --check  --skip-children --unstable-features --edition 2018 | ||||
|  | ||||
| cargo batch  \ | ||||
|     --- build --release --manifest-path embassy-executor/Cargo.toml --target thumbv7em-none-eabi --features nightly \ | ||||
|     --- build --release --manifest-path embassy-executor/Cargo.toml --target thumbv7em-none-eabi --features nightly,log \ | ||||
|     --- build --release --manifest-path embassy-executor/Cargo.toml --target thumbv7em-none-eabi --features nightly,defmt \ | ||||
|     --- build --release --manifest-path embassy-executor/Cargo.toml --target thumbv6m-none-eabi --features nightly,defmt \ | ||||
|     --- build --release --manifest-path embassy-sync/Cargo.toml --target thumbv6m-none-eabi --features nightly,defmt \ | ||||
|     --- build --release --manifest-path embassy-time/Cargo.toml --target thumbv6m-none-eabi --features nightly,unstable-traits,defmt,defmt-timestamp-uptime,tick-hz-32_768,generic-queue-8 \ | ||||
|     --- build --release --manifest-path embassy-net/Cargo.toml --target thumbv7em-none-eabi --features defmt,tcp,udp,dns,medium-ethernet \ | ||||
|     --- build --release --manifest-path embassy-net/Cargo.toml --target thumbv7em-none-eabi --features defmt,tcp,udp,dns,dhcpv4,medium-ethernet \ | ||||
|     --- build --release --manifest-path embassy-net/Cargo.toml --target thumbv7em-none-eabi --features defmt,tcp,udp,dns,dhcpv4,medium-ethernet,unstable-traits \ | ||||
|     --- build --release --manifest-path embassy-net/Cargo.toml --target thumbv7em-none-eabi --features defmt,tcp,udp,dns,dhcpv4,medium-ethernet,nightly \ | ||||
|     --- build --release --manifest-path embassy-net/Cargo.toml --target thumbv7em-none-eabi --features defmt,tcp,udp,dns,dhcpv4,medium-ethernet,unstable-traits,nightly \ | ||||
|     --- build --release --manifest-path embassy-nrf/Cargo.toml --target thumbv7em-none-eabi --features nightly,nrf52805,gpiote,time-driver-rtc1 \ | ||||
|     --- build --release --manifest-path embassy-nrf/Cargo.toml --target thumbv7em-none-eabi --features nightly,nrf52810,gpiote,time-driver-rtc1 \ | ||||
|     --- build --release --manifest-path embassy-nrf/Cargo.toml --target thumbv7em-none-eabi --features nightly,nrf52811,gpiote,time-driver-rtc1 \ | ||||
|     --- build --release --manifest-path embassy-nrf/Cargo.toml --target thumbv7em-none-eabi --features nightly,nrf52820,gpiote,time-driver-rtc1 \ | ||||
|     --- build --release --manifest-path embassy-nrf/Cargo.toml --target thumbv7em-none-eabi --features nightly,nrf52832,gpiote,time-driver-rtc1 \ | ||||
|     --- build --release --manifest-path embassy-nrf/Cargo.toml --target thumbv7em-none-eabi --features nightly,nrf52833,gpiote,time-driver-rtc1,unstable-traits \ | ||||
|     --- build --release --manifest-path embassy-nrf/Cargo.toml --target thumbv7em-none-eabi --features nightly,nrf52832,gpiote,time-driver-rtc1,reset-pin-as-gpio \ | ||||
|     --- build --release --manifest-path embassy-nrf/Cargo.toml --target thumbv7em-none-eabi --features nightly,nrf52833,gpiote,time-driver-rtc1,unstable-traits,nfc-pins-as-gpio \ | ||||
|     --- build --release --manifest-path embassy-nrf/Cargo.toml --target thumbv8m.main-none-eabihf --features nightly,nrf9160-s,gpiote,time-driver-rtc1 \ | ||||
|     --- build --release --manifest-path embassy-nrf/Cargo.toml --target thumbv8m.main-none-eabihf --features nightly,nrf9160-ns,gpiote,time-driver-rtc1,unstable-traits \ | ||||
|     --- build --release --manifest-path embassy-nrf/Cargo.toml --target thumbv8m.main-none-eabihf --features nightly,nrf5340-app-s,gpiote,time-driver-rtc1,unstable-traits \ | ||||
| @@ -54,33 +45,41 @@ cargo batch  \ | ||||
|     --- build --release --manifest-path embassy-rp/Cargo.toml --target thumbv6m-none-eabi --features nightly,unstable-traits,log \ | ||||
|     --- build --release --manifest-path embassy-rp/Cargo.toml --target thumbv6m-none-eabi --features nightly,unstable-traits \ | ||||
|     --- build --release --manifest-path embassy-rp/Cargo.toml --target thumbv6m-none-eabi --features nightly \ | ||||
|     --- build --release --manifest-path embassy-stm32/Cargo.toml --target thumbv7em-none-eabi --features nightly,stm32f410tb,defmt,exti,time-driver-any,embassy-time?/tick-32768hz,unstable-traits \ | ||||
|     --- build --release --manifest-path embassy-stm32/Cargo.toml --target thumbv7em-none-eabi --features nightly,stm32f411ce,defmt,exti,time-driver-any,embassy-time?/tick-32768hz,unstable-traits \ | ||||
|     --- build --release --manifest-path embassy-stm32/Cargo.toml --target thumbv7em-none-eabi --features nightly,stm32f429zi,log,exti,time-driver-any,embassy-time?/tick-32768hz,unstable-traits \ | ||||
|     --- build --release --manifest-path embassy-stm32/Cargo.toml --target thumbv7em-none-eabi --features nightly,stm32h755zi-cm7,defmt,exti,time-driver-any,embassy-time?/tick-32768hz,unstable-traits \ | ||||
|     --- build --release --manifest-path embassy-stm32/Cargo.toml --target thumbv7em-none-eabi --features nightly,stm32h7b3ai,defmt,exti,time-driver-any,embassy-time?/tick-32768hz,unstable-traits \ | ||||
|     --- build --release --manifest-path embassy-stm32/Cargo.toml --target thumbv7em-none-eabi --features nightly,stm32l476vg,defmt,exti,time-driver-any,embassy-time?/tick-32768hz,unstable-traits \ | ||||
|     --- build --release --manifest-path embassy-stm32/Cargo.toml --target thumbv7em-none-eabi --features nightly,stm32wb15cc,defmt,exti,time-driver-any,embassy-time?/tick-32768hz,unstable-traits \ | ||||
|     --- build --release --manifest-path embassy-stm32/Cargo.toml --target thumbv6m-none-eabi --features nightly,stm32l072cz,defmt,exti,time-driver-any,embassy-time?/tick-32768hz,unstable-traits \ | ||||
|     --- build --release --manifest-path embassy-stm32/Cargo.toml --target thumbv6m-none-eabi --features nightly,stm32l041f6,defmt,exti,time-driver-any,embassy-time?/tick-32768hz,unstable-traits \ | ||||
|     --- build --release --manifest-path embassy-stm32/Cargo.toml --target thumbv7m-none-eabi --features nightly,stm32l151cb-a,defmt,exti,time-driver-any,embassy-time?/tick-32768hz,unstable-traits \ | ||||
|     --- build --release --manifest-path embassy-stm32/Cargo.toml --target thumbv7m-none-eabi --features nightly,stm32f398ve,defmt,exti,time-driver-any,embassy-time?/tick-32768hz,unstable-traits \ | ||||
|     --- build --release --manifest-path embassy-stm32/Cargo.toml --target thumbv6m-none-eabi --features nightly,stm32g0c1ve,defmt,exti,time-driver-any,embassy-time?/tick-32768hz,unstable-traits \ | ||||
|     --- build --release --manifest-path embassy-stm32/Cargo.toml --target thumbv7m-none-eabi --features nightly,stm32f217zg,defmt,exti,time-driver-any,embassy-time?/tick-32768hz,unstable-traits \ | ||||
|     --- build --release --manifest-path embassy-stm32/Cargo.toml --target thumbv8m.main-none-eabihf --features nightly,stm32l552ze,defmt,exti,time-driver-any,embassy-time?/tick-32768hz,unstable-traits \ | ||||
|     --- build --release --manifest-path embassy-stm32/Cargo.toml --target thumbv6m-none-eabi --features nightly,stm32wl54jc-cm0p,defmt,exti,time-driver-any,embassy-time?/tick-32768hz,unstable-traits \ | ||||
|     --- build --release --manifest-path embassy-stm32/Cargo.toml --target thumbv7em-none-eabi --features nightly,stm32wle5ub,defmt,exti,time-driver-any,embassy-time?/tick-32768hz,unstable-traits \ | ||||
|     --- build --release --manifest-path embassy-stm32/Cargo.toml --target thumbv7m-none-eabi --features nightly,stm32f107vc,defmt,exti,time-driver-any,embassy-time?/tick-32768hz,unstable-traits \ | ||||
|     --- build --release --manifest-path embassy-stm32/Cargo.toml --target thumbv7m-none-eabi --features nightly,stm32f103re,defmt,exti,time-driver-any,embassy-time?/tick-32768hz,unstable-traits \ | ||||
|     --- build --release --manifest-path embassy-stm32/Cargo.toml --target thumbv7m-none-eabi --features nightly,stm32f100c4,defmt,exti,time-driver-any,embassy-time?/tick-32768hz,unstable-traits \ | ||||
|     --- build --release --manifest-path embassy-boot/nrf/Cargo.toml --target thumbv7em-none-eabi --features embassy-nrf/nrf52840 \ | ||||
|     --- build --release --manifest-path embassy-boot/stm32/Cargo.toml --target thumbv7em-none-eabi --features embassy-stm32/stm32wl55jc-cm4 \ | ||||
|     --- build --release --manifest-path embassy-rp/Cargo.toml --target thumbv6m-none-eabi --features nightly,intrinsics \ | ||||
|     --- build --release --manifest-path embassy-stm32/Cargo.toml --target thumbv7em-none-eabi --features nightly,stm32f410tb,defmt,exti,time-driver-any,unstable-traits \ | ||||
|     --- build --release --manifest-path embassy-stm32/Cargo.toml --target thumbv7em-none-eabi --features nightly,stm32f411ce,defmt,exti,time-driver-any,unstable-traits \ | ||||
|     --- build --release --manifest-path embassy-stm32/Cargo.toml --target thumbv7em-none-eabi --features nightly,stm32f413vh,defmt,exti,time-driver-any,unstable-traits \ | ||||
|     --- build --release --manifest-path embassy-stm32/Cargo.toml --target thumbv7em-none-eabi --features nightly,stm32f429zi,log,exti,time-driver-any,unstable-traits,embedded-sdmmc \ | ||||
|     --- build --release --manifest-path embassy-stm32/Cargo.toml --target thumbv7em-none-eabi --features nightly,stm32f730i8,defmt,exti,time-driver-any,unstable-traits \ | ||||
|     --- build --release --manifest-path embassy-stm32/Cargo.toml --target thumbv7em-none-eabi --features nightly,stm32h755zi-cm7,defmt,exti,time-driver-any,unstable-traits \ | ||||
|     --- build --release --manifest-path embassy-stm32/Cargo.toml --target thumbv7em-none-eabi --features nightly,stm32h7b3ai,defmt,exti,time-driver-any,unstable-traits \ | ||||
|     --- build --release --manifest-path embassy-stm32/Cargo.toml --target thumbv7em-none-eabi --features nightly,stm32l476vg,defmt,exti,time-driver-any,unstable-traits \ | ||||
|     --- build --release --manifest-path embassy-stm32/Cargo.toml --target thumbv7em-none-eabi --features nightly,stm32wb15cc,defmt,exti,time-driver-any,unstable-traits \ | ||||
|     --- build --release --manifest-path embassy-stm32/Cargo.toml --target thumbv6m-none-eabi --features nightly,stm32l072cz,defmt,exti,time-driver-any,unstable-traits \ | ||||
|     --- build --release --manifest-path embassy-stm32/Cargo.toml --target thumbv6m-none-eabi --features nightly,stm32l041f6,defmt,exti,time-driver-any,unstable-traits \ | ||||
|     --- build --release --manifest-path embassy-stm32/Cargo.toml --target thumbv7m-none-eabi --features nightly,stm32l151cb-a,defmt,exti,time-driver-any,unstable-traits \ | ||||
|     --- build --release --manifest-path embassy-stm32/Cargo.toml --target thumbv7m-none-eabi --features nightly,stm32f398ve,defmt,exti,time-driver-any,unstable-traits \ | ||||
|     --- build --release --manifest-path embassy-stm32/Cargo.toml --target thumbv6m-none-eabi --features nightly,stm32g0c1ve,defmt,exti,time-driver-any,unstable-traits \ | ||||
|     --- build --release --manifest-path embassy-stm32/Cargo.toml --target thumbv7m-none-eabi --features nightly,stm32f217zg,defmt,exti,time-driver-any,unstable-traits \ | ||||
|     --- build --release --manifest-path embassy-stm32/Cargo.toml --target thumbv8m.main-none-eabihf --features nightly,stm32l552ze,defmt,exti,time-driver-any,unstable-traits \ | ||||
|     --- build --release --manifest-path embassy-stm32/Cargo.toml --target thumbv6m-none-eabi --features nightly,stm32wl54jc-cm0p,defmt,exti,time-driver-any,unstable-traits \ | ||||
|     --- build --release --manifest-path embassy-stm32/Cargo.toml --target thumbv7em-none-eabi --features nightly,stm32wle5jb,defmt,exti,time-driver-any,unstable-traits \ | ||||
|     --- build --release --manifest-path embassy-stm32/Cargo.toml --target thumbv7m-none-eabi --features nightly,stm32f107vc,defmt,exti,time-driver-any,unstable-traits \ | ||||
|     --- build --release --manifest-path embassy-stm32/Cargo.toml --target thumbv7m-none-eabi --features nightly,stm32f103re,defmt,exti,time-driver-any,unstable-traits \ | ||||
|     --- build --release --manifest-path embassy-stm32/Cargo.toml --target thumbv7m-none-eabi --features nightly,stm32f100c4,defmt,exti,time-driver-any,unstable-traits \ | ||||
|     --- build --release --manifest-path embassy-stm32/Cargo.toml --target thumbv7m-none-eabi --features nightly,stm32h503rb,defmt,exti,time-driver-any,unstable-traits \ | ||||
|     --- build --release --manifest-path embassy-stm32/Cargo.toml --target thumbv7m-none-eabi --features nightly,stm32h562ag,defmt,exti,time-driver-any,unstable-traits \ | ||||
|     --- build --release --manifest-path embassy-boot/nrf/Cargo.toml --target thumbv7em-none-eabi --features embassy-nrf/nrf52840,nightly \ | ||||
|     --- build --release --manifest-path embassy-boot/nrf/Cargo.toml --target thumbv8m.main-none-eabihf --features embassy-nrf/nrf9160-ns,nightly \ | ||||
|     --- build --release --manifest-path embassy-boot/rp/Cargo.toml --target thumbv6m-none-eabi --features nightly \ | ||||
|     --- build --release --manifest-path embassy-boot/stm32/Cargo.toml --target thumbv7em-none-eabi --features embassy-stm32/stm32wl55jc-cm4,nightly \ | ||||
|     --- build --release --manifest-path docs/modules/ROOT/examples/basic/Cargo.toml --target thumbv7em-none-eabi \ | ||||
|     --- build --release --manifest-path docs/modules/ROOT/examples/layer-by-layer/blinky-pac/Cargo.toml --target thumbv7em-none-eabi \ | ||||
|     --- build --release --manifest-path docs/modules/ROOT/examples/layer-by-layer/blinky-hal/Cargo.toml --target thumbv7em-none-eabi \ | ||||
|     --- build --release --manifest-path docs/modules/ROOT/examples/layer-by-layer/blinky-irq/Cargo.toml --target thumbv7em-none-eabi \ | ||||
|     --- build --release --manifest-path docs/modules/ROOT/examples/layer-by-layer/blinky-async/Cargo.toml --target thumbv7em-none-eabi \ | ||||
|     --- build --release --manifest-path examples/nrf/Cargo.toml --target thumbv7em-none-eabi --out-dir out/examples/nrf \ | ||||
|     --- build --release --manifest-path examples/nrf52840/Cargo.toml --target thumbv7em-none-eabi --out-dir out/examples/nrf52840 \ | ||||
|     --- build --release --manifest-path examples/nrf5340/Cargo.toml --target thumbv8m.main-none-eabihf --out-dir out/examples/nrf5340 \ | ||||
|     --- build --release --manifest-path examples/rp/Cargo.toml --target thumbv6m-none-eabi --out-dir out/examples/rp \ | ||||
|     --- build --release --manifest-path examples/stm32f0/Cargo.toml --target thumbv6m-none-eabi --out-dir out/examples/stm32f0 \ | ||||
|     --- build --release --manifest-path examples/stm32f1/Cargo.toml --target thumbv7m-none-eabi --out-dir out/examples/stm32f1 \ | ||||
| @@ -88,16 +87,21 @@ cargo batch  \ | ||||
|     --- build --release --manifest-path examples/stm32f3/Cargo.toml --target thumbv7em-none-eabihf --out-dir out/examples/stm32f3 \ | ||||
|     --- build --release --manifest-path examples/stm32f4/Cargo.toml --target thumbv7em-none-eabi --out-dir out/examples/stm32f4 \ | ||||
|     --- build --release --manifest-path examples/stm32f7/Cargo.toml --target thumbv7em-none-eabihf --out-dir out/examples/stm32f7 \ | ||||
|     --- build --release --manifest-path examples/stm32c0/Cargo.toml --target thumbv6m-none-eabi --out-dir out/examples/stm32c0 \ | ||||
|     --- build --release --manifest-path examples/stm32g0/Cargo.toml --target thumbv6m-none-eabi --out-dir out/examples/stm32g0 \ | ||||
|     --- build --release --manifest-path examples/stm32g4/Cargo.toml --target thumbv7em-none-eabi --out-dir out/examples/stm32g4 \ | ||||
|     --- build --release --manifest-path examples/stm32h5/Cargo.toml --target thumbv7em-none-eabi --out-dir out/examples/stm32h5 \ | ||||
|     --- build --release --manifest-path examples/stm32h7/Cargo.toml --target thumbv7em-none-eabi --out-dir out/examples/stm32h7 \ | ||||
|     --- build --release --manifest-path examples/stm32l0/Cargo.toml --target thumbv6m-none-eabi --out-dir out/examples/stm32l0 \ | ||||
|     --- build --release --manifest-path examples/stm32l1/Cargo.toml --target thumbv7m-none-eabi --out-dir out/examples/stm32l1 \ | ||||
|     --- build --release --manifest-path examples/stm32l4/Cargo.toml --target thumbv7em-none-eabi --out-dir out/examples/stm32l4 \ | ||||
|     --- build --release --manifest-path examples/stm32l5/Cargo.toml --target thumbv8m.main-none-eabihf --out-dir out/examples/stm32l5 \ | ||||
|     --- build --release --manifest-path examples/stm32u5/Cargo.toml --target thumbv8m.main-none-eabihf --out-dir out/examples/stm32u5 \ | ||||
|     --- build --release --manifest-path examples/stm32wb/Cargo.toml --target thumbv7em-none-eabihf --out-dir out/examples/stm32wb \ | ||||
|     --- build --release --manifest-path examples/stm32wl/Cargo.toml --target thumbv7em-none-eabihf --out-dir out/examples/stm32wl \ | ||||
|     --- build --release --manifest-path examples/boot/application/nrf/Cargo.toml --target thumbv7em-none-eabi --out-dir out/examples/boot/nrf --bin b \ | ||||
|     --- build --release --manifest-path examples/boot/application/nrf/Cargo.toml --target thumbv7em-none-eabi --features embassy-nrf/nrf52840 --out-dir out/examples/boot/nrf --bin b \ | ||||
|     --- build --release --manifest-path examples/boot/application/nrf/Cargo.toml --target thumbv8m.main-none-eabihf --features embassy-nrf/nrf9160-ns --out-dir out/examples/boot/nrf --bin b \ | ||||
|     --- build --release --manifest-path examples/boot/application/rp/Cargo.toml --target thumbv6m-none-eabi --out-dir out/examples/boot/rp --bin b \ | ||||
|     --- build --release --manifest-path examples/boot/application/stm32f3/Cargo.toml --target thumbv7em-none-eabi --out-dir out/examples/boot/stm32f3 --bin b \ | ||||
|     --- build --release --manifest-path examples/boot/application/stm32f7/Cargo.toml --target thumbv7em-none-eabi --out-dir out/examples/boot/stm32f7 --bin b \ | ||||
|     --- build --release --manifest-path examples/boot/application/stm32h7/Cargo.toml --target thumbv7em-none-eabi --out-dir out/examples/boot/stm32h7 --bin b \ | ||||
| @@ -106,16 +110,22 @@ cargo batch  \ | ||||
|     --- build --release --manifest-path examples/boot/application/stm32l4/Cargo.toml --target thumbv7em-none-eabi --out-dir out/examples/boot/stm32l4 --bin b \ | ||||
|     --- build --release --manifest-path examples/boot/application/stm32wl/Cargo.toml --target thumbv7em-none-eabihf --out-dir out/examples/boot/stm32wl --bin b \ | ||||
|     --- build --release --manifest-path examples/boot/bootloader/nrf/Cargo.toml --target thumbv7em-none-eabi --features embassy-nrf/nrf52840 \ | ||||
|     --- build --release --manifest-path examples/boot/bootloader/nrf/Cargo.toml --target thumbv8m.main-none-eabihf --features embassy-nrf/nrf9160-ns \ | ||||
|     --- build --release --manifest-path examples/boot/bootloader/rp/Cargo.toml --target thumbv6m-none-eabi \ | ||||
|     --- build --release --manifest-path examples/boot/bootloader/stm32/Cargo.toml --target thumbv7em-none-eabi --features embassy-stm32/stm32wl55jc-cm4 \ | ||||
|     --- build --release --manifest-path examples/wasm/Cargo.toml --target wasm32-unknown-unknown --out-dir out/examples/wasm \ | ||||
|     --- build --release --manifest-path tests/stm32/Cargo.toml --target thumbv7m-none-eabi --features stm32f103c8 --out-dir out/tests/bluepill-stm32f103c8 \ | ||||
|     --- build --release --manifest-path tests/stm32/Cargo.toml --target thumbv7em-none-eabi --features stm32f429zi --out-dir out/tests/nucleo-stm32f429zi \ | ||||
|     --- build --release --manifest-path tests/stm32/Cargo.toml --target thumbv7em-none-eabi --features stm32g491re --out-dir out/tests/nucleo-stm32g491re \ | ||||
|     --- build --release --manifest-path tests/stm32/Cargo.toml --target thumbv6m-none-eabi --features stm32g071rb --out-dir out/tests/nucleo-stm32g071rb \ | ||||
|     --- build --release --manifest-path tests/stm32/Cargo.toml --target thumbv6m-none-eabi --features stm32c031c6 --out-dir out/tests/nucleo-stm32c031c6 \ | ||||
|     --- build --release --manifest-path tests/stm32/Cargo.toml --target thumbv7em-none-eabi --features stm32h755zi --out-dir out/tests/nucleo-stm32h755zi \ | ||||
|     --- build --release --manifest-path tests/stm32/Cargo.toml --target thumbv7em-none-eabi --features stm32wb55rg --out-dir out/tests/nucleo-stm32wb55rg \ | ||||
|     --- build --release --manifest-path tests/stm32/Cargo.toml --target thumbv7em-none-eabi --features stm32h563zi --out-dir out/tests/nucleo-stm32h563zi \ | ||||
|     --- build --release --manifest-path tests/stm32/Cargo.toml --target thumbv7em-none-eabi --features stm32u585ai --out-dir out/tests/iot-stm32u585ai \ | ||||
|     --- build --release --manifest-path tests/rp/Cargo.toml --target thumbv6m-none-eabi --out-dir out/tests/rpi-pico \ | ||||
|     --- build --release --manifest-path tests/nrf/Cargo.toml --target thumbv7em-none-eabi --out-dir out/tests/nrf52840-dk \ | ||||
|     --- build --release --manifest-path tests/riscv32/Cargo.toml --target riscv32imac-unknown-none-elf \ | ||||
|     $BUILD_EXTRA | ||||
|  | ||||
|  | ||||
|   | ||||
							
								
								
									
										75
									
								
								ci_stable.sh
									
									
									
									
									
								
							
							
						
						
									
										75
									
								
								ci_stable.sh
									
									
									
									
									
								
							| @@ -9,10 +9,17 @@ export DEFMT_LOG=trace | ||||
| sed -i 's/channel.*/channel = "stable"/g' rust-toolchain.toml | ||||
|  | ||||
| cargo batch  \ | ||||
|     --- build --release --manifest-path embassy-boot/nrf/Cargo.toml --target thumbv7em-none-eabi --features embassy-nrf/nrf52840 \ | ||||
|     --- build --release --manifest-path embassy-boot/nrf/Cargo.toml --target thumbv8m.main-none-eabihf --features embassy-nrf/nrf9160-ns \ | ||||
|     --- build --release --manifest-path embassy-boot/rp/Cargo.toml --target thumbv6m-none-eabi \ | ||||
|     --- build --release --manifest-path embassy-boot/stm32/Cargo.toml --target thumbv7em-none-eabi --features embassy-stm32/stm32wl55jc-cm4 \ | ||||
|     --- build --release --manifest-path embassy-executor/Cargo.toml --target thumbv7em-none-eabi \ | ||||
|     --- build --release --manifest-path embassy-executor/Cargo.toml --target thumbv7em-none-eabi --features log \ | ||||
|     --- build --release --manifest-path embassy-executor/Cargo.toml --target thumbv7em-none-eabi --features defmt \ | ||||
|     --- build --release --manifest-path embassy-executor/Cargo.toml --target thumbv6m-none-eabi --features defmt \ | ||||
|     --- build --release --manifest-path embassy-net/Cargo.toml --target thumbv7em-none-eabi --features defmt,tcp,udp,dns,medium-ethernet \ | ||||
|     --- build --release --manifest-path embassy-net/Cargo.toml --target thumbv7em-none-eabi --features defmt,tcp,udp,dns,dhcpv4,medium-ethernet \ | ||||
|     --- build --release --manifest-path embassy-net/Cargo.toml --target thumbv7em-none-eabi --features defmt,tcp,udp,dns,dhcpv4,medium-ethernet,unstable-traits \ | ||||
|     --- build --release --manifest-path embassy-nrf/Cargo.toml --target thumbv7em-none-eabi --features nrf52805,gpiote,time-driver-rtc1 \ | ||||
|     --- build --release --manifest-path embassy-nrf/Cargo.toml --target thumbv7em-none-eabi --features nrf52810,gpiote,time-driver-rtc1 \ | ||||
|     --- build --release --manifest-path embassy-nrf/Cargo.toml --target thumbv7em-none-eabi --features nrf52811,gpiote,time-driver-rtc1 \ | ||||
| @@ -30,38 +37,38 @@ cargo batch  \ | ||||
|     --- build --release --manifest-path embassy-rp/Cargo.toml --target thumbv6m-none-eabi --features unstable-traits,defmt \ | ||||
|     --- build --release --manifest-path embassy-rp/Cargo.toml --target thumbv6m-none-eabi --features unstable-traits,log \ | ||||
|     --- build --release --manifest-path embassy-rp/Cargo.toml --target thumbv6m-none-eabi \ | ||||
|     --- build --release --manifest-path embassy-stm32/Cargo.toml --target thumbv7em-none-eabi --features stm32g473cc,defmt,exti,time-driver-any,embassy-time?/tick-32768hz,unstable-traits \ | ||||
|     --- build --release --manifest-path embassy-stm32/Cargo.toml --target thumbv7em-none-eabi --features stm32g491re,defmt,exti,time-driver-any,embassy-time?/tick-32768hz,unstable-traits \ | ||||
|     --- build --release --manifest-path embassy-stm32/Cargo.toml --target thumbv7em-none-eabi --features stm32u585zi,defmt,exti,time-driver-any,embassy-time?/tick-32768hz,unstable-traits \ | ||||
|     --- build --release --manifest-path embassy-stm32/Cargo.toml --target thumbv7em-none-eabi --features stm32wb55vy,defmt,exti,time-driver-any,embassy-time?/tick-32768hz,unstable-traits \ | ||||
|     --- build --release --manifest-path embassy-stm32/Cargo.toml --target thumbv7em-none-eabi --features stm32wl55uc-cm4,defmt,exti,time-driver-any,embassy-time?/tick-32768hz,unstable-traits \ | ||||
|     --- build --release --manifest-path embassy-stm32/Cargo.toml --target thumbv7em-none-eabi --features stm32l4r9zi,defmt,exti,time-driver-any,embassy-time?/tick-32768hz,unstable-traits \ | ||||
|     --- build --release --manifest-path embassy-stm32/Cargo.toml --target thumbv7em-none-eabi --features stm32f303vc,defmt,exti,time-driver-any,embassy-time?/tick-32768hz,unstable-traits \ | ||||
|     --- build --release --manifest-path embassy-stm32/Cargo.toml --target thumbv7em-none-eabi --features stm32f411ce,defmt,time-driver-any,embassy-time?/tick-32768hz \ | ||||
|     --- build --release --manifest-path embassy-stm32/Cargo.toml --target thumbv7em-none-eabi --features stm32f411ce,defmt,time-driver-any,embassy-time?/tick-32768hz,unstable-traits \ | ||||
|     --- build --release --manifest-path embassy-stm32/Cargo.toml --target thumbv7em-none-eabi --features stm32f429zi,log,time-driver-any,embassy-time?/tick-32768hz \ | ||||
|     --- build --release --manifest-path embassy-stm32/Cargo.toml --target thumbv7em-none-eabi --features stm32f429zi,log,time-driver-any,embassy-time?/tick-32768hz,unstable-traits \ | ||||
|     --- build --release --manifest-path embassy-stm32/Cargo.toml --target thumbv7em-none-eabi --features stm32h755zi-cm7,defmt,time-driver-any,embassy-time?/tick-32768hz \ | ||||
|     --- build --release --manifest-path embassy-stm32/Cargo.toml --target thumbv7em-none-eabi --features stm32h755zi-cm7,defmt,time-driver-any,embassy-time?/tick-32768hz,unstable-traits \ | ||||
|     --- build --release --manifest-path embassy-stm32/Cargo.toml --target thumbv7em-none-eabi --features stm32l476vg,defmt,time-driver-any,embassy-time?/tick-32768hz \ | ||||
|     --- build --release --manifest-path embassy-stm32/Cargo.toml --target thumbv7em-none-eabi --features stm32l476vg,defmt,time-driver-any,embassy-time?/tick-32768hz,unstable-traits \ | ||||
|     --- build --release --manifest-path embassy-stm32/Cargo.toml --target thumbv6m-none-eabi --features stm32l072cz,defmt,time-driver-any,embassy-time?/tick-32768hz \ | ||||
|     --- build --release --manifest-path embassy-stm32/Cargo.toml --target thumbv6m-none-eabi --features stm32l072cz,defmt,time-driver-any,embassy-time?/tick-32768hz,unstable-traits \ | ||||
|     --- build --release --manifest-path embassy-stm32/Cargo.toml --target thumbv7m-none-eabi --features stm32l151cb-a,defmt,time-driver-any,embassy-time?/tick-32768hz \ | ||||
|     --- build --release --manifest-path embassy-stm32/Cargo.toml --target thumbv7m-none-eabi --features stm32l151cb-a,defmt,time-driver-any,embassy-time?/tick-32768hz,unstable-traits \ | ||||
|     --- build --release --manifest-path embassy-stm32/Cargo.toml --target thumbv7em-none-eabi --features stm32f410tb,defmt,exti,time-driver-any,embassy-time?/tick-32768hz \ | ||||
|     --- build --release --manifest-path embassy-stm32/Cargo.toml --target thumbv7em-none-eabi --features stm32f410tb,defmt,exti,time-driver-any,embassy-time?/tick-32768hz,unstable-traits \ | ||||
|     --- build --release --manifest-path embassy-stm32/Cargo.toml --target thumbv7em-none-eabi --features stm32f429zi,log,exti,time-driver-any,embassy-time?/tick-32768hz \ | ||||
|     --- build --release --manifest-path embassy-stm32/Cargo.toml --target thumbv7em-none-eabi --features stm32f429zi,log,exti,time-driver-any,embassy-time?/tick-32768hz,unstable-traits \ | ||||
|     --- build --release --manifest-path embassy-stm32/Cargo.toml --target thumbv7em-none-eabi --features stm32h755zi-cm7,defmt,exti,time-driver-any,embassy-time?/tick-32768hz \ | ||||
|     --- build --release --manifest-path embassy-stm32/Cargo.toml --target thumbv7em-none-eabi --features stm32h755zi-cm7,defmt,exti,time-driver-any,embassy-time?/tick-32768hz,unstable-traits \ | ||||
|     --- build --release --manifest-path embassy-stm32/Cargo.toml --target thumbv7em-none-eabi --features stm32l476vg,defmt,exti,time-driver-any,embassy-time?/tick-32768hz \ | ||||
|     --- build --release --manifest-path embassy-stm32/Cargo.toml --target thumbv7em-none-eabi --features stm32l476vg,defmt,exti,time-driver-any,embassy-time?/tick-32768hz,unstable-traits \ | ||||
|     --- build --release --manifest-path embassy-stm32/Cargo.toml --target thumbv6m-none-eabi --features stm32l072cz,defmt,exti,time-driver-any,embassy-time?/tick-32768hz \ | ||||
|     --- build --release --manifest-path embassy-stm32/Cargo.toml --target thumbv6m-none-eabi --features stm32l072cz,defmt,exti,time-driver-any,embassy-time?/tick-32768hz,unstable-traits \ | ||||
|     --- build --release --manifest-path embassy-stm32/Cargo.toml --target thumbv7m-none-eabi --features stm32l151cb-a,defmt,exti,time-driver-any,embassy-time?/tick-32768hz \ | ||||
|     --- build --release --manifest-path embassy-stm32/Cargo.toml --target thumbv7m-none-eabi --features stm32l151cb-a,defmt,exti,time-driver-any,embassy-time?/tick-32768hz,unstable-traits \ | ||||
|     --- build --release --manifest-path embassy-stm32/Cargo.toml --target thumbv7m-none-eabi --features stm32f217zg,defmt,exti,time-driver-any,embassy-time?/tick-32768hz \ | ||||
|     --- build --release --manifest-path embassy-stm32/Cargo.toml --target thumbv7m-none-eabi --features stm32f217zg,defmt,exti,time-driver-any,embassy-time?/tick-32768hz,unstable-traits \ | ||||
|     --- build --release --manifest-path examples/nrf/Cargo.toml --target thumbv7em-none-eabi --no-default-features --out-dir out/examples/nrf --bin raw_spawn \ | ||||
|     --- build --release --manifest-path embassy-stm32/Cargo.toml --target thumbv7em-none-eabi --features stm32g473cc,defmt,exti,time-driver-any,unstable-traits \ | ||||
|     --- build --release --manifest-path embassy-stm32/Cargo.toml --target thumbv7em-none-eabi --features stm32g491re,defmt,exti,time-driver-any,unstable-traits \ | ||||
|     --- build --release --manifest-path embassy-stm32/Cargo.toml --target thumbv7em-none-eabi --features stm32u585zi,defmt,exti,time-driver-any,unstable-traits \ | ||||
|     --- build --release --manifest-path embassy-stm32/Cargo.toml --target thumbv7em-none-eabi --features stm32wb55vy,defmt,exti,time-driver-any,unstable-traits \ | ||||
|     --- build --release --manifest-path embassy-stm32/Cargo.toml --target thumbv7em-none-eabi --features stm32wl55cc-cm4,defmt,exti,time-driver-any,unstable-traits \ | ||||
|     --- build --release --manifest-path embassy-stm32/Cargo.toml --target thumbv7em-none-eabi --features stm32l4r9zi,defmt,exti,time-driver-any,unstable-traits \ | ||||
|     --- build --release --manifest-path embassy-stm32/Cargo.toml --target thumbv7em-none-eabi --features stm32f303vc,defmt,exti,time-driver-any,unstable-traits \ | ||||
|     --- build --release --manifest-path embassy-stm32/Cargo.toml --target thumbv7em-none-eabi --features stm32f411ce,defmt,time-driver-any \ | ||||
|     --- build --release --manifest-path embassy-stm32/Cargo.toml --target thumbv7em-none-eabi --features stm32f411ce,defmt,time-driver-any,unstable-traits \ | ||||
|     --- build --release --manifest-path embassy-stm32/Cargo.toml --target thumbv7em-none-eabi --features stm32f429zi,log,time-driver-any \ | ||||
|     --- build --release --manifest-path embassy-stm32/Cargo.toml --target thumbv7em-none-eabi --features stm32f429zi,log,time-driver-any,unstable-traits \ | ||||
|     --- build --release --manifest-path embassy-stm32/Cargo.toml --target thumbv7em-none-eabi --features stm32h755zi-cm7,defmt,time-driver-any \ | ||||
|     --- build --release --manifest-path embassy-stm32/Cargo.toml --target thumbv7em-none-eabi --features stm32h755zi-cm7,defmt,time-driver-any,unstable-traits \ | ||||
|     --- build --release --manifest-path embassy-stm32/Cargo.toml --target thumbv7em-none-eabi --features stm32l476vg,defmt,time-driver-any \ | ||||
|     --- build --release --manifest-path embassy-stm32/Cargo.toml --target thumbv7em-none-eabi --features stm32l476vg,defmt,time-driver-any,unstable-traits \ | ||||
|     --- build --release --manifest-path embassy-stm32/Cargo.toml --target thumbv6m-none-eabi --features stm32l072cz,defmt,time-driver-any \ | ||||
|     --- build --release --manifest-path embassy-stm32/Cargo.toml --target thumbv6m-none-eabi --features stm32l072cz,defmt,time-driver-any,unstable-traits \ | ||||
|     --- build --release --manifest-path embassy-stm32/Cargo.toml --target thumbv7m-none-eabi --features stm32l151cb-a,defmt,time-driver-any \ | ||||
|     --- build --release --manifest-path embassy-stm32/Cargo.toml --target thumbv7m-none-eabi --features stm32l151cb-a,defmt,time-driver-any,unstable-traits \ | ||||
|     --- build --release --manifest-path embassy-stm32/Cargo.toml --target thumbv7em-none-eabi --features stm32f410tb,defmt,exti,time-driver-any \ | ||||
|     --- build --release --manifest-path embassy-stm32/Cargo.toml --target thumbv7em-none-eabi --features stm32f410tb,defmt,exti,time-driver-any,unstable-traits \ | ||||
|     --- build --release --manifest-path embassy-stm32/Cargo.toml --target thumbv7em-none-eabi --features stm32f429zi,log,exti,time-driver-any \ | ||||
|     --- build --release --manifest-path embassy-stm32/Cargo.toml --target thumbv7em-none-eabi --features stm32f429zi,log,exti,time-driver-any,unstable-traits \ | ||||
|     --- build --release --manifest-path embassy-stm32/Cargo.toml --target thumbv7em-none-eabi --features stm32h755zi-cm7,defmt,exti,time-driver-any \ | ||||
|     --- build --release --manifest-path embassy-stm32/Cargo.toml --target thumbv7em-none-eabi --features stm32h755zi-cm7,defmt,exti,time-driver-any,unstable-traits \ | ||||
|     --- build --release --manifest-path embassy-stm32/Cargo.toml --target thumbv7em-none-eabi --features stm32l476vg,defmt,exti,time-driver-any \ | ||||
|     --- build --release --manifest-path embassy-stm32/Cargo.toml --target thumbv7em-none-eabi --features stm32l476vg,defmt,exti,time-driver-any,unstable-traits \ | ||||
|     --- build --release --manifest-path embassy-stm32/Cargo.toml --target thumbv6m-none-eabi --features stm32l072cz,defmt,exti,time-driver-any \ | ||||
|     --- build --release --manifest-path embassy-stm32/Cargo.toml --target thumbv6m-none-eabi --features stm32l072cz,defmt,exti,time-driver-any,unstable-traits \ | ||||
|     --- build --release --manifest-path embassy-stm32/Cargo.toml --target thumbv7m-none-eabi --features stm32l151cb-a,defmt,exti,time-driver-any \ | ||||
|     --- build --release --manifest-path embassy-stm32/Cargo.toml --target thumbv7m-none-eabi --features stm32l151cb-a,defmt,exti,time-driver-any,unstable-traits \ | ||||
|     --- build --release --manifest-path embassy-stm32/Cargo.toml --target thumbv7m-none-eabi --features stm32f217zg,defmt,exti,time-driver-any \ | ||||
|     --- build --release --manifest-path embassy-stm32/Cargo.toml --target thumbv7m-none-eabi --features stm32f217zg,defmt,exti,time-driver-any,unstable-traits \ | ||||
|     --- build --release --manifest-path examples/nrf52840/Cargo.toml --target thumbv7em-none-eabi --no-default-features --out-dir out/examples/nrf52840 --bin raw_spawn \ | ||||
|     --- build --release --manifest-path examples/stm32l0/Cargo.toml --target thumbv6m-none-eabi --no-default-features --out-dir out/examples/stm32l0 --bin raw_spawn \ | ||||
|   | ||||
| @@ -3,16 +3,16 @@ authors = ["Dario Nieuwenhuis <dirbaio@dirbaio.net>"] | ||||
| edition = "2018" | ||||
| name = "embassy-basic-example" | ||||
| version = "0.1.0" | ||||
| license = "MIT OR Apache-2.0" | ||||
|  | ||||
| [dependencies] | ||||
| embassy-executor = { version = "0.1.0", path = "../../../../../embassy-executor", features = ["defmt", "nightly"] } | ||||
| embassy-executor = { version = "0.2.0", path = "../../../../../embassy-executor", features = ["defmt", "nightly", "integrated-timers", "arch-cortex-m", "executor-thread"] } | ||||
| embassy-time = { version = "0.1.0", path = "../../../../../embassy-time", features = ["defmt", "nightly"] } | ||||
| embassy-nrf = { version = "0.1.0", path = "../../../../../embassy-nrf", features = ["defmt", "nrf52840", "time-driver-rtc1", "gpiote", "nightly"] } | ||||
|  | ||||
| defmt = "0.3" | ||||
| defmt-rtt = "0.3" | ||||
|  | ||||
| cortex-m = "0.7.3" | ||||
| cortex-m = { version = "0.7.6", features = ["critical-section-single-core"] } | ||||
| cortex-m-rt = "0.7.0" | ||||
| embedded-hal = "0.2.6" | ||||
| panic-probe = { version = "0.3", features = ["print-defmt"] } | ||||
|   | ||||
| @@ -10,7 +10,6 @@ members = [ | ||||
| [patch.crates-io] | ||||
| embassy-executor = { path = "../../../../../embassy-executor" } | ||||
| embassy-stm32 = { path = "../../../../../embassy-stm32" } | ||||
| stm32-metapac = { path = "../../../../../stm32-metapac" } | ||||
|  | ||||
| [profile.release] | ||||
| codegen-units = 1 | ||||
|   | ||||
| @@ -2,12 +2,13 @@ | ||||
| name = "blinky-async" | ||||
| version = "0.1.0" | ||||
| edition = "2021" | ||||
| license = "MIT OR Apache-2.0" | ||||
|  | ||||
| [dependencies] | ||||
| cortex-m = "0.7" | ||||
| cortex-m-rt = "0.7" | ||||
| embassy-stm32 = { version = "0.1.0", features = ["stm32l475vg", "memory-x", "exti"], default-features = false  } | ||||
| embassy-executor = { version = "0.1.0", default-features = false, features = ["nightly"] } | ||||
| embassy-executor = { version = "0.2.0", default-features = false, features = ["nightly", "arch-cortex-m", "executor-thread"] } | ||||
|  | ||||
| defmt = "0.3.0" | ||||
| defmt-rtt = "0.3.0" | ||||
|   | ||||
| @@ -2,6 +2,7 @@ | ||||
| name = "blinky-hal" | ||||
| version = "0.1.0" | ||||
| edition = "2021" | ||||
| license = "MIT OR Apache-2.0" | ||||
|  | ||||
| [dependencies] | ||||
| cortex-m = "0.7" | ||||
|   | ||||
| @@ -2,6 +2,7 @@ | ||||
| name = "blinky-irq" | ||||
| version = "0.1.0" | ||||
| edition = "2021" | ||||
| license = "MIT OR Apache-2.0" | ||||
|  | ||||
| [dependencies] | ||||
| cortex-m = "0.7" | ||||
|   | ||||
| @@ -2,11 +2,12 @@ | ||||
| name = "blinky-pac" | ||||
| version = "0.1.0" | ||||
| edition = "2021" | ||||
| license = "MIT OR Apache-2.0" | ||||
|  | ||||
| [dependencies] | ||||
| cortex-m = "0.7" | ||||
| cortex-m-rt = "0.7" | ||||
| stm32-metapac = { version = "0.1.0", features = ["stm32l475vg", "memory-x"] } | ||||
| stm32-metapac = { version = "1", features = ["stm32l475vg", "memory-x"] } | ||||
|  | ||||
| defmt = "0.3.0" | ||||
| defmt-rtt = "0.3.0" | ||||
|   | ||||
| @@ -21,7 +21,7 @@ Then, what follows are some declarations on how to deal with panics and faults. | ||||
|  | ||||
| [source,rust] | ||||
| ---- | ||||
| include::example$basic/src/main.rs[lines="11..12"] | ||||
| include::example$basic/src/main.rs[lines="10"] | ||||
| ---- | ||||
|  | ||||
| === Task declaration | ||||
| @@ -30,7 +30,7 @@ After a bit of import declaration, the tasks run by the application should be de | ||||
|  | ||||
| [source,rust] | ||||
| ---- | ||||
| include::example$basic/src/main.rs[lines="13..22"] | ||||
| include::example$basic/src/main.rs[lines="12..20"] | ||||
| ---- | ||||
|  | ||||
| An embassy task must be declared `async`, and may NOT take generic arguments. In this case, we are handed the LED that should be blinked and the interval of the blinking. | ||||
| @@ -45,23 +45,10 @@ The `Spawner` is the way the main application spawns other tasks. The `Periphera | ||||
|  | ||||
| [source,rust] | ||||
| ---- | ||||
| include::example$basic/src/main.rs[lines="23..-1"] | ||||
| include::example$basic/src/main.rs[lines="22..-1"] | ||||
| ---- | ||||
|  | ||||
| `#[embassy_executor::main]` takes an optional `config` parameter specifying a function that returns an instance of HAL's `Config` struct. For example: | ||||
|  | ||||
| ```rust | ||||
| fn embassy_config() -> embassy_nrf::config::Config { | ||||
|     embassy_nrf::config::Config::default() | ||||
| } | ||||
|  | ||||
| #[embassy_executor::main(config = "embassy_config()")] | ||||
| async fn main(_spawner: Spawner, p: embassy_nrf::Peripherals) { | ||||
|     // ... | ||||
| } | ||||
| ``` | ||||
|  | ||||
| What happens when the `blinker` task have been spawned and main returns? Well, the main entry point is actually just like any other task, except that you can only have one and it takes some specific type arguments. The magic lies within the `#[embassy::main]` macro. The macro does the following: | ||||
| What happens when the `blinker` task has been spawned and main returns? Well, the main entry point is actually just like any other task, except that you can only have one and it takes some specific type arguments. The magic lies within the `#[embassy::main]` macro. The macro does the following: | ||||
|  | ||||
| . Creates an Embassy Executor | ||||
| . Initializes the microcontroller HAL to get the `Peripherals` | ||||
| @@ -76,7 +63,7 @@ The project definition needs to contain the embassy dependencies: | ||||
|  | ||||
| [source,toml] | ||||
| ---- | ||||
| include::example$basic/Cargo.toml[lines="8..9"] | ||||
| include::example$basic/Cargo.toml[lines="9..11"] | ||||
| ---- | ||||
|  | ||||
| Depending on your microcontroller, you may need to replace `embassy-nrf` with something else (`embassy-stm32` for STM32. Remember to update feature flags as well). | ||||
|   | ||||
| @@ -6,7 +6,7 @@ The bootloader can be used either as a library or be flashed directly if you are | ||||
|  | ||||
| By design, the bootloader does not provide any network capabilities. Networking capabilities for fetching new firmware can be provided by the user application, using the bootloader as a library for updating the firmware, or by using the bootloader as a library and adding this capability yourself. | ||||
|  | ||||
| The bootloader supports both internal and external flash by relying on the `embedded-storage` traits. | ||||
| The bootloader supports both internal and external flash by relying on the `embedded-storage` traits. The bootloader optionally supports the verification of firmware that has been digitally signed (recommended). | ||||
|  | ||||
|  | ||||
| == Hardware support | ||||
| @@ -15,6 +15,7 @@ The bootloader supports | ||||
|  | ||||
| * nRF52 with and without softdevice | ||||
| * STM32 L4, WB, WL, L1, L0, F3, F7 and H7 | ||||
| * Raspberry Pi: RP2040 | ||||
|  | ||||
| In general, the bootloader works on any platform that implements the `embedded-storage` traits for its internal flash, but may require custom initialization code to work. | ||||
|  | ||||
| @@ -25,12 +26,69 @@ image::bootloader_flash.png[Bootloader flash layout] | ||||
| The bootloader divides the storage into 4 main partitions, configurable when creating the bootloader | ||||
| instance or via linker scripts: | ||||
|  | ||||
| * BOOTLOADER - Where the bootloader is placed. The bootloader itself consumes about 8kB of flash. | ||||
| * ACTIVE - Where the main application is placed. The bootloader will attempt to load the application at the start of this partition. This partition is only written to by the bootloader. | ||||
| * DFU - Where the application-to-be-swapped is placed. This partition is written to by the application. | ||||
| * BOOTLOADER STATE - Where the bootloader stores the current state describing if the active and dfu partitions need to be swapped. When the new firmware has been written to the DFU partition, a flag is set to instruct the bootloader that the partitions should be swapped. | ||||
| * BOOTLOADER - Where the bootloader is placed. The bootloader itself consumes about 8kB of flash, but if you need to debug it and have space available, increasing this to 24kB will allow you to run the bootloader with probe-rs. | ||||
| * ACTIVE - Where the main application is placed. The bootloader will attempt to load the application at the start of this partition. This partition is only written to by the bootloader. The size required for this partition depends on the size of your application. | ||||
| * DFU - Where the application-to-be-swapped is placed. This partition is written to by the application. This partition must be at least 1 page bigger than the ACTIVE partition, since the swap algorithm uses the extra space to ensure power safe copy of data: | ||||
| + | ||||
| Partition Size~dfu~= Partition Size~active~+ Page Size~active~ | ||||
| + | ||||
| All values are specified in bytes. | ||||
|  | ||||
| * BOOTLOADER STATE - Where the bootloader stores the current state describing if the active and dfu partitions need to be swapped. When the new firmware has been written to the DFU partition, a magic field is written to instruct the bootloader that the partitions should be swapped. This partition must be able to store a magic field as well as the partition swap progress. The partition size given by: | ||||
| + | ||||
| Partition Size~state~ = Write Size~state~ + (2 × Partition Size~active~ / Page Size~active~) | ||||
| + | ||||
| All values are specified in bytes. | ||||
|  | ||||
| The partitions for ACTIVE (+BOOTLOADER), DFU and BOOTLOADER_STATE may be placed in separate flash. The page size used by the bootloader is determined by the lowest common multiple of the ACTIVE and DFU page sizes. | ||||
| The BOOTLOADER_STATE partition must be big enough to store one word per page in the ACTIVE and DFU partitions combined. | ||||
|  | ||||
| The bootloader has a platform-agnostic part, which implements the power fail safe swapping algorithm given the boundaries set by the partitions. The platform-specific part is a minimal shim that provides additional functionality such as watchdogs or supporting the nRF52 softdevice. | ||||
|  | ||||
| === FirmwareUpdater | ||||
|  | ||||
| The `FirmwareUpdater` is an object for conveniently flashing firmware to the DFU partition and subsequently marking it as being ready for swapping with the active partition on the next reset. Its principle methods are `write_firmware`, which is called once per the size of the flash "write block" (typically 4KiB), and `mark_updated`, which is the final call. | ||||
|  | ||||
| === Verification | ||||
|  | ||||
| The bootloader supports the verification of firmware that has been flashed to the DFU partition. Verification requires that firmware has been signed digitally using link:https://ed25519.cr.yp.to/[`ed25519`] signatures. With verification enabled, the `FirmwareUpdater::verify_and_mark_updated` method is called in place of `mark_updated`. A public key and signature are required, along with the actual length of the firmware that has been flashed. If verification fails then the firmware will not be marked as updated and therefore be rejected. | ||||
|  | ||||
| Signatures are normally conveyed with the firmware to be updated and not written to flash. How signatures are provided is a firmware responsibility. | ||||
|  | ||||
| To enable verification use either the `ed25519-dalek` or `ed25519-salty` features when depending on the `embassy-boot` crate. We recommend `ed25519-salty` at this time due to its small size. | ||||
|  | ||||
| ==== Tips on keys and signing with ed25519 | ||||
|  | ||||
| Ed25519 is a public key signature system where you are responsible for keeping the private key secure. We recommend embedding the *public* key in your program so that it can be easily passed to `verify_and_mark_updated`. An example declaration of the public key in your firmware: | ||||
|  | ||||
| [source, rust] | ||||
| ---- | ||||
| static PUBLIC_SIGNING_KEY: &[u8] = include_bytes!("key.pub"); | ||||
| ---- | ||||
|  | ||||
| Signatures are often conveyed along with firmware by appending them. | ||||
|  | ||||
| Ed25519 keys can be generated by a variety of tools. We recommend link:https://man.openbsd.org/signify[`signify`] as it is in wide use to sign and verify OpenBSD distributions, and is straightforward to use. | ||||
|  | ||||
| The following set of Bash commands can be used to generate public and private keys on Unix platforms, and also generate a local `key.pub` file with the `signify` file headers removed. Declare a `SECRETS_DIR` environment variable in a secure location. | ||||
|  | ||||
| [source, bash] | ||||
| ---- | ||||
| signify -G -n -p $SECRETS_DIR/key.pub -s $SECRETS_DIR/key.sec | ||||
| tail -n1 $SECRETS_DIR/key.pub | base64 -d -i - | dd ibs=10 skip=1 > key.pub | ||||
| chmod 700 $SECRETS_DIR/key.sec | ||||
| export SECRET_SIGNING_KEY=$(tail -n1 $SECRETS_DIR/key.sec) | ||||
| ---- | ||||
|  | ||||
| Then, to sign your firmware given a declaration of `FIRMWARE_DIR` and a firmware filename of `myfirmware`: | ||||
|  | ||||
| [source, bash] | ||||
| ---- | ||||
| shasum -a 512 -b $FIRMWARE_DIR/myfirmware > $SECRETS_DIR/message.txt | ||||
| cat $SECRETS_DIR/message.txt | dd ibs=128 count=1 | xxd -p -r > $SECRETS_DIR/message.txt | ||||
| signify -S -s $SECRETS_DIR/key.sec -m $SECRETS_DIR/message.txt -x $SECRETS_DIR/message.txt.sig | ||||
| cp $FIRMWARE_DIR/myfirmware $FIRMWARE_DIR/myfirmware+signed | ||||
| tail -n1 $SECRETS_DIR/message.txt.sig | base64 -d -i - | dd ibs=10 skip=1 >> $FIRMWARE_DIR/myfirmware+signed | ||||
| ---- | ||||
|  | ||||
| Remember, guard the `$SECRETS_DIR/key.sec` key as compromising it means that another party can sign your firmware. | ||||
| @@ -45,7 +45,7 @@ You can run an example by opening a terminal and entering the following commands | ||||
|  | ||||
| [source, bash] | ||||
| ---- | ||||
| cd examples/nrf | ||||
| cd examples/nrf52840 | ||||
| cargo run --bin blinky --release | ||||
| ---- | ||||
|  | ||||
|   | ||||
| @@ -8,7 +8,7 @@ The application we'll write is a simple 'push button, blink led' application, wh | ||||
|  | ||||
| == PAC version | ||||
|  | ||||
| The PAC is the lowest API for accessing peripherals and registers, if you don't count reading/writing directly to memory addresses. It provide distinct types | ||||
| The PAC is the lowest API for accessing peripherals and registers, if you don't count reading/writing directly to memory addresses. It provides distinct types | ||||
| to make accessing peripheral registers easier, but it does not prevent you from writing unsafe code. | ||||
|  | ||||
| Writing an application using the PAC directly is therefore not recommended, but if the functionality you want to use is not exposed in the upper layers, that's what you need to use. | ||||
| @@ -20,13 +20,13 @@ The blinky app using PAC is shown below: | ||||
| include::example$layer-by-layer/blinky-pac/src/main.rs[] | ||||
| ---- | ||||
|  | ||||
| As you can see, there are a lot of code needed to enable the peripheral clocks, configuring the input pins and the output pins of the application. | ||||
| As you can see, a lot of code is needed to enable the peripheral clocks and to configure the input pins and the output pins of the application. | ||||
|  | ||||
| Another downside of this application is that it is busy-looping while polling the button state. This prevents the microcontroller from utilizing any sleep mode to save power. | ||||
|  | ||||
| == HAL version | ||||
|  | ||||
| To simplify our application, we can use the HAL instead. The HAL exposes higher level APIs that handle details such | ||||
| To simplify our application, we can use the HAL instead. The HAL exposes higher level APIs that handle details such as: | ||||
|  | ||||
| * Automatically enabling the peripheral clock when you're using the peripheral | ||||
| * Deriving and applying register configuration from higher level types | ||||
| @@ -39,7 +39,7 @@ The HAL example is shown below: | ||||
| include::example$layer-by-layer/blinky-hal/src/main.rs[] | ||||
| ---- | ||||
|  | ||||
| As you can see, the application becomes a lot simpler, even without using any async code. The `Input` and `Output` hides all the details accessing the GPIO registers, and allow you to use a much simpler API to query the state of the button and toggle the LED output accordingly. | ||||
| As you can see, the application becomes a lot simpler, even without using any async code. The `Input` and `Output` types hide all the details of accessing the GPIO registers and allow you to use a much simpler API for querying the state of the button and toggling the LED output. | ||||
|  | ||||
| The same downside from the PAC example still applies though: the application is busy looping and consuming more power than necessary. | ||||
|  | ||||
|   | ||||
| @@ -20,7 +20,7 @@ IMPORTANT: The executor relies on tasks not blocking indefinitely, as this preve | ||||
|  | ||||
| image::embassy_executor.png[Executor model] | ||||
|  | ||||
| If you use the `#[embassy::main]` macro in your application, it creates the `Executor` for you and spawns the main entry point as the first task. You can also create the Executor manually, and you can in fact create multiple Executors. | ||||
| If you use the `#[embassy_executor::main]` macro in your application, it creates the `Executor` for you and spawns the main entry point as the first task. You can also create the Executor manually, and you can in fact create multiple Executors. | ||||
|  | ||||
|  | ||||
| == Interrupts | ||||
|   | ||||
| @@ -4,9 +4,9 @@ The link:https://github.com/embassy-rs/embassy/tree/master/embassy-stm32[Embassy | ||||
|  | ||||
| == The infinite variant problem | ||||
|  | ||||
| STM32 microcontrollers comes in many families and flavors, and supporting all of them is a big undertaking. Embassy has taken advantage of the fact | ||||
| STM32 microcontrollers come in many families, and flavors and supporting all of them is a big undertaking. Embassy has taken advantage of the fact | ||||
| that the STM32 peripheral versions are shared across chip families. Instead of re-implementing the SPI | ||||
| peripheral for every STM32 chip family, embassy have a single SPI implementation that depends on | ||||
| peripheral for every STM32 chip family, embassy has a single SPI implementation that depends on | ||||
| code-generated register types that are identical for STM32 families with the same version of a given peripheral. | ||||
|  | ||||
| === The metapac | ||||
|   | ||||
| @@ -1,25 +1,54 @@ | ||||
| [package] | ||||
| edition = "2021" | ||||
| name = "embassy-boot" | ||||
| version = "0.1.0" | ||||
| description = "Bootloader using Embassy" | ||||
| version = "0.1.1" | ||||
| description = "A lightweight bootloader supporting firmware updates in a power-fail-safe way, with trial boots and rollbacks." | ||||
| license = "MIT OR Apache-2.0" | ||||
| repository = "https://github.com/embassy-rs/embassy" | ||||
| categories = [ | ||||
|     "embedded", | ||||
|     "no-std", | ||||
|     "asynchronous", | ||||
| ] | ||||
|  | ||||
| [package.metadata.embassy_docs] | ||||
| src_base = "https://github.com/embassy-rs/embassy/blob/embassy-boot-v$VERSION/embassy-boot/boot/src/" | ||||
| src_base_git = "https://github.com/embassy-rs/embassy/blob/$COMMIT/embassy-boot/boot/src/" | ||||
| target = "thumbv7em-none-eabi" | ||||
| features = ["defmt"] | ||||
|  | ||||
| [package.metadata.docs.rs] | ||||
| features = ["defmt"] | ||||
|  | ||||
| [lib] | ||||
|  | ||||
| [dependencies] | ||||
| defmt = { version = "0.3", optional = true } | ||||
| digest = "0.10" | ||||
| log = { version = "0.4", optional = true  } | ||||
| embassy-sync = { version = "0.1.0", path = "../../embassy-sync" } | ||||
| ed25519-dalek = { version = "1.0.1", default_features = false, features = ["u32_backend"], optional = true } | ||||
| embassy-sync = { version = "0.2.0", path = "../../embassy-sync" } | ||||
| embedded-storage = "0.3.0" | ||||
| embedded-storage-async = "0.3.0" | ||||
| embedded-storage-async = { version = "0.4.0", optional = true} | ||||
| salty = { git = "https://github.com/ycrypto/salty.git", rev = "a9f17911a5024698406b75c0fac56ab5ccf6a8c7", optional = true } | ||||
| signature = { version = "1.6.4", default-features = false } | ||||
|  | ||||
| [dev-dependencies] | ||||
| log = "0.4" | ||||
| env_logger = "0.9" | ||||
| rand = "0.8" | ||||
| rand = "0.7" # ed25519-dalek v1.0.1 depends on this exact version | ||||
| futures = { version = "0.3", features = ["executor"] } | ||||
| sha1 = "0.10.5" | ||||
|  | ||||
| [dev-dependencies.ed25519-dalek] | ||||
| default_features = false | ||||
| features = ["rand", "std", "u32_backend"] | ||||
|  | ||||
| [features] | ||||
| ed25519-dalek = ["dep:ed25519-dalek", "_verify"] | ||||
| ed25519-salty = ["dep:salty", "_verify"] | ||||
|  | ||||
| nightly = ["dep:embedded-storage-async"] | ||||
|  | ||||
| #Internal features | ||||
| _verify = [] | ||||
|   | ||||
							
								
								
									
										31
									
								
								embassy-boot/boot/README.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										31
									
								
								embassy-boot/boot/README.md
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,31 @@ | ||||
| # embassy-boot | ||||
|  | ||||
| An [Embassy](https://embassy.dev) project. | ||||
|  | ||||
| A lightweight bootloader supporting firmware updates in a power-fail-safe way, with trial boots and rollbacks. | ||||
|  | ||||
| The bootloader can be used either as a library or be flashed directly with the default configuration derived from linker scripts. | ||||
|  | ||||
| By design, the bootloader does not provide any network capabilities. Networking capabilities for fetching new firmware can be provided by the user application, using the bootloader as a library for updating the firmware, or by using the bootloader as a library and adding this capability yourself. | ||||
|  | ||||
| ## Hardware support | ||||
|  | ||||
| The bootloader supports different hardware in separate crates: | ||||
|  | ||||
| * `embassy-boot-nrf` - for the nRF microcontrollers. | ||||
| * `embassy-boot-rp` - for the RP2040 microcontrollers. | ||||
| * `embassy-boot-stm32` - for the STM32 microcontrollers. | ||||
|  | ||||
| ## Minimum supported Rust version (MSRV) | ||||
|  | ||||
| `embassy-boot` is guaranteed to compile on the latest stable Rust version at the time of release. It might compile with older versions but that may change in any new patch release. | ||||
|  | ||||
| ## License | ||||
|  | ||||
| This work is licensed under either of | ||||
|  | ||||
| - Apache License, Version 2.0 ([LICENSE-APACHE](LICENSE-APACHE) or | ||||
|   <http://www.apache.org/licenses/LICENSE-2.0>) | ||||
| - MIT license ([LICENSE-MIT](LICENSE-MIT) or <http://opensource.org/licenses/MIT>) | ||||
|  | ||||
| at your option. | ||||
							
								
								
									
										533
									
								
								embassy-boot/boot/src/boot_loader.rs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										533
									
								
								embassy-boot/boot/src/boot_loader.rs
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,533 @@ | ||||
| use embedded_storage::nor_flash::{ErrorType, NorFlash, NorFlashError, NorFlashErrorKind, ReadNorFlash}; | ||||
|  | ||||
| use crate::{Partition, State, BOOT_MAGIC, SWAP_MAGIC}; | ||||
|  | ||||
| /// Errors returned by bootloader | ||||
| #[derive(PartialEq, Eq, Debug)] | ||||
| pub enum BootError { | ||||
|     /// Error from flash. | ||||
|     Flash(NorFlashErrorKind), | ||||
|     /// Invalid bootloader magic | ||||
|     BadMagic, | ||||
| } | ||||
|  | ||||
| #[cfg(feature = "defmt")] | ||||
| impl defmt::Format for BootError { | ||||
|     fn format(&self, fmt: defmt::Formatter) { | ||||
|         match self { | ||||
|             BootError::Flash(_) => defmt::write!(fmt, "BootError::Flash(_)"), | ||||
|             BootError::BadMagic => defmt::write!(fmt, "BootError::BadMagic"), | ||||
|         } | ||||
|     } | ||||
| } | ||||
|  | ||||
| impl<E> From<E> for BootError | ||||
| where | ||||
|     E: NorFlashError, | ||||
| { | ||||
|     fn from(error: E) -> Self { | ||||
|         BootError::Flash(error.kind()) | ||||
|     } | ||||
| } | ||||
|  | ||||
| /// Trait defining the flash handles used for active and DFU partition. | ||||
| pub trait FlashConfig { | ||||
|     /// The erase value of the state flash. Typically the default of 0xFF is used, but some flashes use a different value. | ||||
|     const STATE_ERASE_VALUE: u8 = 0xFF; | ||||
|     /// Flash type used for the state partition. | ||||
|     type STATE: NorFlash; | ||||
|     /// Flash type used for the active partition. | ||||
|     type ACTIVE: NorFlash; | ||||
|     /// Flash type used for the dfu partition. | ||||
|     type DFU: NorFlash; | ||||
|  | ||||
|     /// Return flash instance used to write/read to/from active partition. | ||||
|     fn active(&mut self) -> &mut Self::ACTIVE; | ||||
|     /// Return flash instance used to write/read to/from dfu partition. | ||||
|     fn dfu(&mut self) -> &mut Self::DFU; | ||||
|     /// Return flash instance used to write/read to/from bootloader state. | ||||
|     fn state(&mut self) -> &mut Self::STATE; | ||||
| } | ||||
|  | ||||
| trait FlashConfigEx { | ||||
|     fn page_size() -> u32; | ||||
| } | ||||
|  | ||||
| impl<T: FlashConfig> FlashConfigEx for T { | ||||
|     /// Get the page size which is the "unit of operation" within the bootloader. | ||||
|     fn page_size() -> u32 { | ||||
|         core::cmp::max(T::ACTIVE::ERASE_SIZE, T::DFU::ERASE_SIZE) as u32 | ||||
|     } | ||||
| } | ||||
|  | ||||
| /// BootLoader works with any flash implementing embedded_storage. | ||||
| pub struct BootLoader { | ||||
|     // Page with current state of bootloader. The state partition has the following format: | ||||
|     // All ranges are in multiples of WRITE_SIZE bytes. | ||||
|     // | Range    | Description                                                                      | | ||||
|     // | 0..1     | Magic indicating bootloader state. BOOT_MAGIC means boot, SWAP_MAGIC means swap. | | ||||
|     // | 1..2     | Progress validity. ERASE_VALUE means valid, !ERASE_VALUE means invalid.          | | ||||
|     // | 2..2 + N | Progress index used while swapping or reverting                                  | | ||||
|     state: Partition, | ||||
|     // Location of the partition which will be booted from | ||||
|     active: Partition, | ||||
|     // Location of the partition which will be swapped in when requested | ||||
|     dfu: Partition, | ||||
| } | ||||
|  | ||||
| impl BootLoader { | ||||
|     /// Create a new instance of a bootloader with the given partitions. | ||||
|     /// | ||||
|     /// - All partitions must be aligned with the PAGE_SIZE const generic parameter. | ||||
|     /// - The dfu partition must be at least PAGE_SIZE bigger than the active partition. | ||||
|     pub fn new(active: Partition, dfu: Partition, state: Partition) -> Self { | ||||
|         Self { active, dfu, state } | ||||
|     } | ||||
|  | ||||
|     /// Return the offset of the active partition into the active flash. | ||||
|     pub fn boot_address(&self) -> usize { | ||||
|         self.active.from as usize | ||||
|     } | ||||
|  | ||||
|     /// Perform necessary boot preparations like swapping images. | ||||
|     /// | ||||
|     /// The DFU partition is assumed to be 1 page bigger than the active partition for the swap | ||||
|     /// algorithm to work correctly. | ||||
|     /// | ||||
|     /// The provided aligned_buf argument must satisfy any alignment requirements | ||||
|     /// given by the partition flashes. All flash operations will use this buffer. | ||||
|     /// | ||||
|     /// SWAPPING | ||||
|     /// | ||||
|     /// Assume a flash size of 3 pages for the active partition, and 4 pages for the DFU partition. | ||||
|     /// The swap index contains the copy progress, as to allow continuation of the copy process on | ||||
|     /// power failure. The index counter is represented within 1 or more pages (depending on total | ||||
|     /// flash size), where a page X is considered swapped if index at location (X + WRITE_SIZE) | ||||
|     /// contains a zero value. This ensures that index updates can be performed atomically and | ||||
|     /// avoid a situation where the wrong index value is set (page write size is "atomic"). | ||||
|     /// | ||||
|     /// +-----------+------------+--------+--------+--------+--------+ | ||||
|     /// | Partition | Swap Index | Page 0 | Page 1 | Page 3 | Page 4 | | ||||
|     /// +-----------+------------+--------+--------+--------+--------+ | ||||
|     /// |    Active |          0 |      1 |      2 |      3 |      - | | ||||
|     /// |       DFU |          0 |      3 |      2 |      1 |      X | | ||||
|     /// +-----------+------------+--------+--------+--------+--------+ | ||||
|     /// | ||||
|     /// The algorithm starts by copying 'backwards', and after the first step, the layout is | ||||
|     /// as follows: | ||||
|     /// | ||||
|     /// +-----------+------------+--------+--------+--------+--------+ | ||||
|     /// | Partition | Swap Index | Page 0 | Page 1 | Page 3 | Page 4 | | ||||
|     /// +-----------+------------+--------+--------+--------+--------+ | ||||
|     /// |    Active |          1 |      1 |      2 |      1 |      - | | ||||
|     /// |       DFU |          1 |      3 |      2 |      1 |      3 | | ||||
|     /// +-----------+------------+--------+--------+--------+--------+ | ||||
|     /// | ||||
|     /// The next iteration performs the same steps | ||||
|     /// | ||||
|     /// +-----------+------------+--------+--------+--------+--------+ | ||||
|     /// | Partition | Swap Index | Page 0 | Page 1 | Page 3 | Page 4 | | ||||
|     /// +-----------+------------+--------+--------+--------+--------+ | ||||
|     /// |    Active |          2 |      1 |      2 |      1 |      - | | ||||
|     /// |       DFU |          2 |      3 |      2 |      2 |      3 | | ||||
|     /// +-----------+------------+--------+--------+--------+--------+ | ||||
|     /// | ||||
|     /// And again until we're done | ||||
|     /// | ||||
|     /// +-----------+------------+--------+--------+--------+--------+ | ||||
|     /// | Partition | Swap Index | Page 0 | Page 1 | Page 3 | Page 4 | | ||||
|     /// +-----------+------------+--------+--------+--------+--------+ | ||||
|     /// |    Active |          3 |      3 |      2 |      1 |      - | | ||||
|     /// |       DFU |          3 |      3 |      1 |      2 |      3 | | ||||
|     /// +-----------+------------+--------+--------+--------+--------+ | ||||
|     /// | ||||
|     /// REVERTING | ||||
|     /// | ||||
|     /// The reverting algorithm uses the swap index to discover that images were swapped, but that | ||||
|     /// the application failed to mark the boot successful. In this case, the revert algorithm will | ||||
|     /// run. | ||||
|     /// | ||||
|     /// The revert index is located separately from the swap index, to ensure that revert can continue | ||||
|     /// on power failure. | ||||
|     /// | ||||
|     /// The revert algorithm works forwards, by starting copying into the 'unused' DFU page at the start. | ||||
|     /// | ||||
|     /// +-----------+--------------+--------+--------+--------+--------+ | ||||
|     /// | Partition | Revert Index | Page 0 | Page 1 | Page 3 | Page 4 | | ||||
|     //*/ | ||||
|     /// +-----------+--------------+--------+--------+--------+--------+ | ||||
|     /// |    Active |            3 |      1 |      2 |      1 |      - | | ||||
|     /// |       DFU |            3 |      3 |      1 |      2 |      3 | | ||||
|     /// +-----------+--------------+--------+--------+--------+--------+ | ||||
|     /// | ||||
|     /// | ||||
|     /// +-----------+--------------+--------+--------+--------+--------+ | ||||
|     /// | Partition | Revert Index | Page 0 | Page 1 | Page 3 | Page 4 | | ||||
|     /// +-----------+--------------+--------+--------+--------+--------+ | ||||
|     /// |    Active |            3 |      1 |      2 |      1 |      - | | ||||
|     /// |       DFU |            3 |      3 |      2 |      2 |      3 | | ||||
|     /// +-----------+--------------+--------+--------+--------+--------+ | ||||
|     /// | ||||
|     /// +-----------+--------------+--------+--------+--------+--------+ | ||||
|     /// | Partition | Revert Index | Page 0 | Page 1 | Page 3 | Page 4 | | ||||
|     /// +-----------+--------------+--------+--------+--------+--------+ | ||||
|     /// |    Active |            3 |      1 |      2 |      3 |      - | | ||||
|     /// |       DFU |            3 |      3 |      2 |      1 |      3 | | ||||
|     /// +-----------+--------------+--------+--------+--------+--------+ | ||||
|     /// | ||||
|     pub fn prepare_boot<P: FlashConfig>(&mut self, p: &mut P, aligned_buf: &mut [u8]) -> Result<State, BootError> { | ||||
|         // Ensure we have enough progress pages to store copy progress | ||||
|         assert_eq!(0, P::page_size() % aligned_buf.len() as u32); | ||||
|         assert_eq!(0, P::page_size() % P::ACTIVE::WRITE_SIZE as u32); | ||||
|         assert_eq!(0, P::page_size() % P::ACTIVE::ERASE_SIZE as u32); | ||||
|         assert_eq!(0, P::page_size() % P::DFU::WRITE_SIZE as u32); | ||||
|         assert_eq!(0, P::page_size() % P::DFU::ERASE_SIZE as u32); | ||||
|         assert!(aligned_buf.len() >= P::STATE::WRITE_SIZE); | ||||
|         assert_eq!(0, aligned_buf.len() % P::ACTIVE::WRITE_SIZE); | ||||
|         assert_eq!(0, aligned_buf.len() % P::DFU::WRITE_SIZE); | ||||
|         assert_partitions(self.active, self.dfu, self.state, P::page_size(), P::STATE::WRITE_SIZE); | ||||
|  | ||||
|         // Copy contents from partition N to active | ||||
|         let state = self.read_state(p, aligned_buf)?; | ||||
|         if state == State::Swap { | ||||
|             // | ||||
|             // Check if we already swapped. If we're in the swap state, this means we should revert | ||||
|             // since the app has failed to mark boot as successful | ||||
|             // | ||||
|             if !self.is_swapped(p, aligned_buf)? { | ||||
|                 trace!("Swapping"); | ||||
|                 self.swap(p, aligned_buf)?; | ||||
|                 trace!("Swapping done"); | ||||
|             } else { | ||||
|                 trace!("Reverting"); | ||||
|                 self.revert(p, aligned_buf)?; | ||||
|  | ||||
|                 let state_flash = p.state(); | ||||
|                 let state_word = &mut aligned_buf[..P::STATE::WRITE_SIZE]; | ||||
|  | ||||
|                 // Invalidate progress | ||||
|                 state_word.fill(!P::STATE_ERASE_VALUE); | ||||
|                 self.state | ||||
|                     .write_blocking(state_flash, P::STATE::WRITE_SIZE as u32, state_word)?; | ||||
|  | ||||
|                 // Clear magic and progress | ||||
|                 self.state.wipe_blocking(state_flash)?; | ||||
|  | ||||
|                 // Set magic | ||||
|                 state_word.fill(BOOT_MAGIC); | ||||
|                 self.state.write_blocking(state_flash, 0, state_word)?; | ||||
|             } | ||||
|         } | ||||
|         Ok(state) | ||||
|     } | ||||
|  | ||||
|     fn is_swapped<P: FlashConfig>(&mut self, p: &mut P, aligned_buf: &mut [u8]) -> Result<bool, BootError> { | ||||
|         let page_count = (self.active.size() / P::page_size()) as usize; | ||||
|         let progress = self.current_progress(p, aligned_buf)?; | ||||
|  | ||||
|         Ok(progress >= page_count * 2) | ||||
|     } | ||||
|  | ||||
|     fn current_progress<P: FlashConfig>(&mut self, config: &mut P, aligned_buf: &mut [u8]) -> Result<usize, BootError> { | ||||
|         let write_size = P::STATE::WRITE_SIZE as u32; | ||||
|         let max_index = (((self.state.size() - write_size) / write_size) - 2) as usize; | ||||
|         let state_flash = config.state(); | ||||
|         let state_word = &mut aligned_buf[..write_size as usize]; | ||||
|  | ||||
|         self.state.read_blocking(state_flash, write_size, state_word)?; | ||||
|         if state_word.iter().any(|&b| b != P::STATE_ERASE_VALUE) { | ||||
|             // Progress is invalid | ||||
|             return Ok(max_index); | ||||
|         } | ||||
|  | ||||
|         for index in 0..max_index { | ||||
|             self.state | ||||
|                 .read_blocking(state_flash, (2 + index) as u32 * write_size, state_word)?; | ||||
|  | ||||
|             if state_word.iter().any(|&b| b == P::STATE_ERASE_VALUE) { | ||||
|                 return Ok(index); | ||||
|             } | ||||
|         } | ||||
|         Ok(max_index) | ||||
|     } | ||||
|  | ||||
|     fn update_progress<P: FlashConfig>( | ||||
|         &mut self, | ||||
|         progress_index: usize, | ||||
|         p: &mut P, | ||||
|         aligned_buf: &mut [u8], | ||||
|     ) -> Result<(), BootError> { | ||||
|         let state_word = &mut aligned_buf[..P::STATE::WRITE_SIZE]; | ||||
|         state_word.fill(!P::STATE_ERASE_VALUE); | ||||
|         self.state.write_blocking( | ||||
|             p.state(), | ||||
|             (2 + progress_index) as u32 * P::STATE::WRITE_SIZE as u32, | ||||
|             state_word, | ||||
|         )?; | ||||
|         Ok(()) | ||||
|     } | ||||
|  | ||||
|     fn copy_page_once_to_active<P: FlashConfig>( | ||||
|         &mut self, | ||||
|         progress_index: usize, | ||||
|         from_offset: u32, | ||||
|         to_offset: u32, | ||||
|         p: &mut P, | ||||
|         aligned_buf: &mut [u8], | ||||
|     ) -> Result<(), BootError> { | ||||
|         if self.current_progress(p, aligned_buf)? <= progress_index { | ||||
|             let page_size = P::page_size() as u32; | ||||
|  | ||||
|             self.active | ||||
|                 .erase_blocking(p.active(), to_offset, to_offset + page_size)?; | ||||
|  | ||||
|             for offset_in_page in (0..page_size).step_by(aligned_buf.len()) { | ||||
|                 self.dfu | ||||
|                     .read_blocking(p.dfu(), from_offset + offset_in_page as u32, aligned_buf)?; | ||||
|                 self.active | ||||
|                     .write_blocking(p.active(), to_offset + offset_in_page as u32, aligned_buf)?; | ||||
|             } | ||||
|  | ||||
|             self.update_progress(progress_index, p, aligned_buf)?; | ||||
|         } | ||||
|         Ok(()) | ||||
|     } | ||||
|  | ||||
|     fn copy_page_once_to_dfu<P: FlashConfig>( | ||||
|         &mut self, | ||||
|         progress_index: usize, | ||||
|         from_offset: u32, | ||||
|         to_offset: u32, | ||||
|         p: &mut P, | ||||
|         aligned_buf: &mut [u8], | ||||
|     ) -> Result<(), BootError> { | ||||
|         if self.current_progress(p, aligned_buf)? <= progress_index { | ||||
|             let page_size = P::page_size() as u32; | ||||
|  | ||||
|             self.dfu | ||||
|                 .erase_blocking(p.dfu(), to_offset as u32, to_offset + page_size)?; | ||||
|  | ||||
|             for offset_in_page in (0..page_size).step_by(aligned_buf.len()) { | ||||
|                 self.active | ||||
|                     .read_blocking(p.active(), from_offset + offset_in_page as u32, aligned_buf)?; | ||||
|                 self.dfu | ||||
|                     .write_blocking(p.dfu(), to_offset + offset_in_page as u32, aligned_buf)?; | ||||
|             } | ||||
|  | ||||
|             self.update_progress(progress_index, p, aligned_buf)?; | ||||
|         } | ||||
|         Ok(()) | ||||
|     } | ||||
|  | ||||
|     fn swap<P: FlashConfig>(&mut self, p: &mut P, aligned_buf: &mut [u8]) -> Result<(), BootError> { | ||||
|         let page_size = P::page_size(); | ||||
|         let page_count = self.active.size() / page_size; | ||||
|         for page_num in 0..page_count { | ||||
|             let progress_index = (page_num * 2) as usize; | ||||
|  | ||||
|             // Copy active page to the 'next' DFU page. | ||||
|             let active_from_offset = (page_count - 1 - page_num) * page_size; | ||||
|             let dfu_to_offset = (page_count - page_num) * page_size; | ||||
|             //trace!("Copy active {} to dfu {}", active_from_offset, dfu_to_offset); | ||||
|             self.copy_page_once_to_dfu(progress_index, active_from_offset, dfu_to_offset, p, aligned_buf)?; | ||||
|  | ||||
|             // Copy DFU page to the active page | ||||
|             let active_to_offset = (page_count - 1 - page_num) * page_size; | ||||
|             let dfu_from_offset = (page_count - 1 - page_num) * page_size; | ||||
|             //trace!("Copy dfy {} to active {}", dfu_from_offset, active_to_offset); | ||||
|             self.copy_page_once_to_active(progress_index + 1, dfu_from_offset, active_to_offset, p, aligned_buf)?; | ||||
|         } | ||||
|  | ||||
|         Ok(()) | ||||
|     } | ||||
|  | ||||
|     fn revert<P: FlashConfig>(&mut self, p: &mut P, aligned_buf: &mut [u8]) -> Result<(), BootError> { | ||||
|         let page_size = P::page_size(); | ||||
|         let page_count = self.active.size() / page_size; | ||||
|         for page_num in 0..page_count { | ||||
|             let progress_index = (page_count * 2 + page_num * 2) as usize; | ||||
|  | ||||
|             // Copy the bad active page to the DFU page | ||||
|             let active_from_offset = page_num * page_size; | ||||
|             let dfu_to_offset = page_num * page_size; | ||||
|             self.copy_page_once_to_dfu(progress_index, active_from_offset, dfu_to_offset, p, aligned_buf)?; | ||||
|  | ||||
|             // Copy the DFU page back to the active page | ||||
|             let active_to_offset = page_num * page_size; | ||||
|             let dfu_from_offset = (page_num + 1) * page_size; | ||||
|             self.copy_page_once_to_active(progress_index + 1, dfu_from_offset, active_to_offset, p, aligned_buf)?; | ||||
|         } | ||||
|  | ||||
|         Ok(()) | ||||
|     } | ||||
|  | ||||
|     fn read_state<P: FlashConfig>(&mut self, config: &mut P, aligned_buf: &mut [u8]) -> Result<State, BootError> { | ||||
|         let state_word = &mut aligned_buf[..P::STATE::WRITE_SIZE]; | ||||
|         self.state.read_blocking(config.state(), 0, state_word)?; | ||||
|  | ||||
|         if !state_word.iter().any(|&b| b != SWAP_MAGIC) { | ||||
|             Ok(State::Swap) | ||||
|         } else { | ||||
|             Ok(State::Boot) | ||||
|         } | ||||
|     } | ||||
| } | ||||
|  | ||||
| fn assert_partitions(active: Partition, dfu: Partition, state: Partition, page_size: u32, state_write_size: usize) { | ||||
|     assert_eq!(active.size() % page_size, 0); | ||||
|     assert_eq!(dfu.size() % page_size, 0); | ||||
|     assert!(dfu.size() - active.size() >= page_size); | ||||
|     assert!(2 + 2 * (active.size() / page_size) <= state.size() / state_write_size as u32); | ||||
| } | ||||
|  | ||||
| /// A flash wrapper implementing the Flash and embedded_storage traits. | ||||
| pub struct BootFlash<F> | ||||
| where | ||||
|     F: NorFlash, | ||||
| { | ||||
|     flash: F, | ||||
| } | ||||
|  | ||||
| impl<F> BootFlash<F> | ||||
| where | ||||
|     F: NorFlash, | ||||
| { | ||||
|     /// Create a new instance of a bootable flash | ||||
|     pub fn new(flash: F) -> Self { | ||||
|         Self { flash } | ||||
|     } | ||||
| } | ||||
|  | ||||
| impl<F> ErrorType for BootFlash<F> | ||||
| where | ||||
|     F: NorFlash, | ||||
| { | ||||
|     type Error = F::Error; | ||||
| } | ||||
|  | ||||
| impl<F> NorFlash for BootFlash<F> | ||||
| where | ||||
|     F: NorFlash, | ||||
| { | ||||
|     const WRITE_SIZE: usize = F::WRITE_SIZE; | ||||
|     const ERASE_SIZE: usize = F::ERASE_SIZE; | ||||
|  | ||||
|     fn erase(&mut self, from: u32, to: u32) -> Result<(), Self::Error> { | ||||
|         F::erase(&mut self.flash, from, to) | ||||
|     } | ||||
|  | ||||
|     fn write(&mut self, offset: u32, bytes: &[u8]) -> Result<(), Self::Error> { | ||||
|         F::write(&mut self.flash, offset, bytes) | ||||
|     } | ||||
| } | ||||
|  | ||||
| impl<F> ReadNorFlash for BootFlash<F> | ||||
| where | ||||
|     F: NorFlash, | ||||
| { | ||||
|     const READ_SIZE: usize = F::READ_SIZE; | ||||
|  | ||||
|     fn read(&mut self, offset: u32, bytes: &mut [u8]) -> Result<(), Self::Error> { | ||||
|         F::read(&mut self.flash, offset, bytes) | ||||
|     } | ||||
|  | ||||
|     fn capacity(&self) -> usize { | ||||
|         F::capacity(&self.flash) | ||||
|     } | ||||
| } | ||||
|  | ||||
| /// Convenience provider that uses a single flash for all partitions. | ||||
| pub struct SingleFlashConfig<'a, F> | ||||
| where | ||||
|     F: NorFlash, | ||||
| { | ||||
|     flash: &'a mut F, | ||||
| } | ||||
|  | ||||
| impl<'a, F> SingleFlashConfig<'a, F> | ||||
| where | ||||
|     F: NorFlash, | ||||
| { | ||||
|     /// Create a provider for a single flash. | ||||
|     pub fn new(flash: &'a mut F) -> Self { | ||||
|         Self { flash } | ||||
|     } | ||||
| } | ||||
|  | ||||
| impl<'a, F> FlashConfig for SingleFlashConfig<'a, F> | ||||
| where | ||||
|     F: NorFlash, | ||||
| { | ||||
|     type STATE = F; | ||||
|     type ACTIVE = F; | ||||
|     type DFU = F; | ||||
|  | ||||
|     fn active(&mut self) -> &mut Self::STATE { | ||||
|         self.flash | ||||
|     } | ||||
|     fn dfu(&mut self) -> &mut Self::ACTIVE { | ||||
|         self.flash | ||||
|     } | ||||
|     fn state(&mut self) -> &mut Self::DFU { | ||||
|         self.flash | ||||
|     } | ||||
| } | ||||
|  | ||||
| /// Convenience flash provider that uses separate flash instances for each partition. | ||||
| pub struct MultiFlashConfig<'a, ACTIVE, STATE, DFU> | ||||
| where | ||||
|     ACTIVE: NorFlash, | ||||
|     STATE: NorFlash, | ||||
|     DFU: NorFlash, | ||||
| { | ||||
|     active: &'a mut ACTIVE, | ||||
|     state: &'a mut STATE, | ||||
|     dfu: &'a mut DFU, | ||||
| } | ||||
|  | ||||
| impl<'a, ACTIVE, STATE, DFU> MultiFlashConfig<'a, ACTIVE, STATE, DFU> | ||||
| where | ||||
|     ACTIVE: NorFlash, | ||||
|     STATE: NorFlash, | ||||
|     DFU: NorFlash, | ||||
| { | ||||
|     /// Create a new flash provider with separate configuration for all three partitions. | ||||
|     pub fn new(active: &'a mut ACTIVE, state: &'a mut STATE, dfu: &'a mut DFU) -> Self { | ||||
|         Self { active, state, dfu } | ||||
|     } | ||||
| } | ||||
|  | ||||
| impl<'a, ACTIVE, STATE, DFU> FlashConfig for MultiFlashConfig<'a, ACTIVE, STATE, DFU> | ||||
| where | ||||
|     ACTIVE: NorFlash, | ||||
|     STATE: NorFlash, | ||||
|     DFU: NorFlash, | ||||
| { | ||||
|     type STATE = STATE; | ||||
|     type ACTIVE = ACTIVE; | ||||
|     type DFU = DFU; | ||||
|  | ||||
|     fn active(&mut self) -> &mut Self::ACTIVE { | ||||
|         self.active | ||||
|     } | ||||
|     fn dfu(&mut self) -> &mut Self::DFU { | ||||
|         self.dfu | ||||
|     } | ||||
|     fn state(&mut self) -> &mut Self::STATE { | ||||
|         self.state | ||||
|     } | ||||
| } | ||||
|  | ||||
| #[cfg(test)] | ||||
| mod tests { | ||||
|     use super::*; | ||||
|  | ||||
|     #[test] | ||||
|     #[should_panic] | ||||
|     fn test_range_asserts() { | ||||
|         const ACTIVE: Partition = Partition::new(4096, 4194304); | ||||
|         const DFU: Partition = Partition::new(4194304, 2 * 4194304); | ||||
|         const STATE: Partition = Partition::new(0, 4096); | ||||
|         assert_partitions(ACTIVE, DFU, STATE, 4096, 4); | ||||
|     } | ||||
| } | ||||
							
								
								
									
										30
									
								
								embassy-boot/boot/src/digest_adapters/ed25519_dalek.rs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										30
									
								
								embassy-boot/boot/src/digest_adapters/ed25519_dalek.rs
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,30 @@ | ||||
| use digest::typenum::U64; | ||||
| use digest::{FixedOutput, HashMarker, OutputSizeUser, Update}; | ||||
| use ed25519_dalek::Digest as _; | ||||
|  | ||||
| pub struct Sha512(ed25519_dalek::Sha512); | ||||
|  | ||||
| impl Default for Sha512 { | ||||
|     fn default() -> Self { | ||||
|         Self(ed25519_dalek::Sha512::new()) | ||||
|     } | ||||
| } | ||||
|  | ||||
| impl Update for Sha512 { | ||||
|     fn update(&mut self, data: &[u8]) { | ||||
|         self.0.update(data) | ||||
|     } | ||||
| } | ||||
|  | ||||
| impl FixedOutput for Sha512 { | ||||
|     fn finalize_into(self, out: &mut digest::Output<Self>) { | ||||
|         let result = self.0.finalize(); | ||||
|         out.as_mut_slice().copy_from_slice(result.as_slice()) | ||||
|     } | ||||
| } | ||||
|  | ||||
| impl OutputSizeUser for Sha512 { | ||||
|     type OutputSize = U64; | ||||
| } | ||||
|  | ||||
| impl HashMarker for Sha512 {} | ||||
							
								
								
									
										5
									
								
								embassy-boot/boot/src/digest_adapters/mod.rs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										5
									
								
								embassy-boot/boot/src/digest_adapters/mod.rs
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,5 @@ | ||||
| #[cfg(feature = "ed25519-dalek")] | ||||
| pub(crate) mod ed25519_dalek; | ||||
|  | ||||
| #[cfg(feature = "ed25519-salty")] | ||||
| pub(crate) mod salty; | ||||
							
								
								
									
										29
									
								
								embassy-boot/boot/src/digest_adapters/salty.rs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										29
									
								
								embassy-boot/boot/src/digest_adapters/salty.rs
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,29 @@ | ||||
| use digest::typenum::U64; | ||||
| use digest::{FixedOutput, HashMarker, OutputSizeUser, Update}; | ||||
|  | ||||
| pub struct Sha512(salty::Sha512); | ||||
|  | ||||
| impl Default for Sha512 { | ||||
|     fn default() -> Self { | ||||
|         Self(salty::Sha512::new()) | ||||
|     } | ||||
| } | ||||
|  | ||||
| impl Update for Sha512 { | ||||
|     fn update(&mut self, data: &[u8]) { | ||||
|         self.0.update(data) | ||||
|     } | ||||
| } | ||||
|  | ||||
| impl FixedOutput for Sha512 { | ||||
|     fn finalize_into(self, out: &mut digest::Output<Self>) { | ||||
|         let result = self.0.finalize(); | ||||
|         out.as_mut_slice().copy_from_slice(result.as_slice()) | ||||
|     } | ||||
| } | ||||
|  | ||||
| impl OutputSizeUser for Sha512 { | ||||
|     type OutputSize = U64; | ||||
| } | ||||
|  | ||||
| impl HashMarker for Sha512 {} | ||||
							
								
								
									
										542
									
								
								embassy-boot/boot/src/firmware_updater.rs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										542
									
								
								embassy-boot/boot/src/firmware_updater.rs
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,542 @@ | ||||
| use digest::Digest; | ||||
| use embedded_storage::nor_flash::{NorFlash, NorFlashError, NorFlashErrorKind}; | ||||
| #[cfg(feature = "nightly")] | ||||
| use embedded_storage_async::nor_flash::NorFlash as AsyncNorFlash; | ||||
|  | ||||
| use crate::{Partition, State, BOOT_MAGIC, SWAP_MAGIC}; | ||||
|  | ||||
| /// Errors returned by FirmwareUpdater | ||||
| #[derive(Debug)] | ||||
| pub enum FirmwareUpdaterError { | ||||
|     /// Error from flash. | ||||
|     Flash(NorFlashErrorKind), | ||||
|     /// Signature errors. | ||||
|     Signature(signature::Error), | ||||
| } | ||||
|  | ||||
| #[cfg(feature = "defmt")] | ||||
| impl defmt::Format for FirmwareUpdaterError { | ||||
|     fn format(&self, fmt: defmt::Formatter) { | ||||
|         match self { | ||||
|             FirmwareUpdaterError::Flash(_) => defmt::write!(fmt, "FirmwareUpdaterError::Flash(_)"), | ||||
|             FirmwareUpdaterError::Signature(_) => defmt::write!(fmt, "FirmwareUpdaterError::Signature(_)"), | ||||
|         } | ||||
|     } | ||||
| } | ||||
|  | ||||
| impl<E> From<E> for FirmwareUpdaterError | ||||
| where | ||||
|     E: NorFlashError, | ||||
| { | ||||
|     fn from(error: E) -> Self { | ||||
|         FirmwareUpdaterError::Flash(error.kind()) | ||||
|     } | ||||
| } | ||||
|  | ||||
| /// FirmwareUpdater is an application API for interacting with the BootLoader without the ability to | ||||
| /// 'mess up' the internal bootloader state | ||||
| pub struct FirmwareUpdater { | ||||
|     state: Partition, | ||||
|     dfu: Partition, | ||||
| } | ||||
|  | ||||
| impl Default for FirmwareUpdater { | ||||
|     fn default() -> Self { | ||||
|         extern "C" { | ||||
|             static __bootloader_state_start: u32; | ||||
|             static __bootloader_state_end: u32; | ||||
|             static __bootloader_dfu_start: u32; | ||||
|             static __bootloader_dfu_end: u32; | ||||
|         } | ||||
|  | ||||
|         let dfu = unsafe { | ||||
|             Partition::new( | ||||
|                 &__bootloader_dfu_start as *const u32 as u32, | ||||
|                 &__bootloader_dfu_end as *const u32 as u32, | ||||
|             ) | ||||
|         }; | ||||
|         let state = unsafe { | ||||
|             Partition::new( | ||||
|                 &__bootloader_state_start as *const u32 as u32, | ||||
|                 &__bootloader_state_end as *const u32 as u32, | ||||
|             ) | ||||
|         }; | ||||
|  | ||||
|         trace!("DFU: 0x{:x} - 0x{:x}", dfu.from, dfu.to); | ||||
|         trace!("STATE: 0x{:x} - 0x{:x}", state.from, state.to); | ||||
|         FirmwareUpdater::new(dfu, state) | ||||
|     } | ||||
| } | ||||
|  | ||||
| impl FirmwareUpdater { | ||||
|     /// Create a firmware updater instance with partition ranges for the update and state partitions. | ||||
|     pub const fn new(dfu: Partition, state: Partition) -> Self { | ||||
|         Self { dfu, state } | ||||
|     } | ||||
|  | ||||
|     /// Obtain the current state. | ||||
|     /// | ||||
|     /// This is useful to check if the bootloader has just done a swap, in order | ||||
|     /// to do verifications and self-tests of the new image before calling | ||||
|     /// `mark_booted`. | ||||
|     #[cfg(feature = "nightly")] | ||||
|     pub async fn get_state<F: AsyncNorFlash>( | ||||
|         &mut self, | ||||
|         state_flash: &mut F, | ||||
|         aligned: &mut [u8], | ||||
|     ) -> Result<State, FirmwareUpdaterError> { | ||||
|         self.state.read(state_flash, 0, aligned).await?; | ||||
|  | ||||
|         if !aligned.iter().any(|&b| b != SWAP_MAGIC) { | ||||
|             Ok(State::Swap) | ||||
|         } else { | ||||
|             Ok(State::Boot) | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     /// Verify the DFU given a public key. If there is an error then DO NOT | ||||
|     /// proceed with updating the firmware as it must be signed with a | ||||
|     /// corresponding private key (otherwise it could be malicious firmware). | ||||
|     /// | ||||
|     /// Mark to trigger firmware swap on next boot if verify suceeds. | ||||
|     /// | ||||
|     /// If the "ed25519-salty" feature is set (or another similar feature) then the signature is expected to have | ||||
|     /// been generated from a SHA-512 digest of the firmware bytes. | ||||
|     /// | ||||
|     /// If no signature feature is set then this method will always return a | ||||
|     /// signature error. | ||||
|     /// | ||||
|     /// # Safety | ||||
|     /// | ||||
|     /// The `_aligned` buffer must have a size of F::WRITE_SIZE, and follow the alignment rules for the flash being read from | ||||
|     /// and written to. | ||||
|     #[cfg(all(feature = "_verify", feature = "nightly"))] | ||||
|     pub async fn verify_and_mark_updated<F: AsyncNorFlash>( | ||||
|         &mut self, | ||||
|         _state_and_dfu_flash: &mut F, | ||||
|         _public_key: &[u8], | ||||
|         _signature: &[u8], | ||||
|         _update_len: u32, | ||||
|         _aligned: &mut [u8], | ||||
|     ) -> Result<(), FirmwareUpdaterError> { | ||||
|         assert_eq!(_aligned.len(), F::WRITE_SIZE); | ||||
|         assert!(_update_len <= self.dfu.size()); | ||||
|  | ||||
|         #[cfg(feature = "ed25519-dalek")] | ||||
|         { | ||||
|             use ed25519_dalek::{PublicKey, Signature, SignatureError, Verifier}; | ||||
|  | ||||
|             use crate::digest_adapters::ed25519_dalek::Sha512; | ||||
|  | ||||
|             let into_signature_error = |e: SignatureError| FirmwareUpdaterError::Signature(e.into()); | ||||
|  | ||||
|             let public_key = PublicKey::from_bytes(_public_key).map_err(into_signature_error)?; | ||||
|             let signature = Signature::from_bytes(_signature).map_err(into_signature_error)?; | ||||
|  | ||||
|             let mut message = [0; 64]; | ||||
|             self.hash::<_, Sha512>(_state_and_dfu_flash, _update_len, _aligned, &mut message) | ||||
|                 .await?; | ||||
|  | ||||
|             public_key.verify(&message, &signature).map_err(into_signature_error)? | ||||
|         } | ||||
|         #[cfg(feature = "ed25519-salty")] | ||||
|         { | ||||
|             use salty::constants::{PUBLICKEY_SERIALIZED_LENGTH, SIGNATURE_SERIALIZED_LENGTH}; | ||||
|             use salty::{PublicKey, Signature}; | ||||
|  | ||||
|             use crate::digest_adapters::salty::Sha512; | ||||
|  | ||||
|             fn into_signature_error<E>(_: E) -> FirmwareUpdaterError { | ||||
|                 FirmwareUpdaterError::Signature(signature::Error::default()) | ||||
|             } | ||||
|  | ||||
|             let public_key: [u8; PUBLICKEY_SERIALIZED_LENGTH] = _public_key.try_into().map_err(into_signature_error)?; | ||||
|             let public_key = PublicKey::try_from(&public_key).map_err(into_signature_error)?; | ||||
|             let signature: [u8; SIGNATURE_SERIALIZED_LENGTH] = _signature.try_into().map_err(into_signature_error)?; | ||||
|             let signature = Signature::try_from(&signature).map_err(into_signature_error)?; | ||||
|  | ||||
|             let mut message = [0; 64]; | ||||
|             self.hash::<_, Sha512>(_state_and_dfu_flash, _update_len, _aligned, &mut message) | ||||
|                 .await?; | ||||
|  | ||||
|             let r = public_key.verify(&message, &signature); | ||||
|             trace!( | ||||
|                 "Verifying with public key {}, signature {} and message {} yields ok: {}", | ||||
|                 public_key.to_bytes(), | ||||
|                 signature.to_bytes(), | ||||
|                 message, | ||||
|                 r.is_ok() | ||||
|             ); | ||||
|             r.map_err(into_signature_error)? | ||||
|         } | ||||
|  | ||||
|         self.set_magic(_aligned, SWAP_MAGIC, _state_and_dfu_flash).await | ||||
|     } | ||||
|  | ||||
|     /// Verify the update in DFU with any digest. | ||||
|     #[cfg(feature = "nightly")] | ||||
|     pub async fn hash<F: AsyncNorFlash, D: Digest>( | ||||
|         &mut self, | ||||
|         dfu_flash: &mut F, | ||||
|         update_len: u32, | ||||
|         chunk_buf: &mut [u8], | ||||
|         output: &mut [u8], | ||||
|     ) -> Result<(), FirmwareUpdaterError> { | ||||
|         let mut digest = D::new(); | ||||
|         for offset in (0..update_len).step_by(chunk_buf.len()) { | ||||
|             self.dfu.read(dfu_flash, offset, chunk_buf).await?; | ||||
|             let len = core::cmp::min((update_len - offset) as usize, chunk_buf.len()); | ||||
|             digest.update(&chunk_buf[..len]); | ||||
|         } | ||||
|         output.copy_from_slice(digest.finalize().as_slice()); | ||||
|         Ok(()) | ||||
|     } | ||||
|  | ||||
|     /// Mark to trigger firmware swap on next boot. | ||||
|     /// | ||||
|     /// # Safety | ||||
|     /// | ||||
|     /// The `aligned` buffer must have a size of F::WRITE_SIZE, and follow the alignment rules for the flash being written to. | ||||
|     #[cfg(all(feature = "nightly", not(feature = "_verify")))] | ||||
|     pub async fn mark_updated<F: AsyncNorFlash>( | ||||
|         &mut self, | ||||
|         state_flash: &mut F, | ||||
|         aligned: &mut [u8], | ||||
|     ) -> Result<(), FirmwareUpdaterError> { | ||||
|         assert_eq!(aligned.len(), F::WRITE_SIZE); | ||||
|         self.set_magic(aligned, SWAP_MAGIC, state_flash).await | ||||
|     } | ||||
|  | ||||
|     /// Mark firmware boot successful and stop rollback on reset. | ||||
|     /// | ||||
|     /// # Safety | ||||
|     /// | ||||
|     /// The `aligned` buffer must have a size of F::WRITE_SIZE, and follow the alignment rules for the flash being written to. | ||||
|     #[cfg(feature = "nightly")] | ||||
|     pub async fn mark_booted<F: AsyncNorFlash>( | ||||
|         &mut self, | ||||
|         state_flash: &mut F, | ||||
|         aligned: &mut [u8], | ||||
|     ) -> Result<(), FirmwareUpdaterError> { | ||||
|         assert_eq!(aligned.len(), F::WRITE_SIZE); | ||||
|         self.set_magic(aligned, BOOT_MAGIC, state_flash).await | ||||
|     } | ||||
|  | ||||
|     #[cfg(feature = "nightly")] | ||||
|     async fn set_magic<F: AsyncNorFlash>( | ||||
|         &mut self, | ||||
|         aligned: &mut [u8], | ||||
|         magic: u8, | ||||
|         state_flash: &mut F, | ||||
|     ) -> Result<(), FirmwareUpdaterError> { | ||||
|         self.state.read(state_flash, 0, aligned).await?; | ||||
|  | ||||
|         if aligned.iter().any(|&b| b != magic) { | ||||
|             // Read progress validity | ||||
|             self.state.read(state_flash, F::WRITE_SIZE as u32, aligned).await?; | ||||
|  | ||||
|             // FIXME: Do not make this assumption. | ||||
|             const STATE_ERASE_VALUE: u8 = 0xFF; | ||||
|  | ||||
|             if aligned.iter().any(|&b| b != STATE_ERASE_VALUE) { | ||||
|                 // The current progress validity marker is invalid | ||||
|             } else { | ||||
|                 // Invalidate progress | ||||
|                 aligned.fill(!STATE_ERASE_VALUE); | ||||
|                 self.state.write(state_flash, F::WRITE_SIZE as u32, aligned).await?; | ||||
|             } | ||||
|  | ||||
|             // Clear magic and progress | ||||
|             self.state.wipe(state_flash).await?; | ||||
|  | ||||
|             // Set magic | ||||
|             aligned.fill(magic); | ||||
|             self.state.write(state_flash, 0, aligned).await?; | ||||
|         } | ||||
|         Ok(()) | ||||
|     } | ||||
|  | ||||
|     /// Write data to a flash page. | ||||
|     /// | ||||
|     /// The buffer must follow alignment requirements of the target flash and a multiple of page size big. | ||||
|     /// | ||||
|     /// # Safety | ||||
|     /// | ||||
|     /// Failing to meet alignment and size requirements may result in a panic. | ||||
|     #[cfg(feature = "nightly")] | ||||
|     pub async fn write_firmware<F: AsyncNorFlash>( | ||||
|         &mut self, | ||||
|         offset: usize, | ||||
|         data: &[u8], | ||||
|         dfu_flash: &mut F, | ||||
|     ) -> Result<(), FirmwareUpdaterError> { | ||||
|         assert!(data.len() >= F::ERASE_SIZE); | ||||
|  | ||||
|         self.dfu | ||||
|             .erase(dfu_flash, offset as u32, (offset + data.len()) as u32) | ||||
|             .await?; | ||||
|  | ||||
|         self.dfu.write(dfu_flash, offset as u32, data).await?; | ||||
|  | ||||
|         Ok(()) | ||||
|     } | ||||
|  | ||||
|     /// Prepare for an incoming DFU update by erasing the entire DFU area and | ||||
|     /// returning its `Partition`. | ||||
|     /// | ||||
|     /// Using this instead of `write_firmware` allows for an optimized API in | ||||
|     /// exchange for added complexity. | ||||
|     #[cfg(feature = "nightly")] | ||||
|     pub async fn prepare_update<F: AsyncNorFlash>( | ||||
|         &mut self, | ||||
|         dfu_flash: &mut F, | ||||
|     ) -> Result<Partition, FirmwareUpdaterError> { | ||||
|         self.dfu.wipe(dfu_flash).await?; | ||||
|  | ||||
|         Ok(self.dfu) | ||||
|     } | ||||
|  | ||||
|     // | ||||
|     // Blocking API | ||||
|     // | ||||
|  | ||||
|     /// Obtain the current state. | ||||
|     /// | ||||
|     /// This is useful to check if the bootloader has just done a swap, in order | ||||
|     /// to do verifications and self-tests of the new image before calling | ||||
|     /// `mark_booted`. | ||||
|     pub fn get_state_blocking<F: NorFlash>( | ||||
|         &mut self, | ||||
|         state_flash: &mut F, | ||||
|         aligned: &mut [u8], | ||||
|     ) -> Result<State, FirmwareUpdaterError> { | ||||
|         self.state.read_blocking(state_flash, 0, aligned)?; | ||||
|  | ||||
|         if !aligned.iter().any(|&b| b != SWAP_MAGIC) { | ||||
|             Ok(State::Swap) | ||||
|         } else { | ||||
|             Ok(State::Boot) | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     /// Verify the DFU given a public key. If there is an error then DO NOT | ||||
|     /// proceed with updating the firmware as it must be signed with a | ||||
|     /// corresponding private key (otherwise it could be malicious firmware). | ||||
|     /// | ||||
|     /// Mark to trigger firmware swap on next boot if verify suceeds. | ||||
|     /// | ||||
|     /// If the "ed25519-salty" feature is set (or another similar feature) then the signature is expected to have | ||||
|     /// been generated from a SHA-512 digest of the firmware bytes. | ||||
|     /// | ||||
|     /// If no signature feature is set then this method will always return a | ||||
|     /// signature error. | ||||
|     /// | ||||
|     /// # Safety | ||||
|     /// | ||||
|     /// The `_aligned` buffer must have a size of F::WRITE_SIZE, and follow the alignment rules for the flash being read from | ||||
|     /// and written to. | ||||
|     #[cfg(feature = "_verify")] | ||||
|     pub fn verify_and_mark_updated_blocking<F: NorFlash>( | ||||
|         &mut self, | ||||
|         _state_and_dfu_flash: &mut F, | ||||
|         _public_key: &[u8], | ||||
|         _signature: &[u8], | ||||
|         _update_len: u32, | ||||
|         _aligned: &mut [u8], | ||||
|     ) -> Result<(), FirmwareUpdaterError> { | ||||
|         assert_eq!(_aligned.len(), F::WRITE_SIZE); | ||||
|         assert!(_update_len <= self.dfu.size()); | ||||
|  | ||||
|         #[cfg(feature = "ed25519-dalek")] | ||||
|         { | ||||
|             use ed25519_dalek::{PublicKey, Signature, SignatureError, Verifier}; | ||||
|  | ||||
|             use crate::digest_adapters::ed25519_dalek::Sha512; | ||||
|  | ||||
|             let into_signature_error = |e: SignatureError| FirmwareUpdaterError::Signature(e.into()); | ||||
|  | ||||
|             let public_key = PublicKey::from_bytes(_public_key).map_err(into_signature_error)?; | ||||
|             let signature = Signature::from_bytes(_signature).map_err(into_signature_error)?; | ||||
|  | ||||
|             let mut message = [0; 64]; | ||||
|             self.hash_blocking::<_, Sha512>(_state_and_dfu_flash, _update_len, _aligned, &mut message)?; | ||||
|  | ||||
|             public_key.verify(&message, &signature).map_err(into_signature_error)? | ||||
|         } | ||||
|         #[cfg(feature = "ed25519-salty")] | ||||
|         { | ||||
|             use salty::constants::{PUBLICKEY_SERIALIZED_LENGTH, SIGNATURE_SERIALIZED_LENGTH}; | ||||
|             use salty::{PublicKey, Signature}; | ||||
|  | ||||
|             use crate::digest_adapters::salty::Sha512; | ||||
|  | ||||
|             fn into_signature_error<E>(_: E) -> FirmwareUpdaterError { | ||||
|                 FirmwareUpdaterError::Signature(signature::Error::default()) | ||||
|             } | ||||
|  | ||||
|             let public_key: [u8; PUBLICKEY_SERIALIZED_LENGTH] = _public_key.try_into().map_err(into_signature_error)?; | ||||
|             let public_key = PublicKey::try_from(&public_key).map_err(into_signature_error)?; | ||||
|             let signature: [u8; SIGNATURE_SERIALIZED_LENGTH] = _signature.try_into().map_err(into_signature_error)?; | ||||
|             let signature = Signature::try_from(&signature).map_err(into_signature_error)?; | ||||
|  | ||||
|             let mut message = [0; 64]; | ||||
|             self.hash_blocking::<_, Sha512>(_state_and_dfu_flash, _update_len, _aligned, &mut message)?; | ||||
|  | ||||
|             let r = public_key.verify(&message, &signature); | ||||
|             trace!( | ||||
|                 "Verifying with public key {}, signature {} and message {} yields ok: {}", | ||||
|                 public_key.to_bytes(), | ||||
|                 signature.to_bytes(), | ||||
|                 message, | ||||
|                 r.is_ok() | ||||
|             ); | ||||
|             r.map_err(into_signature_error)? | ||||
|         } | ||||
|  | ||||
|         self.set_magic_blocking(_aligned, SWAP_MAGIC, _state_and_dfu_flash) | ||||
|     } | ||||
|  | ||||
|     /// Verify the update in DFU with any digest. | ||||
|     pub fn hash_blocking<F: NorFlash, D: Digest>( | ||||
|         &mut self, | ||||
|         dfu_flash: &mut F, | ||||
|         update_len: u32, | ||||
|         chunk_buf: &mut [u8], | ||||
|         output: &mut [u8], | ||||
|     ) -> Result<(), FirmwareUpdaterError> { | ||||
|         let mut digest = D::new(); | ||||
|         for offset in (0..update_len).step_by(chunk_buf.len()) { | ||||
|             self.dfu.read_blocking(dfu_flash, offset, chunk_buf)?; | ||||
|             let len = core::cmp::min((update_len - offset) as usize, chunk_buf.len()); | ||||
|             digest.update(&chunk_buf[..len]); | ||||
|         } | ||||
|         output.copy_from_slice(digest.finalize().as_slice()); | ||||
|         Ok(()) | ||||
|     } | ||||
|  | ||||
|     /// Mark to trigger firmware swap on next boot. | ||||
|     /// | ||||
|     /// # Safety | ||||
|     /// | ||||
|     /// The `aligned` buffer must have a size of F::WRITE_SIZE, and follow the alignment rules for the flash being written to. | ||||
|     #[cfg(not(feature = "_verify"))] | ||||
|     pub fn mark_updated_blocking<F: NorFlash>( | ||||
|         &mut self, | ||||
|         state_flash: &mut F, | ||||
|         aligned: &mut [u8], | ||||
|     ) -> Result<(), FirmwareUpdaterError> { | ||||
|         assert_eq!(aligned.len(), F::WRITE_SIZE); | ||||
|         self.set_magic_blocking(aligned, SWAP_MAGIC, state_flash) | ||||
|     } | ||||
|  | ||||
|     /// Mark firmware boot successful and stop rollback on reset. | ||||
|     /// | ||||
|     /// # Safety | ||||
|     /// | ||||
|     /// The `aligned` buffer must have a size of F::WRITE_SIZE, and follow the alignment rules for the flash being written to. | ||||
|     pub fn mark_booted_blocking<F: NorFlash>( | ||||
|         &mut self, | ||||
|         state_flash: &mut F, | ||||
|         aligned: &mut [u8], | ||||
|     ) -> Result<(), FirmwareUpdaterError> { | ||||
|         assert_eq!(aligned.len(), F::WRITE_SIZE); | ||||
|         self.set_magic_blocking(aligned, BOOT_MAGIC, state_flash) | ||||
|     } | ||||
|  | ||||
|     fn set_magic_blocking<F: NorFlash>( | ||||
|         &mut self, | ||||
|         aligned: &mut [u8], | ||||
|         magic: u8, | ||||
|         state_flash: &mut F, | ||||
|     ) -> Result<(), FirmwareUpdaterError> { | ||||
|         self.state.read_blocking(state_flash, 0, aligned)?; | ||||
|  | ||||
|         if aligned.iter().any(|&b| b != magic) { | ||||
|             // Read progress validity | ||||
|             self.state.read_blocking(state_flash, F::WRITE_SIZE as u32, aligned)?; | ||||
|  | ||||
|             // FIXME: Do not make this assumption. | ||||
|             const STATE_ERASE_VALUE: u8 = 0xFF; | ||||
|  | ||||
|             if aligned.iter().any(|&b| b != STATE_ERASE_VALUE) { | ||||
|                 // The current progress validity marker is invalid | ||||
|             } else { | ||||
|                 // Invalidate progress | ||||
|                 aligned.fill(!STATE_ERASE_VALUE); | ||||
|                 self.state.write_blocking(state_flash, F::WRITE_SIZE as u32, aligned)?; | ||||
|             } | ||||
|  | ||||
|             // Clear magic and progress | ||||
|             self.state.wipe_blocking(state_flash)?; | ||||
|  | ||||
|             // Set magic | ||||
|             aligned.fill(magic); | ||||
|             self.state.write_blocking(state_flash, 0, aligned)?; | ||||
|         } | ||||
|         Ok(()) | ||||
|     } | ||||
|  | ||||
|     /// Write data to a flash page. | ||||
|     /// | ||||
|     /// The buffer must follow alignment requirements of the target flash and a multiple of page size big. | ||||
|     /// | ||||
|     /// # Safety | ||||
|     /// | ||||
|     /// Failing to meet alignment and size requirements may result in a panic. | ||||
|     pub fn write_firmware_blocking<F: NorFlash>( | ||||
|         &mut self, | ||||
|         offset: usize, | ||||
|         data: &[u8], | ||||
|         dfu_flash: &mut F, | ||||
|     ) -> Result<(), FirmwareUpdaterError> { | ||||
|         assert!(data.len() >= F::ERASE_SIZE); | ||||
|  | ||||
|         self.dfu | ||||
|             .erase_blocking(dfu_flash, offset as u32, (offset + data.len()) as u32)?; | ||||
|  | ||||
|         self.dfu.write_blocking(dfu_flash, offset as u32, data)?; | ||||
|  | ||||
|         Ok(()) | ||||
|     } | ||||
|  | ||||
|     /// Prepare for an incoming DFU update by erasing the entire DFU area and | ||||
|     /// returning its `Partition`. | ||||
|     /// | ||||
|     /// Using this instead of `write_firmware_blocking` allows for an optimized | ||||
|     /// API in exchange for added complexity. | ||||
|     pub fn prepare_update_blocking<F: NorFlash>(&mut self, flash: &mut F) -> Result<Partition, FirmwareUpdaterError> { | ||||
|         self.dfu.wipe_blocking(flash)?; | ||||
|  | ||||
|         Ok(self.dfu) | ||||
|     } | ||||
| } | ||||
|  | ||||
| #[cfg(test)] | ||||
| mod tests { | ||||
|     use futures::executor::block_on; | ||||
|     use sha1::{Digest, Sha1}; | ||||
|  | ||||
|     use super::*; | ||||
|     use crate::mem_flash::MemFlash; | ||||
|  | ||||
|     #[test] | ||||
|     #[cfg(feature = "nightly")] | ||||
|     fn can_verify_sha1() { | ||||
|         const STATE: Partition = Partition::new(0, 4096); | ||||
|         const DFU: Partition = Partition::new(65536, 131072); | ||||
|  | ||||
|         let mut flash = MemFlash::<131072, 4096, 8>::default(); | ||||
|  | ||||
|         let update = [0x00, 0x11, 0x22, 0x33, 0x44, 0x55, 0x66]; | ||||
|         let mut to_write = [0; 4096]; | ||||
|         to_write[..7].copy_from_slice(update.as_slice()); | ||||
|  | ||||
|         let mut updater = FirmwareUpdater::new(DFU, STATE); | ||||
|         block_on(updater.write_firmware(0, to_write.as_slice(), &mut flash)).unwrap(); | ||||
|         let mut chunk_buf = [0; 2]; | ||||
|         let mut hash = [0; 20]; | ||||
|         block_on(updater.hash::<_, Sha1>(&mut flash, update.len() as u32, &mut chunk_buf, &mut hash)).unwrap(); | ||||
|  | ||||
|         assert_eq!(Sha1::digest(update).as_slice(), hash); | ||||
|     } | ||||
| } | ||||
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
							
								
								
									
										167
									
								
								embassy-boot/boot/src/mem_flash.rs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										167
									
								
								embassy-boot/boot/src/mem_flash.rs
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,167 @@ | ||||
| #![allow(unused)] | ||||
|  | ||||
| use core::ops::{Bound, Range, RangeBounds}; | ||||
|  | ||||
| use embedded_storage::nor_flash::{ErrorType, NorFlash, NorFlashError, NorFlashErrorKind, ReadNorFlash}; | ||||
| #[cfg(feature = "nightly")] | ||||
| use embedded_storage_async::nor_flash::{NorFlash as AsyncNorFlash, ReadNorFlash as AsyncReadNorFlash}; | ||||
|  | ||||
| pub struct MemFlash<const SIZE: usize, const ERASE_SIZE: usize, const WRITE_SIZE: usize> { | ||||
|     pub mem: [u8; SIZE], | ||||
|     pub pending_write_successes: Option<usize>, | ||||
| } | ||||
|  | ||||
| #[derive(Debug)] | ||||
| pub struct MemFlashError; | ||||
|  | ||||
| impl<const SIZE: usize, const ERASE_SIZE: usize, const WRITE_SIZE: usize> MemFlash<SIZE, ERASE_SIZE, WRITE_SIZE> { | ||||
|     pub const fn new(fill: u8) -> Self { | ||||
|         Self { | ||||
|             mem: [fill; SIZE], | ||||
|             pending_write_successes: None, | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     #[cfg(test)] | ||||
|     pub fn random() -> Self { | ||||
|         let mut mem = [0; SIZE]; | ||||
|         for byte in mem.iter_mut() { | ||||
|             *byte = rand::random::<u8>(); | ||||
|         } | ||||
|         Self { | ||||
|             mem, | ||||
|             pending_write_successes: None, | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     pub fn program(&mut self, offset: u32, bytes: &[u8]) -> Result<(), MemFlashError> { | ||||
|         let offset = offset as usize; | ||||
|         assert!(bytes.len() % WRITE_SIZE == 0); | ||||
|         assert!(offset % WRITE_SIZE == 0); | ||||
|         assert!(offset + bytes.len() <= SIZE); | ||||
|  | ||||
|         self.mem[offset..offset + bytes.len()].copy_from_slice(bytes); | ||||
|  | ||||
|         Ok(()) | ||||
|     } | ||||
|  | ||||
|     pub fn assert_eq(&self, offset: u32, expectation: &[u8]) { | ||||
|         for i in 0..expectation.len() { | ||||
|             assert_eq!(self.mem[offset as usize + i], expectation[i], "Index {}", i); | ||||
|         } | ||||
|     } | ||||
| } | ||||
|  | ||||
| impl<const SIZE: usize, const ERASE_SIZE: usize, const WRITE_SIZE: usize> Default | ||||
|     for MemFlash<SIZE, ERASE_SIZE, WRITE_SIZE> | ||||
| { | ||||
|     fn default() -> Self { | ||||
|         Self::new(0xFF) | ||||
|     } | ||||
| } | ||||
|  | ||||
| impl<const SIZE: usize, const ERASE_SIZE: usize, const WRITE_SIZE: usize> ErrorType | ||||
|     for MemFlash<SIZE, ERASE_SIZE, WRITE_SIZE> | ||||
| { | ||||
|     type Error = MemFlashError; | ||||
| } | ||||
|  | ||||
| impl NorFlashError for MemFlashError { | ||||
|     fn kind(&self) -> NorFlashErrorKind { | ||||
|         NorFlashErrorKind::Other | ||||
|     } | ||||
| } | ||||
|  | ||||
| impl<const SIZE: usize, const ERASE_SIZE: usize, const WRITE_SIZE: usize> ReadNorFlash | ||||
|     for MemFlash<SIZE, ERASE_SIZE, WRITE_SIZE> | ||||
| { | ||||
|     const READ_SIZE: usize = 1; | ||||
|  | ||||
|     fn read(&mut self, offset: u32, bytes: &mut [u8]) -> Result<(), Self::Error> { | ||||
|         let len = bytes.len(); | ||||
|         bytes.copy_from_slice(&self.mem[offset as usize..offset as usize + len]); | ||||
|         Ok(()) | ||||
|     } | ||||
|  | ||||
|     fn capacity(&self) -> usize { | ||||
|         SIZE | ||||
|     } | ||||
| } | ||||
|  | ||||
| impl<const SIZE: usize, const ERASE_SIZE: usize, const WRITE_SIZE: usize> NorFlash | ||||
|     for MemFlash<SIZE, ERASE_SIZE, WRITE_SIZE> | ||||
| { | ||||
|     const WRITE_SIZE: usize = WRITE_SIZE; | ||||
|     const ERASE_SIZE: usize = ERASE_SIZE; | ||||
|  | ||||
|     fn erase(&mut self, from: u32, to: u32) -> Result<(), Self::Error> { | ||||
|         let from = from as usize; | ||||
|         let to = to as usize; | ||||
|         assert!(from % ERASE_SIZE == 0); | ||||
|         assert!(to % ERASE_SIZE == 0, "To: {}, erase size: {}", to, ERASE_SIZE); | ||||
|         for i in from..to { | ||||
|             self.mem[i] = 0xFF; | ||||
|         } | ||||
|         Ok(()) | ||||
|     } | ||||
|  | ||||
|     fn write(&mut self, offset: u32, bytes: &[u8]) -> Result<(), Self::Error> { | ||||
|         let offset = offset as usize; | ||||
|         assert!(bytes.len() % WRITE_SIZE == 0); | ||||
|         assert!(offset % WRITE_SIZE == 0); | ||||
|         assert!(offset + bytes.len() <= SIZE); | ||||
|  | ||||
|         if let Some(pending_successes) = self.pending_write_successes { | ||||
|             if pending_successes > 0 { | ||||
|                 self.pending_write_successes = Some(pending_successes - 1); | ||||
|             } else { | ||||
|                 return Err(MemFlashError); | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         for ((offset, mem_byte), new_byte) in self | ||||
|             .mem | ||||
|             .iter_mut() | ||||
|             .enumerate() | ||||
|             .skip(offset) | ||||
|             .take(bytes.len()) | ||||
|             .zip(bytes) | ||||
|         { | ||||
|             assert_eq!(0xFF, *mem_byte, "Offset {} is not erased", offset); | ||||
|             *mem_byte = *new_byte; | ||||
|         } | ||||
|  | ||||
|         Ok(()) | ||||
|     } | ||||
| } | ||||
|  | ||||
| #[cfg(feature = "nightly")] | ||||
| impl<const SIZE: usize, const ERASE_SIZE: usize, const WRITE_SIZE: usize> AsyncReadNorFlash | ||||
|     for MemFlash<SIZE, ERASE_SIZE, WRITE_SIZE> | ||||
| { | ||||
|     const READ_SIZE: usize = 1; | ||||
|  | ||||
|     async fn read(&mut self, offset: u32, bytes: &mut [u8]) -> Result<(), Self::Error> { | ||||
|         <Self as ReadNorFlash>::read(self, offset, bytes) | ||||
|     } | ||||
|  | ||||
|     fn capacity(&self) -> usize { | ||||
|         <Self as ReadNorFlash>::capacity(self) | ||||
|     } | ||||
| } | ||||
|  | ||||
| #[cfg(feature = "nightly")] | ||||
| impl<const SIZE: usize, const ERASE_SIZE: usize, const WRITE_SIZE: usize> AsyncNorFlash | ||||
|     for MemFlash<SIZE, ERASE_SIZE, WRITE_SIZE> | ||||
| { | ||||
|     const WRITE_SIZE: usize = WRITE_SIZE; | ||||
|     const ERASE_SIZE: usize = ERASE_SIZE; | ||||
|  | ||||
|     async fn erase(&mut self, from: u32, to: u32) -> Result<(), Self::Error> { | ||||
|         <Self as NorFlash>::erase(self, from, to) | ||||
|     } | ||||
|  | ||||
|     async fn write(&mut self, offset: u32, bytes: &[u8]) -> Result<(), Self::Error> { | ||||
|         <Self as NorFlash>::write(self, offset, bytes) | ||||
|     } | ||||
| } | ||||
							
								
								
									
										144
									
								
								embassy-boot/boot/src/partition.rs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										144
									
								
								embassy-boot/boot/src/partition.rs
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,144 @@ | ||||
| use embedded_storage::nor_flash::{NorFlash, ReadNorFlash}; | ||||
| #[cfg(feature = "nightly")] | ||||
| use embedded_storage_async::nor_flash::{NorFlash as AsyncNorFlash, ReadNorFlash as AsyncReadNorFlash}; | ||||
|  | ||||
| /// A region in flash used by the bootloader. | ||||
| #[derive(Copy, Clone, Debug)] | ||||
| #[cfg_attr(feature = "defmt", derive(defmt::Format))] | ||||
| pub struct Partition { | ||||
|     /// The offset into the flash where the partition starts. | ||||
|     pub from: u32, | ||||
|     /// The offset into the flash where the partition ends. | ||||
|     pub to: u32, | ||||
| } | ||||
|  | ||||
| impl Partition { | ||||
|     /// Create a new partition with the provided range | ||||
|     pub const fn new(from: u32, to: u32) -> Self { | ||||
|         Self { from, to } | ||||
|     } | ||||
|  | ||||
|     /// Return the size of the partition | ||||
|     pub const fn size(&self) -> u32 { | ||||
|         self.to - self.from | ||||
|     } | ||||
|  | ||||
|     /// Read from the partition on the provided flash | ||||
|     #[cfg(feature = "nightly")] | ||||
|     pub async fn read<F: AsyncReadNorFlash>( | ||||
|         &self, | ||||
|         flash: &mut F, | ||||
|         offset: u32, | ||||
|         bytes: &mut [u8], | ||||
|     ) -> Result<(), F::Error> { | ||||
|         let offset = self.from as u32 + offset; | ||||
|         flash.read(offset, bytes).await | ||||
|     } | ||||
|  | ||||
|     /// Write to the partition on the provided flash | ||||
|     #[cfg(feature = "nightly")] | ||||
|     pub async fn write<F: AsyncNorFlash>(&self, flash: &mut F, offset: u32, bytes: &[u8]) -> Result<(), F::Error> { | ||||
|         let offset = self.from as u32 + offset; | ||||
|         flash.write(offset, bytes).await?; | ||||
|         trace!("Wrote from 0x{:x} len {}", offset, bytes.len()); | ||||
|         Ok(()) | ||||
|     } | ||||
|  | ||||
|     /// Erase part of the partition on the provided flash | ||||
|     #[cfg(feature = "nightly")] | ||||
|     pub async fn erase<F: AsyncNorFlash>(&self, flash: &mut F, from: u32, to: u32) -> Result<(), F::Error> { | ||||
|         let from = self.from as u32 + from; | ||||
|         let to = self.from as u32 + to; | ||||
|         flash.erase(from, to).await?; | ||||
|         trace!("Erased from 0x{:x} to 0x{:x}", from, to); | ||||
|         Ok(()) | ||||
|     } | ||||
|  | ||||
|     /// Erase the entire partition | ||||
|     #[cfg(feature = "nightly")] | ||||
|     pub(crate) async fn wipe<F: AsyncNorFlash>(&self, flash: &mut F) -> Result<(), F::Error> { | ||||
|         let from = self.from as u32; | ||||
|         let to = self.to as u32; | ||||
|         flash.erase(from, to).await?; | ||||
|         trace!("Wiped from 0x{:x} to 0x{:x}", from, to); | ||||
|         Ok(()) | ||||
|     } | ||||
|  | ||||
|     /// Read from the partition on the provided flash | ||||
|     pub fn read_blocking<F: ReadNorFlash>(&self, flash: &mut F, offset: u32, bytes: &mut [u8]) -> Result<(), F::Error> { | ||||
|         let offset = self.from as u32 + offset; | ||||
|         flash.read(offset, bytes) | ||||
|     } | ||||
|  | ||||
|     /// Write to the partition on the provided flash | ||||
|     pub fn write_blocking<F: NorFlash>(&self, flash: &mut F, offset: u32, bytes: &[u8]) -> Result<(), F::Error> { | ||||
|         let offset = self.from as u32 + offset; | ||||
|         flash.write(offset, bytes)?; | ||||
|         trace!("Wrote from 0x{:x} len {}", offset, bytes.len()); | ||||
|         Ok(()) | ||||
|     } | ||||
|  | ||||
|     /// Erase part of the partition on the provided flash | ||||
|     pub fn erase_blocking<F: NorFlash>(&self, flash: &mut F, from: u32, to: u32) -> Result<(), F::Error> { | ||||
|         let from = self.from as u32 + from; | ||||
|         let to = self.from as u32 + to; | ||||
|         flash.erase(from, to)?; | ||||
|         trace!("Erased from 0x{:x} to 0x{:x}", from, to); | ||||
|         Ok(()) | ||||
|     } | ||||
|  | ||||
|     /// Erase the entire partition | ||||
|     pub(crate) fn wipe_blocking<F: NorFlash>(&self, flash: &mut F) -> Result<(), F::Error> { | ||||
|         let from = self.from as u32; | ||||
|         let to = self.to as u32; | ||||
|         flash.erase(from, to)?; | ||||
|         trace!("Wiped from 0x{:x} to 0x{:x}", from, to); | ||||
|         Ok(()) | ||||
|     } | ||||
| } | ||||
|  | ||||
| #[cfg(test)] | ||||
| mod tests { | ||||
|     use crate::mem_flash::MemFlash; | ||||
|     use crate::Partition; | ||||
|  | ||||
|     #[test] | ||||
|     fn can_erase() { | ||||
|         let mut flash = MemFlash::<1024, 64, 4>::new(0x00); | ||||
|         let partition = Partition::new(256, 512); | ||||
|  | ||||
|         partition.erase_blocking(&mut flash, 64, 192).unwrap(); | ||||
|  | ||||
|         for (index, byte) in flash.mem.iter().copied().enumerate().take(256 + 64) { | ||||
|             assert_eq!(0x00, byte, "Index {}", index); | ||||
|         } | ||||
|  | ||||
|         for (index, byte) in flash.mem.iter().copied().enumerate().skip(256 + 64).take(128) { | ||||
|             assert_eq!(0xFF, byte, "Index {}", index); | ||||
|         } | ||||
|  | ||||
|         for (index, byte) in flash.mem.iter().copied().enumerate().skip(256 + 64 + 128) { | ||||
|             assert_eq!(0x00, byte, "Index {}", index); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     #[test] | ||||
|     fn can_wipe() { | ||||
|         let mut flash = MemFlash::<1024, 64, 4>::new(0x00); | ||||
|         let partition = Partition::new(256, 512); | ||||
|  | ||||
|         partition.wipe_blocking(&mut flash).unwrap(); | ||||
|  | ||||
|         for (index, byte) in flash.mem.iter().copied().enumerate().take(256) { | ||||
|             assert_eq!(0x00, byte, "Index {}", index); | ||||
|         } | ||||
|  | ||||
|         for (index, byte) in flash.mem.iter().copied().enumerate().skip(256).take(256) { | ||||
|             assert_eq!(0xFF, byte, "Index {}", index); | ||||
|         } | ||||
|  | ||||
|         for (index, byte) in flash.mem.iter().copied().enumerate().skip(512) { | ||||
|             assert_eq!(0x00, byte, "Index {}", index); | ||||
|         } | ||||
|     } | ||||
| } | ||||
| @@ -3,6 +3,7 @@ edition = "2021" | ||||
| name = "embassy-boot-nrf" | ||||
| version = "0.1.0" | ||||
| description = "Bootloader lib for nRF chips" | ||||
| license = "MIT OR Apache-2.0" | ||||
|  | ||||
| [package.metadata.embassy_docs] | ||||
| src_base = "https://github.com/embassy-rs/embassy/blob/embassy-boot-nrf-v$VERSION/embassy-boot/nrf/src/" | ||||
| @@ -16,12 +17,12 @@ target = "thumbv7em-none-eabi" | ||||
| defmt = { version = "0.3", optional = true } | ||||
|  | ||||
| embassy-sync = { path = "../../embassy-sync" } | ||||
| embassy-nrf = { path = "../../embassy-nrf", default-features = false, features = ["nightly"] } | ||||
| embassy-nrf = { path = "../../embassy-nrf", default-features = false } | ||||
| embassy-boot = { path = "../boot", default-features = false } | ||||
| cortex-m = { version = "0.7.6" } | ||||
| cortex-m-rt = { version = "0.7" } | ||||
| embedded-storage = "0.3.0" | ||||
| embedded-storage-async = "0.3.0" | ||||
| embedded-storage-async = { version = "0.4.0", optional = true } | ||||
| cfg-if = "1.0.0" | ||||
|  | ||||
| nrf-softdevice-mbr = { version = "0.1.0", git = "https://github.com/embassy-rs/nrf-softdevice.git", branch = "master", optional = true } | ||||
| @@ -35,3 +36,8 @@ defmt = [ | ||||
| softdevice = [ | ||||
|     "nrf-softdevice-mbr", | ||||
| ] | ||||
| nightly = [ | ||||
|     "dep:embedded-storage-async", | ||||
|     "embassy-boot/nightly", | ||||
|     "embassy-nrf/nightly" | ||||
| ] | ||||
|   | ||||
							
								
								
									
										26
									
								
								embassy-boot/nrf/README.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										26
									
								
								embassy-boot/nrf/README.md
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,26 @@ | ||||
| # embassy-boot-nrf | ||||
|  | ||||
| An [Embassy](https://embassy.dev) project. | ||||
|  | ||||
| An adaptation of `embassy-boot` for nRF.  | ||||
|  | ||||
| ## Features | ||||
|  | ||||
| * Load applications with our without the softdevice. | ||||
| * Configure bootloader partitions based on linker script. | ||||
| * Using watchdog timer to detect application failure. | ||||
|  | ||||
|  | ||||
| ## Minimum supported Rust version (MSRV) | ||||
|  | ||||
| `embassy-boot-nrf` is guaranteed to compile on the latest stable Rust version at the time of release. It might compile with older versions but that may change in any new patch release. | ||||
|  | ||||
| ## License | ||||
|  | ||||
| This work is licensed under either of | ||||
|  | ||||
| - Apache License, Version 2.0 ([LICENSE-APACHE](LICENSE-APACHE) or | ||||
|   <http://www.apache.org/licenses/LICENSE-2.0>) | ||||
| - MIT license ([LICENSE-MIT](LICENSE-MIT) or <http://opensource.org/licenses/MIT>) | ||||
|  | ||||
| at your option. | ||||
| @@ -1,24 +1,23 @@ | ||||
| #![no_std] | ||||
| #![feature(generic_associated_types)] | ||||
| #![feature(type_alias_impl_trait)] | ||||
| #![allow(incomplete_features)] | ||||
| #![feature(generic_const_exprs)] | ||||
|  | ||||
| #![warn(missing_docs)] | ||||
| #![doc = include_str!("../README.md")] | ||||
| mod fmt; | ||||
|  | ||||
| pub use embassy_boot::{FirmwareUpdater, FlashConfig, FlashProvider, Partition, SingleFlashProvider}; | ||||
| pub use embassy_boot::{AlignedBuffer, BootFlash, FirmwareUpdater, FlashConfig, Partition, SingleFlashConfig}; | ||||
| use embassy_nrf::nvmc::{Nvmc, PAGE_SIZE}; | ||||
| use embassy_nrf::peripherals::WDT; | ||||
| use embassy_nrf::wdt; | ||||
| use embedded_storage::nor_flash::{ErrorType, NorFlash, ReadNorFlash}; | ||||
|  | ||||
| pub struct BootLoader { | ||||
|     boot: embassy_boot::BootLoader<PAGE_SIZE>, | ||||
| /// A bootloader for nRF devices. | ||||
| pub struct BootLoader<const BUFFER_SIZE: usize = PAGE_SIZE> { | ||||
|     boot: embassy_boot::BootLoader, | ||||
|     aligned_buf: AlignedBuffer<BUFFER_SIZE>, | ||||
| } | ||||
|  | ||||
| impl BootLoader { | ||||
| impl Default for BootLoader<PAGE_SIZE> { | ||||
|     /// Create a new bootloader instance using parameters from linker script | ||||
|     pub fn default() -> Self { | ||||
|     fn default() -> Self { | ||||
|         extern "C" { | ||||
|             static __bootloader_state_start: u32; | ||||
|             static __bootloader_state_end: u32; | ||||
| @@ -30,20 +29,20 @@ impl BootLoader { | ||||
|  | ||||
|         let active = unsafe { | ||||
|             Partition::new( | ||||
|                 &__bootloader_active_start as *const u32 as usize, | ||||
|                 &__bootloader_active_end as *const u32 as usize, | ||||
|                 &__bootloader_active_start as *const u32 as u32, | ||||
|                 &__bootloader_active_end as *const u32 as u32, | ||||
|             ) | ||||
|         }; | ||||
|         let dfu = unsafe { | ||||
|             Partition::new( | ||||
|                 &__bootloader_dfu_start as *const u32 as usize, | ||||
|                 &__bootloader_dfu_end as *const u32 as usize, | ||||
|                 &__bootloader_dfu_start as *const u32 as u32, | ||||
|                 &__bootloader_dfu_end as *const u32 as u32, | ||||
|             ) | ||||
|         }; | ||||
|         let state = unsafe { | ||||
|             Partition::new( | ||||
|                 &__bootloader_state_start as *const u32 as usize, | ||||
|                 &__bootloader_state_end as *const u32 as usize, | ||||
|                 &__bootloader_state_start as *const u32 as u32, | ||||
|                 &__bootloader_state_end as *const u32 as u32, | ||||
|             ) | ||||
|         }; | ||||
|  | ||||
| @@ -53,26 +52,31 @@ impl BootLoader { | ||||
|  | ||||
|         Self::new(active, dfu, state) | ||||
|     } | ||||
| } | ||||
|  | ||||
| impl<const BUFFER_SIZE: usize> BootLoader<BUFFER_SIZE> { | ||||
|     /// Create a new bootloader instance using the supplied partitions for active, dfu and state. | ||||
|     pub fn new(active: Partition, dfu: Partition, state: Partition) -> Self { | ||||
|         Self { | ||||
|             boot: embassy_boot::BootLoader::new(active, dfu, state), | ||||
|             aligned_buf: AlignedBuffer([0; BUFFER_SIZE]), | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     /// Boots the application without softdevice mechanisms | ||||
|     pub fn prepare<F: FlashProvider>(&mut self, flash: &mut F) -> usize | ||||
|     where | ||||
|         [(); <<F as FlashProvider>::STATE as FlashConfig>::FLASH::WRITE_SIZE]:, | ||||
|         [(); <<F as FlashProvider>::ACTIVE as FlashConfig>::FLASH::ERASE_SIZE]:, | ||||
|     { | ||||
|         match self.boot.prepare_boot(flash) { | ||||
|     /// Inspect the bootloader state and perform actions required before booting, such as swapping | ||||
|     /// firmware. | ||||
|     pub fn prepare<F: FlashConfig>(&mut self, flash: &mut F) -> usize { | ||||
|         match self.boot.prepare_boot(flash, &mut self.aligned_buf.0) { | ||||
|             Ok(_) => self.boot.boot_address(), | ||||
|             Err(_) => panic!("boot prepare error!"), | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     /// Boots the application without softdevice mechanisms. | ||||
|     /// | ||||
|     /// # Safety | ||||
|     /// | ||||
|     /// This modifies the stack pointer and reset vector and will run code placed in the active partition. | ||||
|     #[cfg(not(feature = "softdevice"))] | ||||
|     pub unsafe fn load(&mut self, start: usize) -> ! { | ||||
|         let mut p = cortex_m::Peripherals::steal(); | ||||
| @@ -81,6 +85,11 @@ impl BootLoader { | ||||
|         cortex_m::asm::bootload(start as *const u32) | ||||
|     } | ||||
|  | ||||
|     /// Boots the application assuming softdevice is present. | ||||
|     /// | ||||
|     /// # Safety | ||||
|     /// | ||||
|     /// This modifies the stack pointer and reset vector and will run code placed in the active partition. | ||||
|     #[cfg(feature = "softdevice")] | ||||
|     pub unsafe fn load(&mut self, _app: usize) -> ! { | ||||
|         use nrf_softdevice_mbr as mbr; | ||||
| @@ -137,11 +146,7 @@ pub struct WatchdogFlash<'d> { | ||||
|  | ||||
| impl<'d> WatchdogFlash<'d> { | ||||
|     /// Start a new watchdog with a given flash and WDT peripheral and a timeout | ||||
|     pub fn start(flash: Nvmc<'d>, wdt: WDT, timeout: u32) -> Self { | ||||
|         let mut config = wdt::Config::default(); | ||||
|         config.timeout_ticks = 32768 * timeout; // timeout seconds | ||||
|         config.run_during_sleep = true; | ||||
|         config.run_during_debug_halt = false; | ||||
|     pub fn start(flash: Nvmc<'d>, wdt: WDT, config: wdt::Config) -> Self { | ||||
|         let (_wdt, [wdt]) = match wdt::Watchdog::try_new(wdt, config) { | ||||
|             Ok(x) => x, | ||||
|             Err(_) => { | ||||
|   | ||||
							
								
								
									
										79
									
								
								embassy-boot/rp/Cargo.toml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										79
									
								
								embassy-boot/rp/Cargo.toml
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,79 @@ | ||||
| [package] | ||||
| edition = "2021" | ||||
| name = "embassy-boot-rp" | ||||
| version = "0.1.0" | ||||
| description = "Bootloader lib for RP2040 chips" | ||||
| license = "MIT OR Apache-2.0" | ||||
|  | ||||
| [package.metadata.embassy_docs] | ||||
| src_base = "https://github.com/embassy-rs/embassy/blob/embassy-boot-rp-v$VERSION/src/" | ||||
| src_base_git = "https://github.com/embassy-rs/embassy/blob/$COMMIT/embassy-boot/rp/src/" | ||||
| target = "thumbv6m-none-eabi" | ||||
|  | ||||
| [lib] | ||||
|  | ||||
| [dependencies] | ||||
| defmt = { version = "0.3", optional = true } | ||||
| defmt-rtt = { version = "0.4", optional = true } | ||||
| log = { version = "0.4", optional = true } | ||||
|  | ||||
| embassy-sync = { path = "../../embassy-sync" } | ||||
| embassy-rp = { path = "../../embassy-rp", default-features = false } | ||||
| embassy-boot = { path = "../boot", default-features = false } | ||||
| embassy-time = { path = "../../embassy-time" } | ||||
|  | ||||
| cortex-m = { version = "0.7.6" } | ||||
| cortex-m-rt = { version = "0.7" } | ||||
| embedded-storage = "0.3.0" | ||||
| embedded-storage-async = { version = "0.4.0", optional = true } | ||||
| cfg-if = "1.0.0" | ||||
|  | ||||
| [features] | ||||
| defmt = [ | ||||
|     "dep:defmt", | ||||
|     "embassy-boot/defmt", | ||||
|     "embassy-rp/defmt", | ||||
| ] | ||||
| log = [ | ||||
|     "dep:log", | ||||
|     "embassy-boot/log", | ||||
|     "embassy-rp/log", | ||||
| ] | ||||
| debug = ["defmt-rtt"] | ||||
| nightly = [ | ||||
|     "dep:embedded-storage-async", | ||||
|     "embassy-boot/nightly", | ||||
|     "embassy-rp/nightly", | ||||
|     "embassy-time/nightly" | ||||
| ] | ||||
|  | ||||
| [profile.dev] | ||||
| debug = 2 | ||||
| debug-assertions = true | ||||
| incremental = false | ||||
| opt-level = 'z' | ||||
| overflow-checks = true | ||||
|  | ||||
| [profile.release] | ||||
| codegen-units = 1 | ||||
| debug = 2 | ||||
| debug-assertions = false | ||||
| incremental = false | ||||
| lto = 'fat' | ||||
| opt-level = 'z' | ||||
| overflow-checks = false | ||||
|  | ||||
| # do not optimize proc-macro crates = faster builds from scratch | ||||
| [profile.dev.build-override] | ||||
| codegen-units = 8 | ||||
| debug = false | ||||
| debug-assertions = false | ||||
| opt-level = 0 | ||||
| overflow-checks = false | ||||
|  | ||||
| [profile.release.build-override] | ||||
| codegen-units = 8 | ||||
| debug = false | ||||
| debug-assertions = false | ||||
| opt-level = 0 | ||||
| overflow-checks = false | ||||
							
								
								
									
										26
									
								
								embassy-boot/rp/README.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										26
									
								
								embassy-boot/rp/README.md
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,26 @@ | ||||
| # embassy-boot-rp | ||||
|  | ||||
| An [Embassy](https://embassy.dev) project. | ||||
|  | ||||
| An adaptation of `embassy-boot` for RP2040. | ||||
|  | ||||
| NOTE: The applications using this bootloader should not link with the `link-rp.x` linker script. | ||||
|  | ||||
| ## Features | ||||
|  | ||||
| * Configure bootloader partitions based on linker script. | ||||
| * Load applications from active partition. | ||||
|  | ||||
| ## Minimum supported Rust version (MSRV) | ||||
|  | ||||
| `embassy-boot-rp` is guaranteed to compile on the latest stable Rust version at the time of release. It might compile with older versions but that may change in any new patch release. | ||||
|  | ||||
| ## License | ||||
|  | ||||
| This work is licensed under either of | ||||
|  | ||||
| - Apache License, Version 2.0 ([LICENSE-APACHE](LICENSE-APACHE) or | ||||
|   <http://www.apache.org/licenses/LICENSE-2.0>) | ||||
| - MIT license ([LICENSE-MIT](LICENSE-MIT) or <http://opensource.org/licenses/MIT>) | ||||
|  | ||||
| at your option. | ||||
							
								
								
									
										8
									
								
								embassy-boot/rp/build.rs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										8
									
								
								embassy-boot/rp/build.rs
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,8 @@ | ||||
| use std::env; | ||||
|  | ||||
| fn main() { | ||||
|     let target = env::var("TARGET").unwrap(); | ||||
|     if target.starts_with("thumbv6m-") { | ||||
|         println!("cargo:rustc-cfg=armv6m"); | ||||
|     } | ||||
| } | ||||
							
								
								
									
										136
									
								
								embassy-boot/rp/src/lib.rs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										136
									
								
								embassy-boot/rp/src/lib.rs
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,136 @@ | ||||
| #![no_std] | ||||
| #![warn(missing_docs)] | ||||
| #![doc = include_str!("../README.md")] | ||||
| mod fmt; | ||||
|  | ||||
| pub use embassy_boot::{AlignedBuffer, BootFlash, FirmwareUpdater, FlashConfig, Partition, SingleFlashConfig, State}; | ||||
| use embassy_rp::flash::{Flash, ERASE_SIZE}; | ||||
| use embassy_rp::peripherals::{FLASH, WATCHDOG}; | ||||
| use embassy_rp::watchdog::Watchdog; | ||||
| use embassy_time::Duration; | ||||
| use embedded_storage::nor_flash::{ErrorType, NorFlash, ReadNorFlash}; | ||||
|  | ||||
| /// A bootloader for RP2040 devices. | ||||
| pub struct BootLoader<const BUFFER_SIZE: usize = ERASE_SIZE> { | ||||
|     boot: embassy_boot::BootLoader, | ||||
|     aligned_buf: AlignedBuffer<BUFFER_SIZE>, | ||||
| } | ||||
|  | ||||
| impl<const BUFFER_SIZE: usize> BootLoader<BUFFER_SIZE> { | ||||
|     /// Create a new bootloader instance using the supplied partitions for active, dfu and state. | ||||
|     pub fn new(active: Partition, dfu: Partition, state: Partition) -> Self { | ||||
|         Self { | ||||
|             boot: embassy_boot::BootLoader::new(active, dfu, state), | ||||
|             aligned_buf: AlignedBuffer([0; BUFFER_SIZE]), | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     /// Inspect the bootloader state and perform actions required before booting, such as swapping | ||||
|     /// firmware. | ||||
|     pub fn prepare<F: FlashConfig>(&mut self, flash: &mut F) -> usize { | ||||
|         match self.boot.prepare_boot(flash, self.aligned_buf.as_mut()) { | ||||
|             Ok(_) => embassy_rp::flash::FLASH_BASE + self.boot.boot_address(), | ||||
|             Err(_) => panic!("boot prepare error!"), | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     /// Boots the application. | ||||
|     /// | ||||
|     /// # Safety | ||||
|     /// | ||||
|     /// This modifies the stack pointer and reset vector and will run code placed in the active partition. | ||||
|     pub unsafe fn load(&mut self, start: usize) -> ! { | ||||
|         trace!("Loading app at 0x{:x}", start); | ||||
|         #[allow(unused_mut)] | ||||
|         let mut p = cortex_m::Peripherals::steal(); | ||||
|         #[cfg(not(armv6m))] | ||||
|         p.SCB.invalidate_icache(); | ||||
|         p.SCB.vtor.write(start as u32); | ||||
|  | ||||
|         cortex_m::asm::bootload(start as *const u32) | ||||
|     } | ||||
| } | ||||
|  | ||||
| impl Default for BootLoader<ERASE_SIZE> { | ||||
|     /// Create a new bootloader instance using parameters from linker script | ||||
|     fn default() -> Self { | ||||
|         extern "C" { | ||||
|             static __bootloader_state_start: u32; | ||||
|             static __bootloader_state_end: u32; | ||||
|             static __bootloader_active_start: u32; | ||||
|             static __bootloader_active_end: u32; | ||||
|             static __bootloader_dfu_start: u32; | ||||
|             static __bootloader_dfu_end: u32; | ||||
|         } | ||||
|  | ||||
|         let active = unsafe { | ||||
|             Partition::new( | ||||
|                 &__bootloader_active_start as *const u32 as u32, | ||||
|                 &__bootloader_active_end as *const u32 as u32, | ||||
|             ) | ||||
|         }; | ||||
|         let dfu = unsafe { | ||||
|             Partition::new( | ||||
|                 &__bootloader_dfu_start as *const u32 as u32, | ||||
|                 &__bootloader_dfu_end as *const u32 as u32, | ||||
|             ) | ||||
|         }; | ||||
|         let state = unsafe { | ||||
|             Partition::new( | ||||
|                 &__bootloader_state_start as *const u32 as u32, | ||||
|                 &__bootloader_state_end as *const u32 as u32, | ||||
|             ) | ||||
|         }; | ||||
|  | ||||
|         trace!("ACTIVE: 0x{:x} - 0x{:x}", active.from, active.to); | ||||
|         trace!("DFU: 0x{:x} - 0x{:x}", dfu.from, dfu.to); | ||||
|         trace!("STATE: 0x{:x} - 0x{:x}", state.from, state.to); | ||||
|  | ||||
|         Self::new(active, dfu, state) | ||||
|     } | ||||
| } | ||||
|  | ||||
| /// A flash implementation that will feed a watchdog when touching flash. | ||||
| pub struct WatchdogFlash<'d, const SIZE: usize> { | ||||
|     flash: Flash<'d, FLASH, SIZE>, | ||||
|     watchdog: Watchdog, | ||||
| } | ||||
|  | ||||
| impl<'d, const SIZE: usize> WatchdogFlash<'d, SIZE> { | ||||
|     /// Start a new watchdog with a given flash and watchdog peripheral and a timeout | ||||
|     pub fn start(flash: FLASH, watchdog: WATCHDOG, timeout: Duration) -> Self { | ||||
|         let flash: Flash<'_, FLASH, SIZE> = Flash::new(flash); | ||||
|         let mut watchdog = Watchdog::new(watchdog); | ||||
|         watchdog.start(timeout); | ||||
|         Self { flash, watchdog } | ||||
|     } | ||||
| } | ||||
|  | ||||
| impl<'d, const SIZE: usize> ErrorType for WatchdogFlash<'d, SIZE> { | ||||
|     type Error = <Flash<'d, FLASH, SIZE> as ErrorType>::Error; | ||||
| } | ||||
|  | ||||
| impl<'d, const SIZE: usize> NorFlash for WatchdogFlash<'d, SIZE> { | ||||
|     const WRITE_SIZE: usize = <Flash<'d, FLASH, SIZE> as NorFlash>::WRITE_SIZE; | ||||
|     const ERASE_SIZE: usize = <Flash<'d, FLASH, SIZE> as NorFlash>::ERASE_SIZE; | ||||
|  | ||||
|     fn erase(&mut self, from: u32, to: u32) -> Result<(), Self::Error> { | ||||
|         self.watchdog.feed(); | ||||
|         self.flash.erase(from, to) | ||||
|     } | ||||
|     fn write(&mut self, offset: u32, data: &[u8]) -> Result<(), Self::Error> { | ||||
|         self.watchdog.feed(); | ||||
|         self.flash.write(offset, data) | ||||
|     } | ||||
| } | ||||
|  | ||||
| impl<'d, const SIZE: usize> ReadNorFlash for WatchdogFlash<'d, SIZE> { | ||||
|     const READ_SIZE: usize = <Flash<'d, FLASH, SIZE> as ReadNorFlash>::READ_SIZE; | ||||
|     fn read(&mut self, offset: u32, data: &mut [u8]) -> Result<(), Self::Error> { | ||||
|         self.watchdog.feed(); | ||||
|         self.flash.read(offset, data) | ||||
|     } | ||||
|     fn capacity(&self) -> usize { | ||||
|         self.flash.capacity() | ||||
|     } | ||||
| } | ||||
| @@ -3,6 +3,7 @@ edition = "2021" | ||||
| name = "embassy-boot-stm32" | ||||
| version = "0.1.0" | ||||
| description = "Bootloader lib for STM32 chips" | ||||
| license = "MIT OR Apache-2.0" | ||||
|  | ||||
| [package.metadata.embassy_docs] | ||||
| src_base = "https://github.com/embassy-rs/embassy/blob/embassy-boot-nrf-v$VERSION/embassy-boot/stm32/src/" | ||||
| @@ -14,16 +15,16 @@ target = "thumbv7em-none-eabi" | ||||
|  | ||||
| [dependencies] | ||||
| defmt = { version = "0.3", optional = true } | ||||
| defmt-rtt = { version = "0.3", optional = true } | ||||
| defmt-rtt = { version = "0.4", optional = true } | ||||
| log = { version = "0.4", optional = true } | ||||
|  | ||||
| embassy-sync = { path = "../../embassy-sync" } | ||||
| embassy-stm32 = { path = "../../embassy-stm32", default-features = false, features = ["nightly"] } | ||||
| embassy-stm32 = { path = "../../embassy-stm32", default-features = false } | ||||
| embassy-boot = { path = "../boot", default-features = false } | ||||
| cortex-m = { version = "0.7.6" } | ||||
| cortex-m-rt = { version = "0.7" } | ||||
| embedded-storage = "0.3.0" | ||||
| embedded-storage-async = "0.3.0" | ||||
| embedded-storage-async = { version = "0.4.0", optional = true } | ||||
| cfg-if = "1.0.0" | ||||
|  | ||||
| [features] | ||||
| @@ -38,6 +39,11 @@ log = [ | ||||
|     "embassy-stm32/log", | ||||
| ] | ||||
| debug = ["defmt-rtt"] | ||||
| nightly = [ | ||||
|     "dep:embedded-storage-async", | ||||
|     "embassy-boot/nightly", | ||||
|     "embassy-stm32/nightly" | ||||
| ] | ||||
|  | ||||
| [profile.dev] | ||||
| debug = 2 | ||||
|   | ||||
| @@ -1,11 +1,24 @@ | ||||
| # Bootloader for STM32 | ||||
| # embassy-boot-stm32 | ||||
|  | ||||
| The bootloader uses `embassy-boot` to interact with the flash. | ||||
| An [Embassy](https://embassy.dev) project. | ||||
|  | ||||
| # Usage | ||||
| An adaptation of `embassy-boot` for STM32. | ||||
|  | ||||
| Flash the bootloader | ||||
| ## Features | ||||
|  | ||||
| ``` | ||||
| cargo flash --features embassy-stm32/stm32wl55jc-cm4 --release --chip STM32WLE5JCIx | ||||
| ``` | ||||
| * Configure bootloader partitions based on linker script. | ||||
| * Load applications from active partition. | ||||
|  | ||||
| ## Minimum supported Rust version (MSRV) | ||||
|  | ||||
| `embassy-boot-stm32` is guaranteed to compile on the latest stable Rust version at the time of release. It might compile with older versions but that may change in any new patch release. | ||||
|  | ||||
| ## License | ||||
|  | ||||
| This work is licensed under either of | ||||
|  | ||||
| - Apache License, Version 2.0 ([LICENSE-APACHE](LICENSE-APACHE) or | ||||
|   <http://www.apache.org/licenses/LICENSE-2.0>) | ||||
| - MIT license ([LICENSE-MIT](LICENSE-MIT) or <http://opensource.org/licenses/MIT>) | ||||
|  | ||||
| at your option. | ||||
|   | ||||
| @@ -1,75 +1,39 @@ | ||||
| #![no_std] | ||||
| #![feature(generic_associated_types)] | ||||
| #![feature(type_alias_impl_trait)] | ||||
| #![allow(incomplete_features)] | ||||
| #![feature(generic_const_exprs)] | ||||
|  | ||||
| #![warn(missing_docs)] | ||||
| #![doc = include_str!("../README.md")] | ||||
| mod fmt; | ||||
|  | ||||
| pub use embassy_boot::{FirmwareUpdater, FlashConfig, FlashProvider, Partition, SingleFlashProvider, State}; | ||||
| use embedded_storage::nor_flash::NorFlash; | ||||
| pub use embassy_boot::{AlignedBuffer, BootFlash, FirmwareUpdater, FlashConfig, Partition, SingleFlashConfig, State}; | ||||
|  | ||||
| pub struct BootLoader<const PAGE_SIZE: usize> { | ||||
|     boot: embassy_boot::BootLoader<PAGE_SIZE>, | ||||
| /// A bootloader for STM32 devices. | ||||
| pub struct BootLoader<const BUFFER_SIZE: usize> { | ||||
|     boot: embassy_boot::BootLoader, | ||||
|     aligned_buf: AlignedBuffer<BUFFER_SIZE>, | ||||
| } | ||||
|  | ||||
| impl<const PAGE_SIZE: usize> BootLoader<PAGE_SIZE> { | ||||
|     /// Create a new bootloader instance using parameters from linker script | ||||
|     pub fn default() -> Self { | ||||
|         extern "C" { | ||||
|             static __bootloader_state_start: u32; | ||||
|             static __bootloader_state_end: u32; | ||||
|             static __bootloader_active_start: u32; | ||||
|             static __bootloader_active_end: u32; | ||||
|             static __bootloader_dfu_start: u32; | ||||
|             static __bootloader_dfu_end: u32; | ||||
|         } | ||||
|  | ||||
|         let active = unsafe { | ||||
|             Partition::new( | ||||
|                 &__bootloader_active_start as *const u32 as usize, | ||||
|                 &__bootloader_active_end as *const u32 as usize, | ||||
|             ) | ||||
|         }; | ||||
|         let dfu = unsafe { | ||||
|             Partition::new( | ||||
|                 &__bootloader_dfu_start as *const u32 as usize, | ||||
|                 &__bootloader_dfu_end as *const u32 as usize, | ||||
|             ) | ||||
|         }; | ||||
|         let state = unsafe { | ||||
|             Partition::new( | ||||
|                 &__bootloader_state_start as *const u32 as usize, | ||||
|                 &__bootloader_state_end as *const u32 as usize, | ||||
|             ) | ||||
|         }; | ||||
|  | ||||
|         trace!("ACTIVE: 0x{:x} - 0x{:x}", active.from, active.to); | ||||
|         trace!("DFU: 0x{:x} - 0x{:x}", dfu.from, dfu.to); | ||||
|         trace!("STATE: 0x{:x} - 0x{:x}", state.from, state.to); | ||||
|  | ||||
|         Self::new(active, dfu, state) | ||||
|     } | ||||
|  | ||||
| impl<const BUFFER_SIZE: usize> BootLoader<BUFFER_SIZE> { | ||||
|     /// Create a new bootloader instance using the supplied partitions for active, dfu and state. | ||||
|     pub fn new(active: Partition, dfu: Partition, state: Partition) -> Self { | ||||
|         Self { | ||||
|             boot: embassy_boot::BootLoader::new(active, dfu, state), | ||||
|             aligned_buf: AlignedBuffer([0; BUFFER_SIZE]), | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     /// Boots the application | ||||
|     pub fn prepare<F: FlashProvider>(&mut self, flash: &mut F) -> usize | ||||
|     where | ||||
|         [(); <<F as FlashProvider>::STATE as FlashConfig>::FLASH::WRITE_SIZE]:, | ||||
|         [(); <<F as FlashProvider>::ACTIVE as FlashConfig>::FLASH::ERASE_SIZE]:, | ||||
|     { | ||||
|         match self.boot.prepare_boot(flash) { | ||||
|     /// Inspect the bootloader state and perform actions required before booting, such as swapping | ||||
|     /// firmware. | ||||
|     pub fn prepare<F: FlashConfig>(&mut self, flash: &mut F) -> usize { | ||||
|         match self.boot.prepare_boot(flash, self.aligned_buf.as_mut()) { | ||||
|             Ok(_) => embassy_stm32::flash::FLASH_BASE + self.boot.boot_address(), | ||||
|             Err(_) => panic!("boot prepare error!"), | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     /// Boots the application. | ||||
|     /// | ||||
|     /// # Safety | ||||
|     /// | ||||
|     /// This modifies the stack pointer and reset vector and will run code placed in the active partition. | ||||
|     pub unsafe fn load(&mut self, start: usize) -> ! { | ||||
|         trace!("Loading app at 0x{:x}", start); | ||||
|         #[allow(unused_mut)] | ||||
| @@ -81,3 +45,42 @@ impl<const PAGE_SIZE: usize> BootLoader<PAGE_SIZE> { | ||||
|         cortex_m::asm::bootload(start as *const u32) | ||||
|     } | ||||
| } | ||||
|  | ||||
| impl<const BUFFER_SIZE: usize> Default for BootLoader<BUFFER_SIZE> { | ||||
|     /// Create a new bootloader instance using parameters from linker script | ||||
|     fn default() -> Self { | ||||
|         extern "C" { | ||||
|             static __bootloader_state_start: u32; | ||||
|             static __bootloader_state_end: u32; | ||||
|             static __bootloader_active_start: u32; | ||||
|             static __bootloader_active_end: u32; | ||||
|             static __bootloader_dfu_start: u32; | ||||
|             static __bootloader_dfu_end: u32; | ||||
|         } | ||||
|  | ||||
|         let active = unsafe { | ||||
|             Partition::new( | ||||
|                 &__bootloader_active_start as *const u32 as u32, | ||||
|                 &__bootloader_active_end as *const u32 as u32, | ||||
|             ) | ||||
|         }; | ||||
|         let dfu = unsafe { | ||||
|             Partition::new( | ||||
|                 &__bootloader_dfu_start as *const u32 as u32, | ||||
|                 &__bootloader_dfu_end as *const u32 as u32, | ||||
|             ) | ||||
|         }; | ||||
|         let state = unsafe { | ||||
|             Partition::new( | ||||
|                 &__bootloader_state_start as *const u32 as u32, | ||||
|                 &__bootloader_state_end as *const u32 as u32, | ||||
|             ) | ||||
|         }; | ||||
|  | ||||
|         trace!("ACTIVE: 0x{:x} - 0x{:x}", active.from, active.to); | ||||
|         trace!("DFU: 0x{:x} - 0x{:x}", dfu.from, dfu.to); | ||||
|         trace!("STATE: 0x{:x} - 0x{:x}", state.from, state.to); | ||||
|  | ||||
|         Self::new(active, dfu, state) | ||||
|     } | ||||
| } | ||||
|   | ||||
| @@ -2,6 +2,7 @@ | ||||
| name = "embassy-cortex-m" | ||||
| version = "0.1.0" | ||||
| edition = "2021" | ||||
| license = "MIT OR Apache-2.0" | ||||
|  | ||||
| [package.metadata.embassy_docs] | ||||
| src_base = "https://github.com/embassy-rs/embassy/blob/embassy-cortex-m-v$VERSION/embassy-cortex-m/src/" | ||||
| @@ -35,9 +36,9 @@ prio-bits-8 = [] | ||||
| defmt = { version = "0.3", optional = true } | ||||
| log = { version = "0.4.14", optional = true } | ||||
|  | ||||
| embassy-sync = { version = "0.1.0", path = "../embassy-sync" } | ||||
| embassy-executor = { version = "0.1.0", path = "../embassy-executor"} | ||||
| embassy-macros = { version = "0.1.0", path = "../embassy-macros"} | ||||
| embassy-sync = { version = "0.2.0", path = "../embassy-sync" } | ||||
| embassy-executor = { version = "0.2.0", path = "../embassy-executor"} | ||||
| embassy-macros = { version = "0.2.0", path = "../embassy-macros"} | ||||
| embassy-hal-common = { version = "0.1.0", path = "../embassy-hal-common"} | ||||
| atomic-polyfill = "1.0.1" | ||||
| critical-section = "1.1" | ||||
|   | ||||
| @@ -1,89 +0,0 @@ | ||||
| //! Executor specific to cortex-m devices. | ||||
| use core::marker::PhantomData; | ||||
|  | ||||
| pub use embassy_executor::*; | ||||
|  | ||||
| use crate::interrupt::{Interrupt, InterruptExt}; | ||||
|  | ||||
| fn pend_by_number(n: u16) { | ||||
|     #[derive(Clone, Copy)] | ||||
|     struct N(u16); | ||||
|     unsafe impl cortex_m::interrupt::InterruptNumber for N { | ||||
|         fn number(self) -> u16 { | ||||
|             self.0 | ||||
|         } | ||||
|     } | ||||
|     cortex_m::peripheral::NVIC::pend(N(n)) | ||||
| } | ||||
|  | ||||
| /// Interrupt mode executor. | ||||
| /// | ||||
| /// This executor runs tasks in interrupt mode. The interrupt handler is set up | ||||
| /// to poll tasks, and when a task is woken the interrupt is pended from software. | ||||
| /// | ||||
| /// This allows running async tasks at a priority higher than thread mode. One | ||||
| /// use case is to leave thread mode free for non-async tasks. Another use case is | ||||
| /// to run multiple executors: one in thread mode for low priority tasks and another in | ||||
| /// interrupt mode for higher priority tasks. Higher priority tasks will preempt lower | ||||
| /// priority ones. | ||||
| /// | ||||
| /// It is even possible to run multiple interrupt mode executors at different priorities, | ||||
| /// by assigning different priorities to the interrupts. For an example on how to do this, | ||||
| /// See the 'multiprio' example for 'embassy-nrf'. | ||||
| /// | ||||
| /// To use it, you have to pick an interrupt that won't be used by the hardware. | ||||
| /// Some chips reserve some interrupts for this purpose, sometimes named "software interrupts" (SWI). | ||||
| /// If this is not the case, you may use an interrupt from any unused peripheral. | ||||
| /// | ||||
| /// It is somewhat more complex to use, it's recommended to use the thread-mode | ||||
| /// [`Executor`] instead, if it works for your use case. | ||||
| pub struct InterruptExecutor<I: Interrupt> { | ||||
|     irq: I, | ||||
|     inner: raw::Executor, | ||||
|     not_send: PhantomData<*mut ()>, | ||||
| } | ||||
|  | ||||
| impl<I: Interrupt> InterruptExecutor<I> { | ||||
|     /// Create a new Executor. | ||||
|     pub fn new(irq: I) -> Self { | ||||
|         let ctx = irq.number() as *mut (); | ||||
|         Self { | ||||
|             irq, | ||||
|             inner: raw::Executor::new(|ctx| pend_by_number(ctx as u16), ctx), | ||||
|             not_send: PhantomData, | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     /// Start the executor. | ||||
|     /// | ||||
|     /// This initializes the executor, configures and enables the interrupt, and returns. | ||||
|     /// The executor keeps running in the background through the interrupt. | ||||
|     /// | ||||
|     /// This returns a [`SendSpawner`] you can use to spawn tasks on it. A [`SendSpawner`] | ||||
|     /// is returned instead of a [`Spawner`](embassy_executor::Spawner) because the executor effectively runs in a | ||||
|     /// different "thread" (the interrupt), so spawning tasks on it is effectively | ||||
|     /// sending them. | ||||
|     /// | ||||
|     /// To obtain a [`Spawner`](embassy_executor::Spawner) for this executor, use [`Spawner::for_current_executor()`](embassy_executor::Spawner::for_current_executor()) from | ||||
|     /// a task running in it. | ||||
|     /// | ||||
|     /// This function requires `&'static mut self`. This means you have to store the | ||||
|     /// Executor instance in a place where it'll live forever and grants you mutable | ||||
|     /// access. There's a few ways to do this: | ||||
|     /// | ||||
|     /// - a [StaticCell](https://docs.rs/static_cell/latest/static_cell/) (safe) | ||||
|     /// - a `static mut` (unsafe) | ||||
|     /// - a local variable in a function you know never returns (like `fn main() -> !`), upgrading its lifetime with `transmute`. (unsafe) | ||||
|     pub fn start(&'static mut self) -> SendSpawner { | ||||
|         self.irq.disable(); | ||||
|  | ||||
|         self.irq.set_handler(|ctx| unsafe { | ||||
|             let executor = &*(ctx as *const raw::Executor); | ||||
|             executor.poll(); | ||||
|         }); | ||||
|         self.irq.set_handler_context(&self.inner as *const _ as _); | ||||
|         self.irq.enable(); | ||||
|  | ||||
|         self.inner.spawner().make_send() | ||||
|     } | ||||
| } | ||||
| @@ -13,14 +13,44 @@ pub mod _export { | ||||
|     pub use embassy_macros::{cortex_m_interrupt as interrupt, cortex_m_interrupt_declare as declare}; | ||||
| } | ||||
|  | ||||
| /// Interrupt handler trait. | ||||
| /// | ||||
| /// Drivers that need to handle interrupts implement this trait. | ||||
| /// The user must ensure `on_interrupt()` is called every time the interrupt fires. | ||||
| /// Drivers must use use [`Binding`] to assert at compile time that the user has done so. | ||||
| pub trait Handler<I: Interrupt> { | ||||
|     /// Interrupt handler function. | ||||
|     /// | ||||
|     /// Must be called every time the `I` interrupt fires, synchronously from | ||||
|     /// the interrupt handler context. | ||||
|     /// | ||||
|     /// # Safety | ||||
|     /// | ||||
|     /// This function must ONLY be called from the interrupt handler for `I`. | ||||
|     unsafe fn on_interrupt(); | ||||
| } | ||||
|  | ||||
| /// Compile-time assertion that an interrupt has been bound to a handler. | ||||
| /// | ||||
| /// For the vast majority of cases, you should use the `bind_interrupts!` | ||||
| /// macro instead of writing `unsafe impl`s of this trait. | ||||
| /// | ||||
| /// # Safety | ||||
| /// | ||||
| /// By implementing this trait, you are asserting that you have arranged for `H::on_interrupt()` | ||||
| /// to be called every time the `I` interrupt fires. | ||||
| /// | ||||
| /// This allows drivers to check bindings at compile-time. | ||||
| pub unsafe trait Binding<I: Interrupt, H: Handler<I>> {} | ||||
|  | ||||
| /// Implementation detail, do not use outside embassy crates. | ||||
| #[doc(hidden)] | ||||
| pub struct Handler { | ||||
| pub struct DynHandler { | ||||
|     pub func: AtomicPtr<()>, | ||||
|     pub ctx: AtomicPtr<()>, | ||||
| } | ||||
|  | ||||
| impl Handler { | ||||
| impl DynHandler { | ||||
|     pub const fn new() -> Self { | ||||
|         Self { | ||||
|             func: AtomicPtr::new(ptr::null_mut()), | ||||
| @@ -51,7 +81,7 @@ pub unsafe trait Interrupt: Peripheral<P = Self> { | ||||
|  | ||||
|     /// Implementation detail, do not use outside embassy crates. | ||||
|     #[doc(hidden)] | ||||
|     unsafe fn __handler(&self) -> &'static Handler; | ||||
|     unsafe fn __handler(&self) -> &'static DynHandler; | ||||
| } | ||||
|  | ||||
| /// Represents additional behavior for all interrupts. | ||||
|   | ||||
| @@ -5,6 +5,6 @@ | ||||
| // This mod MUST go first, so that the others see its macros. | ||||
| pub(crate) mod fmt; | ||||
|  | ||||
| pub mod executor; | ||||
| pub use embassy_executor as executor; | ||||
| pub mod interrupt; | ||||
| pub mod peripheral; | ||||
|   | ||||
| @@ -2,13 +2,14 @@ | ||||
| name = "embassy-embedded-hal" | ||||
| version = "0.1.0" | ||||
| edition = "2021" | ||||
| license = "MIT OR Apache-2.0" | ||||
|  | ||||
|  | ||||
| [package.metadata.embassy_docs] | ||||
| src_base = "https://github.com/embassy-rs/embassy/blob/embassy-embedded-hal-v$VERSION/embassy-embedded-hal/src/" | ||||
| src_base_git = "https://github.com/embassy-rs/embassy/blob/$COMMIT/embassy-embedded-hal/src/" | ||||
| features = ["nightly", "std"] | ||||
| target = "thumbv7em-none-eabi" | ||||
| target = "x86_64-unknown-linux-gnu" | ||||
|  | ||||
| [features] | ||||
| std = [] | ||||
| @@ -16,12 +17,12 @@ std = [] | ||||
| nightly = ["embedded-hal-async", "embedded-storage-async"] | ||||
|  | ||||
| [dependencies] | ||||
| embassy-sync = { version = "0.1.0", path = "../embassy-sync" } | ||||
| embassy-sync = { version = "0.2.0", path = "../embassy-sync" } | ||||
| embedded-hal-02 = { package = "embedded-hal", version = "0.2.6", features = ["unproven"] } | ||||
| embedded-hal-1 = { package = "embedded-hal", version = "1.0.0-alpha.8" } | ||||
| embedded-hal-async = { version = "0.1.0-alpha.1", optional = true } | ||||
| embedded-hal-1 = { package = "embedded-hal", version = "=1.0.0-alpha.10" } | ||||
| embedded-hal-async = { version = "=0.2.0-alpha.1", optional = true } | ||||
| embedded-storage = "0.3.0" | ||||
| embedded-storage-async = { version = "0.3.0", optional = true } | ||||
| embedded-storage-async = { version = "0.4.0", optional = true } | ||||
| nb = "1.0.0" | ||||
|  | ||||
| defmt = { version = "0.3", optional = true } | ||||
|   | ||||
| @@ -1,7 +1,5 @@ | ||||
| //! Adapters between embedded-hal traits. | ||||
|  | ||||
| use core::future::Future; | ||||
|  | ||||
| use embedded_hal_02::{blocking, serial}; | ||||
|  | ||||
| /// Wrapper that implements async traits using blocking implementations. | ||||
| @@ -38,32 +36,26 @@ where | ||||
|     E: embedded_hal_1::i2c::Error + 'static, | ||||
|     T: blocking::i2c::WriteRead<Error = E> + blocking::i2c::Read<Error = E> + blocking::i2c::Write<Error = E>, | ||||
| { | ||||
|     type WriteFuture<'a> = impl Future<Output = Result<(), Self::Error>> + 'a where Self: 'a; | ||||
|     type ReadFuture<'a> = impl Future<Output = Result<(), Self::Error>> + 'a where Self: 'a; | ||||
|     type WriteReadFuture<'a> = impl Future<Output = Result<(), Self::Error>> + 'a where Self: 'a; | ||||
|  | ||||
|     fn read<'a>(&'a mut self, address: u8, buffer: &'a mut [u8]) -> Self::ReadFuture<'a> { | ||||
|         async move { self.wrapped.read(address, buffer) } | ||||
|     async fn read(&mut self, address: u8, read: &mut [u8]) -> Result<(), Self::Error> { | ||||
|         self.wrapped.read(address, read) | ||||
|     } | ||||
|  | ||||
|     fn write<'a>(&'a mut self, address: u8, bytes: &'a [u8]) -> Self::WriteFuture<'a> { | ||||
|         async move { self.wrapped.write(address, bytes) } | ||||
|     async fn write(&mut self, address: u8, write: &[u8]) -> Result<(), Self::Error> { | ||||
|         self.wrapped.write(address, write) | ||||
|     } | ||||
|  | ||||
|     fn write_read<'a>(&'a mut self, address: u8, bytes: &'a [u8], buffer: &'a mut [u8]) -> Self::WriteReadFuture<'a> { | ||||
|         async move { self.wrapped.write_read(address, bytes, buffer) } | ||||
|     async fn write_read(&mut self, address: u8, write: &[u8], read: &mut [u8]) -> Result<(), Self::Error> { | ||||
|         self.wrapped.write_read(address, write, read) | ||||
|     } | ||||
|  | ||||
|     type TransactionFuture<'a, 'b> = impl Future<Output = Result<(), Self::Error>> + 'a where Self: 'a, 'b: 'a; | ||||
|  | ||||
|     fn transaction<'a, 'b>( | ||||
|         &'a mut self, | ||||
|     async fn transaction( | ||||
|         &mut self, | ||||
|         address: u8, | ||||
|         operations: &'a mut [embedded_hal_async::i2c::Operation<'b>], | ||||
|     ) -> Self::TransactionFuture<'a, 'b> { | ||||
|         operations: &mut [embedded_hal_1::i2c::Operation<'_>], | ||||
|     ) -> Result<(), Self::Error> { | ||||
|         let _ = address; | ||||
|         let _ = operations; | ||||
|         async move { todo!() } | ||||
|         todo!() | ||||
|     } | ||||
| } | ||||
|  | ||||
| @@ -84,23 +76,17 @@ where | ||||
|     E: embedded_hal_1::spi::Error + 'static, | ||||
|     T: blocking::spi::Transfer<u8, Error = E> + blocking::spi::Write<u8, Error = E>, | ||||
| { | ||||
|     type TransferFuture<'a> = impl Future<Output = Result<(), Self::Error>> + 'a where Self: 'a; | ||||
|  | ||||
|     fn transfer<'a>(&'a mut self, read: &'a mut [u8], write: &'a [u8]) -> Self::TransferFuture<'a> { | ||||
|         async move { | ||||
|             // Ensure we write the expected bytes | ||||
|             for i in 0..core::cmp::min(read.len(), write.len()) { | ||||
|                 read[i] = write[i].clone(); | ||||
|             } | ||||
|             self.wrapped.transfer(read)?; | ||||
|             Ok(()) | ||||
|     async fn transfer<'a>(&'a mut self, read: &'a mut [u8], write: &'a [u8]) -> Result<(), Self::Error> { | ||||
|         // Ensure we write the expected bytes | ||||
|         for i in 0..core::cmp::min(read.len(), write.len()) { | ||||
|             read[i] = write[i].clone(); | ||||
|         } | ||||
|         self.wrapped.transfer(read)?; | ||||
|         Ok(()) | ||||
|     } | ||||
|  | ||||
|     type TransferInPlaceFuture<'a> = impl Future<Output = Result<(), Self::Error>> + 'a where Self: 'a; | ||||
|  | ||||
|     fn transfer_in_place<'a>(&'a mut self, _: &'a mut [u8]) -> Self::TransferInPlaceFuture<'a> { | ||||
|         async move { todo!() } | ||||
|     async fn transfer_in_place<'a>(&'a mut self, _: &'a mut [u8]) -> Result<(), Self::Error> { | ||||
|         todo!() | ||||
|     } | ||||
| } | ||||
|  | ||||
| @@ -109,10 +95,8 @@ where | ||||
|     E: embedded_hal_1::spi::Error + 'static, | ||||
|     T: blocking::spi::Transfer<u8, Error = E> + blocking::spi::Write<u8, Error = E>, | ||||
| { | ||||
|     type FlushFuture<'a> = impl Future<Output = Result<(), Self::Error>> + 'a where Self: 'a; | ||||
|  | ||||
|     fn flush<'a>(&'a mut self) -> Self::FlushFuture<'a> { | ||||
|         async move { Ok(()) } | ||||
|     async fn flush(&mut self) -> Result<(), Self::Error> { | ||||
|         Ok(()) | ||||
|     } | ||||
| } | ||||
|  | ||||
| @@ -121,13 +105,9 @@ where | ||||
|     E: embedded_hal_1::spi::Error + 'static, | ||||
|     T: blocking::spi::Transfer<u8, Error = E> + blocking::spi::Write<u8, Error = E>, | ||||
| { | ||||
|     type WriteFuture<'a> = impl Future<Output = Result<(), Self::Error>> + 'a where Self: 'a; | ||||
|  | ||||
|     fn write<'a>(&'a mut self, data: &'a [u8]) -> Self::WriteFuture<'a> { | ||||
|         async move { | ||||
|             self.wrapped.write(data)?; | ||||
|             Ok(()) | ||||
|         } | ||||
|     async fn write(&mut self, data: &[u8]) -> Result<(), Self::Error> { | ||||
|         self.wrapped.write(data)?; | ||||
|         Ok(()) | ||||
|     } | ||||
| } | ||||
|  | ||||
| @@ -136,13 +116,9 @@ where | ||||
|     E: embedded_hal_1::spi::Error + 'static, | ||||
|     T: blocking::spi::Transfer<u8, Error = E> + blocking::spi::Write<u8, Error = E>, | ||||
| { | ||||
|     type ReadFuture<'a> = impl Future<Output = Result<(), Self::Error>> + 'a where Self: 'a; | ||||
|  | ||||
|     fn read<'a>(&'a mut self, data: &'a mut [u8]) -> Self::ReadFuture<'a> { | ||||
|         async move { | ||||
|             self.wrapped.transfer(data)?; | ||||
|             Ok(()) | ||||
|         } | ||||
|     async fn read(&mut self, data: &mut [u8]) -> Result<(), Self::Error> { | ||||
|         self.wrapped.transfer(data)?; | ||||
|         Ok(()) | ||||
|     } | ||||
| } | ||||
|  | ||||
| @@ -155,51 +131,9 @@ where | ||||
|     type Error = E; | ||||
| } | ||||
|  | ||||
| #[cfg(feature = "_todo_embedded_hal_serial")] | ||||
| impl<T, E> embedded_hal_async::serial::Read for BlockingAsync<T> | ||||
| where | ||||
|     T: serial::Read<u8, Error = E>, | ||||
|     E: embedded_hal_1::serial::Error + 'static, | ||||
| { | ||||
|     type ReadFuture<'a> = impl Future<Output = Result<(), Self::Error>> + 'a where T: 'a; | ||||
|     fn read<'a>(&'a mut self, buf: &'a mut [u8]) -> Self::ReadFuture<'a> { | ||||
|         async move { | ||||
|             let mut pos = 0; | ||||
|             while pos < buf.len() { | ||||
|                 match self.wrapped.read() { | ||||
|                     Err(nb::Error::WouldBlock) => {} | ||||
|                     Err(nb::Error::Other(e)) => return Err(e), | ||||
|                     Ok(b) => { | ||||
|                         buf[pos] = b; | ||||
|                         pos += 1; | ||||
|                     } | ||||
|                 } | ||||
|             } | ||||
|             Ok(()) | ||||
|         } | ||||
|     } | ||||
| } | ||||
|  | ||||
| #[cfg(feature = "_todo_embedded_hal_serial")] | ||||
| impl<T, E> embedded_hal_async::serial::Write for BlockingAsync<T> | ||||
| where | ||||
|     T: blocking::serial::Write<u8, Error = E> + serial::Read<u8, Error = E>, | ||||
|     E: embedded_hal_1::serial::Error + 'static, | ||||
| { | ||||
|     type WriteFuture<'a> = impl Future<Output = Result<(), Self::Error>> + 'a where T: 'a; | ||||
|     fn write<'a>(&'a mut self, buf: &'a [u8]) -> Self::WriteFuture<'a> { | ||||
|         async move { self.wrapped.bwrite_all(buf) } | ||||
|     } | ||||
|  | ||||
|     type FlushFuture<'a> = impl Future<Output = Result<(), Self::Error>> + 'a where T: 'a; | ||||
|     fn flush<'a>(&'a mut self) -> Self::FlushFuture<'a> { | ||||
|         async move { self.wrapped.bflush() } | ||||
|     } | ||||
| } | ||||
|  | ||||
| /// NOR flash wrapper | ||||
| use embedded_storage::nor_flash::{ErrorType, NorFlash, ReadNorFlash}; | ||||
| use embedded_storage_async::nor_flash::{AsyncNorFlash, AsyncReadNorFlash}; | ||||
| use embedded_storage_async::nor_flash::{NorFlash as AsyncNorFlash, ReadNorFlash as AsyncReadNorFlash}; | ||||
|  | ||||
| impl<T> ErrorType for BlockingAsync<T> | ||||
| where | ||||
| @@ -215,14 +149,12 @@ where | ||||
|     const WRITE_SIZE: usize = <T as NorFlash>::WRITE_SIZE; | ||||
|     const ERASE_SIZE: usize = <T as NorFlash>::ERASE_SIZE; | ||||
|  | ||||
|     type WriteFuture<'a> = impl Future<Output = Result<(), Self::Error>> + 'a where Self: 'a; | ||||
|     fn write<'a>(&'a mut self, offset: u32, data: &'a [u8]) -> Self::WriteFuture<'a> { | ||||
|         async move { self.wrapped.write(offset, data) } | ||||
|     async fn write(&mut self, offset: u32, data: &[u8]) -> Result<(), Self::Error> { | ||||
|         self.wrapped.write(offset, data) | ||||
|     } | ||||
|  | ||||
|     type EraseFuture<'a> = impl Future<Output = Result<(), Self::Error>> + 'a where Self: 'a; | ||||
|     fn erase<'a>(&'a mut self, from: u32, to: u32) -> Self::EraseFuture<'a> { | ||||
|         async move { self.wrapped.erase(from, to) } | ||||
|     async fn erase(&mut self, from: u32, to: u32) -> Result<(), Self::Error> { | ||||
|         self.wrapped.erase(from, to) | ||||
|     } | ||||
| } | ||||
|  | ||||
| @@ -231,9 +163,8 @@ where | ||||
|     T: ReadNorFlash, | ||||
| { | ||||
|     const READ_SIZE: usize = <T as ReadNorFlash>::READ_SIZE; | ||||
|     type ReadFuture<'a> = impl Future<Output = Result<(), Self::Error>> + 'a where Self: 'a; | ||||
|     fn read<'a>(&'a mut self, address: u32, data: &'a mut [u8]) -> Self::ReadFuture<'a> { | ||||
|         async move { self.wrapped.read(address, data) } | ||||
|     async fn read(&mut self, address: u32, data: &mut [u8]) -> Result<(), Self::Error> { | ||||
|         self.wrapped.read(address, data) | ||||
|     } | ||||
|  | ||||
|     fn capacity(&self) -> usize { | ||||
|   | ||||
| @@ -1,5 +1,9 @@ | ||||
| #![cfg_attr(not(feature = "std"), no_std)] | ||||
| #![cfg_attr(feature = "nightly", feature(generic_associated_types, type_alias_impl_trait))] | ||||
| #![cfg_attr( | ||||
|     feature = "nightly", | ||||
|     feature(type_alias_impl_trait, async_fn_in_trait, impl_trait_projections, try_blocks) | ||||
| )] | ||||
| #![cfg_attr(feature = "nightly", allow(incomplete_features))] | ||||
| #![warn(missing_docs)] | ||||
|  | ||||
| //! Utilities to use `embedded-hal` traits with Embassy. | ||||
|   | ||||
| @@ -22,7 +22,6 @@ | ||||
| //! let i2c_dev2 = I2cDevice::new(i2c_bus); | ||||
| //! let mpu = Mpu6050::new(i2c_dev2); | ||||
| //! ``` | ||||
| use core::future::Future; | ||||
|  | ||||
| use embassy_sync::blocking_mutex::raw::RawMutex; | ||||
| use embassy_sync::mutex::Mutex; | ||||
| @@ -55,53 +54,39 @@ where | ||||
|     M: RawMutex + 'static, | ||||
|     BUS: i2c::I2c + 'static, | ||||
| { | ||||
|     type ReadFuture<'a> = impl Future<Output = Result<(), Self::Error>> + 'a where Self: 'a; | ||||
|  | ||||
|     fn read<'a>(&'a mut self, address: u8, buffer: &'a mut [u8]) -> Self::ReadFuture<'a> { | ||||
|         async move { | ||||
|             let mut bus = self.bus.lock().await; | ||||
|             bus.read(address, buffer).await.map_err(I2cDeviceError::I2c)?; | ||||
|             Ok(()) | ||||
|         } | ||||
|     async fn read(&mut self, address: u8, read: &mut [u8]) -> Result<(), I2cDeviceError<BUS::Error>> { | ||||
|         let mut bus = self.bus.lock().await; | ||||
|         bus.read(address, read).await.map_err(I2cDeviceError::I2c)?; | ||||
|         Ok(()) | ||||
|     } | ||||
|  | ||||
|     type WriteFuture<'a> = impl Future<Output = Result<(), Self::Error>> + 'a where Self: 'a; | ||||
|  | ||||
|     fn write<'a>(&'a mut self, address: u8, bytes: &'a [u8]) -> Self::WriteFuture<'a> { | ||||
|         async move { | ||||
|             let mut bus = self.bus.lock().await; | ||||
|             bus.write(address, bytes).await.map_err(I2cDeviceError::I2c)?; | ||||
|             Ok(()) | ||||
|         } | ||||
|     async fn write(&mut self, address: u8, write: &[u8]) -> Result<(), I2cDeviceError<BUS::Error>> { | ||||
|         let mut bus = self.bus.lock().await; | ||||
|         bus.write(address, write).await.map_err(I2cDeviceError::I2c)?; | ||||
|         Ok(()) | ||||
|     } | ||||
|  | ||||
|     type WriteReadFuture<'a> = impl Future<Output = Result<(), Self::Error>> + 'a where Self: 'a; | ||||
|  | ||||
|     fn write_read<'a>( | ||||
|         &'a mut self, | ||||
|     async fn write_read( | ||||
|         &mut self, | ||||
|         address: u8, | ||||
|         wr_buffer: &'a [u8], | ||||
|         rd_buffer: &'a mut [u8], | ||||
|     ) -> Self::WriteReadFuture<'a> { | ||||
|         async move { | ||||
|             let mut bus = self.bus.lock().await; | ||||
|             bus.write_read(address, wr_buffer, rd_buffer) | ||||
|                 .await | ||||
|                 .map_err(I2cDeviceError::I2c)?; | ||||
|             Ok(()) | ||||
|         } | ||||
|         write: &[u8], | ||||
|         read: &mut [u8], | ||||
|     ) -> Result<(), I2cDeviceError<BUS::Error>> { | ||||
|         let mut bus = self.bus.lock().await; | ||||
|         bus.write_read(address, write, read) | ||||
|             .await | ||||
|             .map_err(I2cDeviceError::I2c)?; | ||||
|         Ok(()) | ||||
|     } | ||||
|  | ||||
|     type TransactionFuture<'a, 'b> = impl Future<Output = Result<(), Self::Error>> + 'a where Self: 'a, 'b: 'a; | ||||
|  | ||||
|     fn transaction<'a, 'b>( | ||||
|         &'a mut self, | ||||
|     async fn transaction( | ||||
|         &mut self, | ||||
|         address: u8, | ||||
|         operations: &'a mut [embedded_hal_async::i2c::Operation<'b>], | ||||
|     ) -> Self::TransactionFuture<'a, 'b> { | ||||
|         operations: &mut [embedded_hal_async::i2c::Operation<'_>], | ||||
|     ) -> Result<(), I2cDeviceError<BUS::Error>> { | ||||
|         let _ = address; | ||||
|         let _ = operations; | ||||
|         async move { todo!() } | ||||
|         todo!() | ||||
|     } | ||||
| } | ||||
|  | ||||
| @@ -136,55 +121,37 @@ where | ||||
|     M: RawMutex + 'static, | ||||
|     BUS: i2c::I2c + SetConfig + 'static, | ||||
| { | ||||
|     type ReadFuture<'a> = impl Future<Output = Result<(), Self::Error>> + 'a where Self: 'a; | ||||
|  | ||||
|     fn read<'a>(&'a mut self, address: u8, buffer: &'a mut [u8]) -> Self::ReadFuture<'a> { | ||||
|         async move { | ||||
|             let mut bus = self.bus.lock().await; | ||||
|             bus.set_config(&self.config); | ||||
|             bus.read(address, buffer).await.map_err(I2cDeviceError::I2c)?; | ||||
|             Ok(()) | ||||
|         } | ||||
|     async fn read(&mut self, address: u8, buffer: &mut [u8]) -> Result<(), I2cDeviceError<BUS::Error>> { | ||||
|         let mut bus = self.bus.lock().await; | ||||
|         bus.set_config(&self.config); | ||||
|         bus.read(address, buffer).await.map_err(I2cDeviceError::I2c)?; | ||||
|         Ok(()) | ||||
|     } | ||||
|  | ||||
|     type WriteFuture<'a> = impl Future<Output = Result<(), Self::Error>> + 'a where Self: 'a; | ||||
|  | ||||
|     fn write<'a>(&'a mut self, address: u8, bytes: &'a [u8]) -> Self::WriteFuture<'a> { | ||||
|         async move { | ||||
|             let mut bus = self.bus.lock().await; | ||||
|             bus.set_config(&self.config); | ||||
|             bus.write(address, bytes).await.map_err(I2cDeviceError::I2c)?; | ||||
|             Ok(()) | ||||
|         } | ||||
|     async fn write(&mut self, address: u8, bytes: &[u8]) -> Result<(), I2cDeviceError<BUS::Error>> { | ||||
|         let mut bus = self.bus.lock().await; | ||||
|         bus.set_config(&self.config); | ||||
|         bus.write(address, bytes).await.map_err(I2cDeviceError::I2c)?; | ||||
|         Ok(()) | ||||
|     } | ||||
|  | ||||
|     type WriteReadFuture<'a> = impl Future<Output = Result<(), Self::Error>> + 'a where Self: 'a; | ||||
|  | ||||
|     fn write_read<'a>( | ||||
|         &'a mut self, | ||||
|     async fn write_read( | ||||
|         &mut self, | ||||
|         address: u8, | ||||
|         wr_buffer: &'a [u8], | ||||
|         rd_buffer: &'a mut [u8], | ||||
|     ) -> Self::WriteReadFuture<'a> { | ||||
|         async move { | ||||
|             let mut bus = self.bus.lock().await; | ||||
|             bus.set_config(&self.config); | ||||
|             bus.write_read(address, wr_buffer, rd_buffer) | ||||
|                 .await | ||||
|                 .map_err(I2cDeviceError::I2c)?; | ||||
|             Ok(()) | ||||
|         } | ||||
|         wr_buffer: &[u8], | ||||
|         rd_buffer: &mut [u8], | ||||
|     ) -> Result<(), I2cDeviceError<BUS::Error>> { | ||||
|         let mut bus = self.bus.lock().await; | ||||
|         bus.set_config(&self.config); | ||||
|         bus.write_read(address, wr_buffer, rd_buffer) | ||||
|             .await | ||||
|             .map_err(I2cDeviceError::I2c)?; | ||||
|         Ok(()) | ||||
|     } | ||||
|  | ||||
|     type TransactionFuture<'a, 'b> = impl Future<Output = Result<(), Self::Error>> + 'a where Self: 'a, 'b: 'a; | ||||
|  | ||||
|     fn transaction<'a, 'b>( | ||||
|         &'a mut self, | ||||
|         address: u8, | ||||
|         operations: &'a mut [embedded_hal_async::i2c::Operation<'b>], | ||||
|     ) -> Self::TransactionFuture<'a, 'b> { | ||||
|     async fn transaction(&mut self, address: u8, operations: &mut [i2c::Operation<'_>]) -> Result<(), Self::Error> { | ||||
|         let _ = address; | ||||
|         let _ = operations; | ||||
|         async move { todo!() } | ||||
|         todo!() | ||||
|     } | ||||
| } | ||||
|   | ||||
| @@ -25,12 +25,11 @@ | ||||
| //! let spi_dev2 = SpiDevice::new(spi_bus, cs_pin2); | ||||
| //! let display2 = ST7735::new(spi_dev2, dc2, rst2, Default::default(), 160, 128); | ||||
| //! ``` | ||||
| use core::future::Future; | ||||
|  | ||||
| use embassy_sync::blocking_mutex::raw::RawMutex; | ||||
| use embassy_sync::mutex::Mutex; | ||||
| use embedded_hal_1::digital::blocking::OutputPin; | ||||
| use embedded_hal_1::spi::ErrorType; | ||||
| use embedded_hal_1::digital::OutputPin; | ||||
| use embedded_hal_1::spi::Operation; | ||||
| use embedded_hal_async::spi; | ||||
|  | ||||
| use crate::shared_bus::SpiDeviceError; | ||||
| @@ -57,41 +56,92 @@ where | ||||
|     type Error = SpiDeviceError<BUS::Error, CS::Error>; | ||||
| } | ||||
|  | ||||
| impl<M, BUS, CS> spi::SpiDevice for SpiDevice<'_, M, BUS, CS> | ||||
| impl<M, BUS, CS> spi::SpiDeviceRead for SpiDevice<'_, M, BUS, CS> | ||||
| where | ||||
|     M: RawMutex + 'static, | ||||
|     BUS: spi::SpiBusFlush + 'static, | ||||
|     M: RawMutex, | ||||
|     BUS: spi::SpiBusRead, | ||||
|     CS: OutputPin, | ||||
| { | ||||
|     type Bus = BUS; | ||||
|     async fn read_transaction(&mut self, operations: &mut [&mut [u8]]) -> Result<(), Self::Error> { | ||||
|         let mut bus = self.bus.lock().await; | ||||
|         self.cs.set_low().map_err(SpiDeviceError::Cs)?; | ||||
|  | ||||
|     type TransactionFuture<'a, R, F, Fut> = impl Future<Output = Result<R, Self::Error>> + 'a | ||||
|     where | ||||
|         Self: 'a, R: 'a, F: FnOnce(*mut Self::Bus) -> Fut + 'a, | ||||
|         Fut: Future<Output =  Result<R, <Self::Bus as ErrorType>::Error>> + 'a; | ||||
|         let op_res: Result<(), BUS::Error> = try { | ||||
|             for buf in operations { | ||||
|                 bus.read(buf).await?; | ||||
|             } | ||||
|         }; | ||||
|  | ||||
|     fn transaction<'a, R, F, Fut>(&'a mut self, f: F) -> Self::TransactionFuture<'a, R, F, Fut> | ||||
|     where | ||||
|         R: 'a, | ||||
|         F: FnOnce(*mut Self::Bus) -> Fut + 'a, | ||||
|         Fut: Future<Output = Result<R, <Self::Bus as ErrorType>::Error>> + 'a, | ||||
|     { | ||||
|         async move { | ||||
|             let mut bus = self.bus.lock().await; | ||||
|             self.cs.set_low().map_err(SpiDeviceError::Cs)?; | ||||
|         // On failure, it's important to still flush and deassert CS. | ||||
|         let flush_res = bus.flush().await; | ||||
|         let cs_res = self.cs.set_high(); | ||||
|  | ||||
|             let f_res = f(&mut *bus).await; | ||||
|         let op_res = op_res.map_err(SpiDeviceError::Spi)?; | ||||
|         flush_res.map_err(SpiDeviceError::Spi)?; | ||||
|         cs_res.map_err(SpiDeviceError::Cs)?; | ||||
|  | ||||
|             // On failure, it's important to still flush and deassert CS. | ||||
|             let flush_res = bus.flush().await; | ||||
|             let cs_res = self.cs.set_high(); | ||||
|         Ok(op_res) | ||||
|     } | ||||
| } | ||||
|  | ||||
|             let f_res = f_res.map_err(SpiDeviceError::Spi)?; | ||||
|             flush_res.map_err(SpiDeviceError::Spi)?; | ||||
|             cs_res.map_err(SpiDeviceError::Cs)?; | ||||
| impl<M, BUS, CS> spi::SpiDeviceWrite for SpiDevice<'_, M, BUS, CS> | ||||
| where | ||||
|     M: RawMutex, | ||||
|     BUS: spi::SpiBusWrite, | ||||
|     CS: OutputPin, | ||||
| { | ||||
|     async fn write_transaction(&mut self, operations: &[&[u8]]) -> Result<(), Self::Error> { | ||||
|         let mut bus = self.bus.lock().await; | ||||
|         self.cs.set_low().map_err(SpiDeviceError::Cs)?; | ||||
|  | ||||
|             Ok(f_res) | ||||
|         } | ||||
|         let op_res: Result<(), BUS::Error> = try { | ||||
|             for buf in operations { | ||||
|                 bus.write(buf).await?; | ||||
|             } | ||||
|         }; | ||||
|  | ||||
|         // On failure, it's important to still flush and deassert CS. | ||||
|         let flush_res = bus.flush().await; | ||||
|         let cs_res = self.cs.set_high(); | ||||
|  | ||||
|         let op_res = op_res.map_err(SpiDeviceError::Spi)?; | ||||
|         flush_res.map_err(SpiDeviceError::Spi)?; | ||||
|         cs_res.map_err(SpiDeviceError::Cs)?; | ||||
|  | ||||
|         Ok(op_res) | ||||
|     } | ||||
| } | ||||
|  | ||||
| impl<M, BUS, CS> spi::SpiDevice for SpiDevice<'_, M, BUS, CS> | ||||
| where | ||||
|     M: RawMutex, | ||||
|     BUS: spi::SpiBus, | ||||
|     CS: OutputPin, | ||||
| { | ||||
|     async fn transaction(&mut self, operations: &mut [spi::Operation<'_, u8>]) -> Result<(), Self::Error> { | ||||
|         let mut bus = self.bus.lock().await; | ||||
|         self.cs.set_low().map_err(SpiDeviceError::Cs)?; | ||||
|  | ||||
|         let op_res: Result<(), BUS::Error> = try { | ||||
|             for op in operations { | ||||
|                 match op { | ||||
|                     Operation::Read(buf) => bus.read(buf).await?, | ||||
|                     Operation::Write(buf) => bus.write(buf).await?, | ||||
|                     Operation::Transfer(read, write) => bus.transfer(read, write).await?, | ||||
|                     Operation::TransferInPlace(buf) => bus.transfer_in_place(buf).await?, | ||||
|                 } | ||||
|             } | ||||
|         }; | ||||
|  | ||||
|         // On failure, it's important to still flush and deassert CS. | ||||
|         let flush_res = bus.flush().await; | ||||
|         let cs_res = self.cs.set_high(); | ||||
|  | ||||
|         let op_res = op_res.map_err(SpiDeviceError::Spi)?; | ||||
|         flush_res.map_err(SpiDeviceError::Spi)?; | ||||
|         cs_res.map_err(SpiDeviceError::Cs)?; | ||||
|  | ||||
|         Ok(op_res) | ||||
|     } | ||||
| } | ||||
|  | ||||
| @@ -122,41 +172,94 @@ where | ||||
|     type Error = SpiDeviceError<BUS::Error, CS::Error>; | ||||
| } | ||||
|  | ||||
| impl<M, BUS, CS> spi::SpiDevice for SpiDeviceWithConfig<'_, M, BUS, CS> | ||||
| impl<M, BUS, CS> spi::SpiDeviceWrite for SpiDeviceWithConfig<'_, M, BUS, CS> | ||||
| where | ||||
|     M: RawMutex + 'static, | ||||
|     BUS: spi::SpiBusFlush + SetConfig + 'static, | ||||
|     M: RawMutex, | ||||
|     BUS: spi::SpiBusWrite + SetConfig, | ||||
|     CS: OutputPin, | ||||
| { | ||||
|     type Bus = BUS; | ||||
|     async fn write_transaction(&mut self, operations: &[&[u8]]) -> Result<(), Self::Error> { | ||||
|         let mut bus = self.bus.lock().await; | ||||
|         bus.set_config(&self.config); | ||||
|         self.cs.set_low().map_err(SpiDeviceError::Cs)?; | ||||
|  | ||||
|     type TransactionFuture<'a, R, F, Fut> = impl Future<Output = Result<R, Self::Error>> + 'a | ||||
|     where | ||||
|         Self: 'a, R: 'a, F: FnOnce(*mut Self::Bus) -> Fut + 'a, | ||||
|         Fut: Future<Output =  Result<R, <Self::Bus as ErrorType>::Error>> + 'a; | ||||
|         let op_res: Result<(), BUS::Error> = try { | ||||
|             for buf in operations { | ||||
|                 bus.write(buf).await?; | ||||
|             } | ||||
|         }; | ||||
|  | ||||
|     fn transaction<'a, R, F, Fut>(&'a mut self, f: F) -> Self::TransactionFuture<'a, R, F, Fut> | ||||
|     where | ||||
|         R: 'a, | ||||
|         F: FnOnce(*mut Self::Bus) -> Fut + 'a, | ||||
|         Fut: Future<Output = Result<R, <Self::Bus as ErrorType>::Error>> + 'a, | ||||
|     { | ||||
|         async move { | ||||
|             let mut bus = self.bus.lock().await; | ||||
|             bus.set_config(&self.config); | ||||
|             self.cs.set_low().map_err(SpiDeviceError::Cs)?; | ||||
|         // On failure, it's important to still flush and deassert CS. | ||||
|         let flush_res = bus.flush().await; | ||||
|         let cs_res = self.cs.set_high(); | ||||
|  | ||||
|             let f_res = f(&mut *bus).await; | ||||
|         let op_res = op_res.map_err(SpiDeviceError::Spi)?; | ||||
|         flush_res.map_err(SpiDeviceError::Spi)?; | ||||
|         cs_res.map_err(SpiDeviceError::Cs)?; | ||||
|  | ||||
|             // On failure, it's important to still flush and deassert CS. | ||||
|             let flush_res = bus.flush().await; | ||||
|             let cs_res = self.cs.set_high(); | ||||
|  | ||||
|             let f_res = f_res.map_err(SpiDeviceError::Spi)?; | ||||
|             flush_res.map_err(SpiDeviceError::Spi)?; | ||||
|             cs_res.map_err(SpiDeviceError::Cs)?; | ||||
|  | ||||
|             Ok(f_res) | ||||
|         } | ||||
|         Ok(op_res) | ||||
|     } | ||||
| } | ||||
|  | ||||
| impl<M, BUS, CS> spi::SpiDeviceRead for SpiDeviceWithConfig<'_, M, BUS, CS> | ||||
| where | ||||
|     M: RawMutex, | ||||
|     BUS: spi::SpiBusRead + SetConfig, | ||||
|     CS: OutputPin, | ||||
| { | ||||
|     async fn read_transaction(&mut self, operations: &mut [&mut [u8]]) -> Result<(), Self::Error> { | ||||
|         let mut bus = self.bus.lock().await; | ||||
|         bus.set_config(&self.config); | ||||
|         self.cs.set_low().map_err(SpiDeviceError::Cs)?; | ||||
|  | ||||
|         let op_res: Result<(), BUS::Error> = try { | ||||
|             for buf in operations { | ||||
|                 bus.read(buf).await?; | ||||
|             } | ||||
|         }; | ||||
|  | ||||
|         // On failure, it's important to still flush and deassert CS. | ||||
|         let flush_res = bus.flush().await; | ||||
|         let cs_res = self.cs.set_high(); | ||||
|  | ||||
|         let op_res = op_res.map_err(SpiDeviceError::Spi)?; | ||||
|         flush_res.map_err(SpiDeviceError::Spi)?; | ||||
|         cs_res.map_err(SpiDeviceError::Cs)?; | ||||
|  | ||||
|         Ok(op_res) | ||||
|     } | ||||
| } | ||||
|  | ||||
| impl<M, BUS, CS> spi::SpiDevice for SpiDeviceWithConfig<'_, M, BUS, CS> | ||||
| where | ||||
|     M: RawMutex, | ||||
|     BUS: spi::SpiBus + SetConfig, | ||||
|     CS: OutputPin, | ||||
| { | ||||
|     async fn transaction(&mut self, operations: &mut [spi::Operation<'_, u8>]) -> Result<(), Self::Error> { | ||||
|         let mut bus = self.bus.lock().await; | ||||
|         bus.set_config(&self.config); | ||||
|         self.cs.set_low().map_err(SpiDeviceError::Cs)?; | ||||
|  | ||||
|         let op_res: Result<(), BUS::Error> = try { | ||||
|             for op in operations { | ||||
|                 match op { | ||||
|                     Operation::Read(buf) => bus.read(buf).await?, | ||||
|                     Operation::Write(buf) => bus.write(buf).await?, | ||||
|                     Operation::Transfer(read, write) => bus.transfer(read, write).await?, | ||||
|                     Operation::TransferInPlace(buf) => bus.transfer_in_place(buf).await?, | ||||
|                 } | ||||
|             } | ||||
|         }; | ||||
|  | ||||
|         // On failure, it's important to still flush and deassert CS. | ||||
|         let flush_res = bus.flush().await; | ||||
|         let cs_res = self.cs.set_high(); | ||||
|  | ||||
|         let op_res = op_res.map_err(SpiDeviceError::Spi)?; | ||||
|         flush_res.map_err(SpiDeviceError::Spi)?; | ||||
|         cs_res.map_err(SpiDeviceError::Cs)?; | ||||
|  | ||||
|         Ok(op_res) | ||||
|     } | ||||
| } | ||||
|   | ||||
| @@ -20,8 +20,7 @@ use core::cell::RefCell; | ||||
|  | ||||
| use embassy_sync::blocking_mutex::raw::RawMutex; | ||||
| use embassy_sync::blocking_mutex::Mutex; | ||||
| use embedded_hal_1::i2c::blocking::{I2c, Operation}; | ||||
| use embedded_hal_1::i2c::ErrorType; | ||||
| use embedded_hal_1::i2c::{ErrorType, I2c, Operation}; | ||||
|  | ||||
| use crate::shared_bus::I2cDeviceError; | ||||
| use crate::SetConfig; | ||||
| @@ -73,34 +72,6 @@ where | ||||
|         let _ = operations; | ||||
|         todo!() | ||||
|     } | ||||
|  | ||||
|     fn write_iter<B: IntoIterator<Item = u8>>(&mut self, addr: u8, bytes: B) -> Result<(), Self::Error> { | ||||
|         let _ = addr; | ||||
|         let _ = bytes; | ||||
|         todo!() | ||||
|     } | ||||
|  | ||||
|     fn write_iter_read<B: IntoIterator<Item = u8>>( | ||||
|         &mut self, | ||||
|         addr: u8, | ||||
|         bytes: B, | ||||
|         buffer: &mut [u8], | ||||
|     ) -> Result<(), Self::Error> { | ||||
|         let _ = addr; | ||||
|         let _ = bytes; | ||||
|         let _ = buffer; | ||||
|         todo!() | ||||
|     } | ||||
|  | ||||
|     fn transaction_iter<'a, O: IntoIterator<Item = Operation<'a>>>( | ||||
|         &mut self, | ||||
|         address: u8, | ||||
|         operations: O, | ||||
|     ) -> Result<(), Self::Error> { | ||||
|         let _ = address; | ||||
|         let _ = operations; | ||||
|         todo!() | ||||
|     } | ||||
| } | ||||
|  | ||||
| impl<'a, M, BUS, E> embedded_hal_02::blocking::i2c::Write for I2cDevice<'_, M, BUS> | ||||
| @@ -205,32 +176,4 @@ where | ||||
|         let _ = operations; | ||||
|         todo!() | ||||
|     } | ||||
|  | ||||
|     fn write_iter<B: IntoIterator<Item = u8>>(&mut self, addr: u8, bytes: B) -> Result<(), Self::Error> { | ||||
|         let _ = addr; | ||||
|         let _ = bytes; | ||||
|         todo!() | ||||
|     } | ||||
|  | ||||
|     fn write_iter_read<B: IntoIterator<Item = u8>>( | ||||
|         &mut self, | ||||
|         addr: u8, | ||||
|         bytes: B, | ||||
|         buffer: &mut [u8], | ||||
|     ) -> Result<(), Self::Error> { | ||||
|         let _ = addr; | ||||
|         let _ = bytes; | ||||
|         let _ = buffer; | ||||
|         todo!() | ||||
|     } | ||||
|  | ||||
|     fn transaction_iter<'a, O: IntoIterator<Item = Operation<'a>>>( | ||||
|         &mut self, | ||||
|         address: u8, | ||||
|         operations: O, | ||||
|     ) -> Result<(), Self::Error> { | ||||
|         let _ = address; | ||||
|         let _ = operations; | ||||
|         todo!() | ||||
|     } | ||||
| } | ||||
|   | ||||
| @@ -22,9 +22,8 @@ use core::cell::RefCell; | ||||
|  | ||||
| use embassy_sync::blocking_mutex::raw::RawMutex; | ||||
| use embassy_sync::blocking_mutex::Mutex; | ||||
| use embedded_hal_1::digital::blocking::OutputPin; | ||||
| use embedded_hal_1::spi; | ||||
| use embedded_hal_1::spi::blocking::SpiBusFlush; | ||||
| use embedded_hal_1::digital::OutputPin; | ||||
| use embedded_hal_1::spi::{self, Operation, SpiBus, SpiBusRead, SpiBusWrite}; | ||||
|  | ||||
| use crate::shared_bus::SpiDeviceError; | ||||
| use crate::SetConfig; | ||||
| @@ -50,30 +49,85 @@ where | ||||
|     type Error = SpiDeviceError<BUS::Error, CS::Error>; | ||||
| } | ||||
|  | ||||
| impl<BUS, M, CS> embedded_hal_1::spi::blocking::SpiDevice for SpiDevice<'_, M, BUS, CS> | ||||
| impl<BUS, M, CS> embedded_hal_1::spi::SpiDeviceRead for SpiDevice<'_, M, BUS, CS> | ||||
| where | ||||
|     M: RawMutex, | ||||
|     BUS: SpiBusFlush, | ||||
|     BUS: SpiBusRead, | ||||
|     CS: OutputPin, | ||||
| { | ||||
|     type Bus = BUS; | ||||
|  | ||||
|     fn transaction<R>(&mut self, f: impl FnOnce(&mut Self::Bus) -> Result<R, BUS::Error>) -> Result<R, Self::Error> { | ||||
|     fn read_transaction(&mut self, operations: &mut [&mut [u8]]) -> Result<(), Self::Error> { | ||||
|         self.bus.lock(|bus| { | ||||
|             let mut bus = bus.borrow_mut(); | ||||
|             self.cs.set_low().map_err(SpiDeviceError::Cs)?; | ||||
|  | ||||
|             let f_res = f(&mut bus); | ||||
|             let op_res = operations.iter_mut().try_for_each(|buf| bus.read(buf)); | ||||
|  | ||||
|             // On failure, it's important to still flush and deassert CS. | ||||
|             let flush_res = bus.flush(); | ||||
|             let cs_res = self.cs.set_high(); | ||||
|  | ||||
|             let f_res = f_res.map_err(SpiDeviceError::Spi)?; | ||||
|             let op_res = op_res.map_err(SpiDeviceError::Spi)?; | ||||
|             flush_res.map_err(SpiDeviceError::Spi)?; | ||||
|             cs_res.map_err(SpiDeviceError::Cs)?; | ||||
|  | ||||
|             Ok(f_res) | ||||
|             Ok(op_res) | ||||
|         }) | ||||
|     } | ||||
| } | ||||
|  | ||||
| impl<BUS, M, CS> embedded_hal_1::spi::SpiDeviceWrite for SpiDevice<'_, M, BUS, CS> | ||||
| where | ||||
|     M: RawMutex, | ||||
|     BUS: SpiBusWrite, | ||||
|     CS: OutputPin, | ||||
| { | ||||
|     fn write_transaction(&mut self, operations: &[&[u8]]) -> Result<(), Self::Error> { | ||||
|         self.bus.lock(|bus| { | ||||
|             let mut bus = bus.borrow_mut(); | ||||
|             self.cs.set_low().map_err(SpiDeviceError::Cs)?; | ||||
|  | ||||
|             let op_res = operations.iter().try_for_each(|buf| bus.write(buf)); | ||||
|  | ||||
|             // On failure, it's important to still flush and deassert CS. | ||||
|             let flush_res = bus.flush(); | ||||
|             let cs_res = self.cs.set_high(); | ||||
|  | ||||
|             let op_res = op_res.map_err(SpiDeviceError::Spi)?; | ||||
|             flush_res.map_err(SpiDeviceError::Spi)?; | ||||
|             cs_res.map_err(SpiDeviceError::Cs)?; | ||||
|  | ||||
|             Ok(op_res) | ||||
|         }) | ||||
|     } | ||||
| } | ||||
|  | ||||
| impl<BUS, M, CS> embedded_hal_1::spi::SpiDevice for SpiDevice<'_, M, BUS, CS> | ||||
| where | ||||
|     M: RawMutex, | ||||
|     BUS: SpiBus, | ||||
|     CS: OutputPin, | ||||
| { | ||||
|     fn transaction(&mut self, operations: &mut [Operation<'_, u8>]) -> Result<(), Self::Error> { | ||||
|         self.bus.lock(|bus| { | ||||
|             let mut bus = bus.borrow_mut(); | ||||
|             self.cs.set_low().map_err(SpiDeviceError::Cs)?; | ||||
|  | ||||
|             let op_res = operations.iter_mut().try_for_each(|op| match op { | ||||
|                 Operation::Read(buf) => bus.read(buf), | ||||
|                 Operation::Write(buf) => bus.write(buf), | ||||
|                 Operation::Transfer(read, write) => bus.transfer(read, write), | ||||
|                 Operation::TransferInPlace(buf) => bus.transfer_in_place(buf), | ||||
|             }); | ||||
|  | ||||
|             // On failure, it's important to still flush and deassert CS. | ||||
|             let flush_res = bus.flush(); | ||||
|             let cs_res = self.cs.set_high(); | ||||
|  | ||||
|             let op_res = op_res.map_err(SpiDeviceError::Spi)?; | ||||
|             flush_res.map_err(SpiDeviceError::Spi)?; | ||||
|             cs_res.map_err(SpiDeviceError::Cs)?; | ||||
|  | ||||
|             Ok(op_res) | ||||
|         }) | ||||
|     } | ||||
| } | ||||
| @@ -89,11 +143,11 @@ where | ||||
|         self.bus.lock(|bus| { | ||||
|             let mut bus = bus.borrow_mut(); | ||||
|             self.cs.set_low().map_err(SpiDeviceError::Cs)?; | ||||
|             let f_res = bus.transfer(words); | ||||
|             let op_res = bus.transfer(words); | ||||
|             let cs_res = self.cs.set_high(); | ||||
|             let f_res = f_res.map_err(SpiDeviceError::Spi)?; | ||||
|             let op_res = op_res.map_err(SpiDeviceError::Spi)?; | ||||
|             cs_res.map_err(SpiDeviceError::Cs)?; | ||||
|             Ok(f_res) | ||||
|             Ok(op_res) | ||||
|         }) | ||||
|     } | ||||
| } | ||||
| @@ -110,11 +164,11 @@ where | ||||
|         self.bus.lock(|bus| { | ||||
|             let mut bus = bus.borrow_mut(); | ||||
|             self.cs.set_low().map_err(SpiDeviceError::Cs)?; | ||||
|             let f_res = bus.write(words); | ||||
|             let op_res = bus.write(words); | ||||
|             let cs_res = self.cs.set_high(); | ||||
|             let f_res = f_res.map_err(SpiDeviceError::Spi)?; | ||||
|             let op_res = op_res.map_err(SpiDeviceError::Spi)?; | ||||
|             cs_res.map_err(SpiDeviceError::Cs)?; | ||||
|             Ok(f_res) | ||||
|             Ok(op_res) | ||||
|         }) | ||||
|     } | ||||
| } | ||||
| @@ -146,30 +200,85 @@ where | ||||
|     type Error = SpiDeviceError<BUS::Error, CS::Error>; | ||||
| } | ||||
|  | ||||
| impl<BUS, M, CS> embedded_hal_1::spi::blocking::SpiDevice for SpiDeviceWithConfig<'_, M, BUS, CS> | ||||
| impl<BUS, M, CS> embedded_hal_1::spi::SpiDeviceRead for SpiDeviceWithConfig<'_, M, BUS, CS> | ||||
| where | ||||
|     M: RawMutex, | ||||
|     BUS: SpiBusFlush + SetConfig, | ||||
|     BUS: SpiBusRead + SetConfig, | ||||
|     CS: OutputPin, | ||||
| { | ||||
|     type Bus = BUS; | ||||
|  | ||||
|     fn transaction<R>(&mut self, f: impl FnOnce(&mut Self::Bus) -> Result<R, BUS::Error>) -> Result<R, Self::Error> { | ||||
|     fn read_transaction(&mut self, operations: &mut [&mut [u8]]) -> Result<(), Self::Error> { | ||||
|         self.bus.lock(|bus| { | ||||
|             let mut bus = bus.borrow_mut(); | ||||
|             bus.set_config(&self.config); | ||||
|             self.cs.set_low().map_err(SpiDeviceError::Cs)?; | ||||
|  | ||||
|             let f_res = f(&mut bus); | ||||
|             let op_res = operations.iter_mut().try_for_each(|buf| bus.read(buf)); | ||||
|  | ||||
|             // On failure, it's important to still flush and deassert CS. | ||||
|             let flush_res = bus.flush(); | ||||
|             let cs_res = self.cs.set_high(); | ||||
|  | ||||
|             let f_res = f_res.map_err(SpiDeviceError::Spi)?; | ||||
|             let op_res = op_res.map_err(SpiDeviceError::Spi)?; | ||||
|             flush_res.map_err(SpiDeviceError::Spi)?; | ||||
|             cs_res.map_err(SpiDeviceError::Cs)?; | ||||
|             Ok(f_res) | ||||
|             Ok(op_res) | ||||
|         }) | ||||
|     } | ||||
| } | ||||
|  | ||||
| impl<BUS, M, CS> embedded_hal_1::spi::SpiDeviceWrite for SpiDeviceWithConfig<'_, M, BUS, CS> | ||||
| where | ||||
|     M: RawMutex, | ||||
|     BUS: SpiBusWrite + SetConfig, | ||||
|     CS: OutputPin, | ||||
| { | ||||
|     fn write_transaction(&mut self, operations: &[&[u8]]) -> Result<(), Self::Error> { | ||||
|         self.bus.lock(|bus| { | ||||
|             let mut bus = bus.borrow_mut(); | ||||
|             bus.set_config(&self.config); | ||||
|             self.cs.set_low().map_err(SpiDeviceError::Cs)?; | ||||
|  | ||||
|             let op_res = operations.iter().try_for_each(|buf| bus.write(buf)); | ||||
|  | ||||
|             // On failure, it's important to still flush and deassert CS. | ||||
|             let flush_res = bus.flush(); | ||||
|             let cs_res = self.cs.set_high(); | ||||
|  | ||||
|             let op_res = op_res.map_err(SpiDeviceError::Spi)?; | ||||
|             flush_res.map_err(SpiDeviceError::Spi)?; | ||||
|             cs_res.map_err(SpiDeviceError::Cs)?; | ||||
|             Ok(op_res) | ||||
|         }) | ||||
|     } | ||||
| } | ||||
|  | ||||
| impl<BUS, M, CS> embedded_hal_1::spi::SpiDevice for SpiDeviceWithConfig<'_, M, BUS, CS> | ||||
| where | ||||
|     M: RawMutex, | ||||
|     BUS: SpiBus + SetConfig, | ||||
|     CS: OutputPin, | ||||
| { | ||||
|     fn transaction(&mut self, operations: &mut [Operation<'_, u8>]) -> Result<(), Self::Error> { | ||||
|         self.bus.lock(|bus| { | ||||
|             let mut bus = bus.borrow_mut(); | ||||
|             bus.set_config(&self.config); | ||||
|             self.cs.set_low().map_err(SpiDeviceError::Cs)?; | ||||
|  | ||||
|             let op_res = operations.iter_mut().try_for_each(|op| match op { | ||||
|                 Operation::Read(buf) => bus.read(buf), | ||||
|                 Operation::Write(buf) => bus.write(buf), | ||||
|                 Operation::Transfer(read, write) => bus.transfer(read, write), | ||||
|                 Operation::TransferInPlace(buf) => bus.transfer_in_place(buf), | ||||
|             }); | ||||
|  | ||||
|             // On failure, it's important to still flush and deassert CS. | ||||
|             let flush_res = bus.flush(); | ||||
|             let cs_res = self.cs.set_high(); | ||||
|  | ||||
|             let op_res = op_res.map_err(SpiDeviceError::Spi)?; | ||||
|             flush_res.map_err(SpiDeviceError::Spi)?; | ||||
|             cs_res.map_err(SpiDeviceError::Cs)?; | ||||
|             Ok(op_res) | ||||
|         }) | ||||
|     } | ||||
| } | ||||
|   | ||||
							
								
								
									
										23
									
								
								embassy-executor/CHANGELOG.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										23
									
								
								embassy-executor/CHANGELOG.md
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,23 @@ | ||||
| # Changelog | ||||
|  | ||||
| All notable changes to this project will be documented in this file. | ||||
|  | ||||
| The format is based on [Keep a Changelog](https://keepachangelog.com/en/1.0.0/), | ||||
| and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0.html). | ||||
|  | ||||
| ## 0.2.0 - 2023-04-27 | ||||
|  | ||||
| - Replace unnecessary atomics in runqueue | ||||
| - add Pender, rework Cargo features. | ||||
| - add support for turbo-wakers. | ||||
| - Allow TaskStorage to auto-implement `Sync` | ||||
| - Use AtomicPtr for signal_ctx, removes 1 unsafe. | ||||
| - Replace unsound critical sections with atomics | ||||
|  | ||||
| ## 0.1.1 - 2022-11-23 | ||||
|  | ||||
| - Fix features for documentation | ||||
|  | ||||
| ## 0.1.0 - 2022-11-23 | ||||
|  | ||||
| - First release | ||||
| @@ -1,33 +1,55 @@ | ||||
| [package] | ||||
| name = "embassy-executor" | ||||
| version = "0.1.0" | ||||
| version = "0.2.0" | ||||
| edition = "2021" | ||||
|  | ||||
| license = "MIT OR Apache-2.0" | ||||
| description = "async/await executor designed for embedded usage" | ||||
| repository = "https://github.com/embassy-rs/embassy" | ||||
| categories = [ | ||||
|     "embedded", | ||||
|     "no-std", | ||||
|     "asynchronous", | ||||
| ] | ||||
|  | ||||
| [package.metadata.embassy_docs] | ||||
| src_base = "https://github.com/embassy-rs/embassy/blob/embassy-executor-v$VERSION/embassy-executor/src/" | ||||
| src_base_git = "https://github.com/embassy-rs/embassy/blob/$COMMIT/embassy-executor/src/" | ||||
| features = ["nightly", "defmt", "unstable-traits"] | ||||
| features = ["nightly", "defmt", "pender-callback"] | ||||
| flavors = [ | ||||
|     { name = "std",                       target = "x86_64-unknown-linux-gnu",   features = ["std"] }, | ||||
|     { name = "wasm",                      target = "wasm32-unknown-unknown",     features = ["wasm"] }, | ||||
|     { name = "thumbv6m-none-eabi",        target = "thumbv6m-none-eabi",         features = [] }, | ||||
|     { name = "thumbv7m-none-eabi",        target = "thumbv7m-none-eabi",         features = [] }, | ||||
|     { name = "thumbv7em-none-eabi",       target = "thumbv7em-none-eabi",        features = [] }, | ||||
|     { name = "thumbv7em-none-eabihf",     target = "thumbv7em-none-eabihf",      features = [] }, | ||||
|     { name = "thumbv8m.base-none-eabi",   target = "thumbv8m.base-none-eabi",    features = [] }, | ||||
|     { name = "thumbv8m.main-none-eabi",   target = "thumbv8m.main-none-eabi",    features = [] }, | ||||
|     { name = "thumbv8m.main-none-eabihf", target = "thumbv8m.main-none-eabihf",  features = [] }, | ||||
|     { name = "std",             target = "x86_64-unknown-linux-gnu",     features = ["arch-std", "executor-thread"] }, | ||||
|     { name = "wasm",            target = "wasm32-unknown-unknown",       features = ["arch-wasm", "executor-thread"] }, | ||||
|     { name = "cortex-m",        target = "thumbv7em-none-eabi",          features = ["arch-cortex-m", "executor-thread", "executor-interrupt"] }, | ||||
|     { name = "riscv32",         target = "riscv32imac-unknown-none-elf", features = ["arch-riscv32", "executor-thread"] }, | ||||
| ] | ||||
|  | ||||
| [package.metadata.docs.rs] | ||||
| default-target = "thumbv7em-none-eabi" | ||||
| targets = ["thumbv7em-none-eabi"] | ||||
| features = ["nightly", "defmt", "pender-callback", "arch-cortex-m", "executor-thread", "executor-interrupt"] | ||||
|  | ||||
| [features] | ||||
| default = [] | ||||
| std = ["embassy-macros/std"] | ||||
| wasm = ["dep:wasm-bindgen", "dep:js-sys", "embassy-macros/wasm"] | ||||
|  | ||||
| # Architecture | ||||
| _arch = [] # some arch was picked | ||||
| arch-std = ["_arch", "critical-section/std"] | ||||
| arch-cortex-m = ["_arch", "dep:cortex-m"] | ||||
| arch-xtensa = ["_arch"] | ||||
| arch-riscv32 = ["_arch"] | ||||
| arch-wasm = ["_arch", "dep:wasm-bindgen", "dep:js-sys"] | ||||
|  | ||||
| # Enable creating a `Pender` from an arbitrary function pointer callback. | ||||
| pender-callback = [] | ||||
|  | ||||
| # Enable the thread-mode executor (using WFE/SEV in Cortex-M, WFI in other embedded archs) | ||||
| executor-thread = [] | ||||
| # Enable the interrupt-mode executor (available in Cortex-M only) | ||||
| executor-interrupt = [] | ||||
|  | ||||
| # Enable nightly-only features | ||||
| nightly = [] | ||||
|  | ||||
| turbowakers = [] | ||||
|  | ||||
| integrated-timers = ["dep:embassy-time"] | ||||
|  | ||||
| # Trace interrupt invocations with rtos-trace. | ||||
| @@ -39,13 +61,15 @@ log = { version = "0.4.14", optional = true } | ||||
| rtos-trace = { version = "0.1.2", optional = true } | ||||
|  | ||||
| futures-util = { version = "0.3.17", default-features = false } | ||||
| embassy-macros  = { version = "0.1.0", path = "../embassy-macros" } | ||||
| embassy-macros  = { version = "0.2.0", path = "../embassy-macros" } | ||||
| embassy-time  = { version = "0.1.0", path = "../embassy-time", optional = true} | ||||
| atomic-polyfill = "1.0.1" | ||||
| critical-section = "1.1" | ||||
| cfg-if = "1.0.0" | ||||
| static_cell = "1.0" | ||||
|  | ||||
| # WASM dependencies | ||||
| # arch-cortex-m dependencies | ||||
| cortex-m = { version = "0.7.6", optional = true } | ||||
|  | ||||
| # arch-wasm dependencies | ||||
| wasm-bindgen = { version = "0.2.82", optional = true } | ||||
| js-sys = { version = "0.3", optional = true } | ||||
|   | ||||
| @@ -1,59 +1,209 @@ | ||||
| use core::arch::asm; | ||||
| use core::marker::PhantomData; | ||||
| use core::ptr; | ||||
| #[cfg(feature = "executor-thread")] | ||||
| pub use thread::*; | ||||
| #[cfg(feature = "executor-thread")] | ||||
| mod thread { | ||||
|     use core::arch::asm; | ||||
|     use core::marker::PhantomData; | ||||
|  | ||||
| use super::{raw, Spawner}; | ||||
|     #[cfg(feature = "nightly")] | ||||
|     pub use embassy_macros::main_cortex_m as main; | ||||
|  | ||||
| /// Thread mode executor, using WFE/SEV. | ||||
| /// | ||||
| /// This is the simplest and most common kind of executor. It runs on | ||||
| /// thread mode (at the lowest priority level), and uses the `WFE` ARM instruction | ||||
| /// to sleep when it has no more work to do. When a task is woken, a `SEV` instruction | ||||
| /// is executed, to make the `WFE` exit from sleep and poll the task. | ||||
| /// | ||||
| /// This executor allows for ultra low power consumption for chips where `WFE` | ||||
| /// triggers low-power sleep without extra steps. If your chip requires extra steps, | ||||
| /// you may use [`raw::Executor`] directly to program custom behavior. | ||||
| pub struct Executor { | ||||
|     inner: raw::Executor, | ||||
|     not_send: PhantomData<*mut ()>, | ||||
| } | ||||
|     use crate::raw::{Pender, PenderInner}; | ||||
|     use crate::{raw, Spawner}; | ||||
|  | ||||
| impl Executor { | ||||
|     /// Create a new Executor. | ||||
|     pub fn new() -> Self { | ||||
|         Self { | ||||
|             inner: raw::Executor::new(|_| unsafe { asm!("sev") }, ptr::null_mut()), | ||||
|             not_send: PhantomData, | ||||
|     #[derive(Copy, Clone)] | ||||
|     pub(crate) struct ThreadPender; | ||||
|  | ||||
|     impl ThreadPender { | ||||
|         pub(crate) fn pend(self) { | ||||
|             unsafe { core::arch::asm!("sev") } | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     /// Run the executor. | ||||
|     /// Thread mode executor, using WFE/SEV. | ||||
|     /// | ||||
|     /// The `init` closure is called with a [`Spawner`] that spawns tasks on | ||||
|     /// this executor. Use it to spawn the initial task(s). After `init` returns, | ||||
|     /// the executor starts running the tasks. | ||||
|     /// This is the simplest and most common kind of executor. It runs on | ||||
|     /// thread mode (at the lowest priority level), and uses the `WFE` ARM instruction | ||||
|     /// to sleep when it has no more work to do. When a task is woken, a `SEV` instruction | ||||
|     /// is executed, to make the `WFE` exit from sleep and poll the task. | ||||
|     /// | ||||
|     /// To spawn more tasks later, you may keep copies of the [`Spawner`] (it is `Copy`), | ||||
|     /// for example by passing it as an argument to the initial tasks. | ||||
|     /// | ||||
|     /// This function requires `&'static mut self`. This means you have to store the | ||||
|     /// Executor instance in a place where it'll live forever and grants you mutable | ||||
|     /// access. There's a few ways to do this: | ||||
|     /// | ||||
|     /// - a [StaticCell](https://docs.rs/static_cell/latest/static_cell/) (safe) | ||||
|     /// - a `static mut` (unsafe) | ||||
|     /// - a local variable in a function you know never returns (like `fn main() -> !`), upgrading its lifetime with `transmute`. (unsafe) | ||||
|     /// | ||||
|     /// This function never returns. | ||||
|     pub fn run(&'static mut self, init: impl FnOnce(Spawner)) -> ! { | ||||
|         init(self.inner.spawner()); | ||||
|     /// This executor allows for ultra low power consumption for chips where `WFE` | ||||
|     /// triggers low-power sleep without extra steps. If your chip requires extra steps, | ||||
|     /// you may use [`raw::Executor`] directly to program custom behavior. | ||||
|     pub struct Executor { | ||||
|         inner: raw::Executor, | ||||
|         not_send: PhantomData<*mut ()>, | ||||
|     } | ||||
|  | ||||
|     impl Executor { | ||||
|         /// Create a new Executor. | ||||
|         pub fn new() -> Self { | ||||
|             Self { | ||||
|                 inner: raw::Executor::new(Pender(PenderInner::Thread(ThreadPender))), | ||||
|                 not_send: PhantomData, | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         /// Run the executor. | ||||
|         /// | ||||
|         /// The `init` closure is called with a [`Spawner`] that spawns tasks on | ||||
|         /// this executor. Use it to spawn the initial task(s). After `init` returns, | ||||
|         /// the executor starts running the tasks. | ||||
|         /// | ||||
|         /// To spawn more tasks later, you may keep copies of the [`Spawner`] (it is `Copy`), | ||||
|         /// for example by passing it as an argument to the initial tasks. | ||||
|         /// | ||||
|         /// This function requires `&'static mut self`. This means you have to store the | ||||
|         /// Executor instance in a place where it'll live forever and grants you mutable | ||||
|         /// access. There's a few ways to do this: | ||||
|         /// | ||||
|         /// - a [StaticCell](https://docs.rs/static_cell/latest/static_cell/) (safe) | ||||
|         /// - a `static mut` (unsafe) | ||||
|         /// - a local variable in a function you know never returns (like `fn main() -> !`), upgrading its lifetime with `transmute`. (unsafe) | ||||
|         /// | ||||
|         /// This function never returns. | ||||
|         pub fn run(&'static mut self, init: impl FnOnce(Spawner)) -> ! { | ||||
|             init(self.inner.spawner()); | ||||
|  | ||||
|             loop { | ||||
|                 unsafe { | ||||
|                     self.inner.poll(); | ||||
|                     asm!("wfe"); | ||||
|                 }; | ||||
|             } | ||||
|         } | ||||
|     } | ||||
| } | ||||
|  | ||||
| #[cfg(feature = "executor-interrupt")] | ||||
| pub use interrupt::*; | ||||
| #[cfg(feature = "executor-interrupt")] | ||||
| mod interrupt { | ||||
|     use core::cell::UnsafeCell; | ||||
|     use core::mem::MaybeUninit; | ||||
|  | ||||
|     use atomic_polyfill::{AtomicBool, Ordering}; | ||||
|     use cortex_m::interrupt::InterruptNumber; | ||||
|     use cortex_m::peripheral::NVIC; | ||||
|  | ||||
|     use crate::raw::{self, Pender, PenderInner}; | ||||
|  | ||||
|     #[derive(Clone, Copy)] | ||||
|     pub(crate) struct InterruptPender(u16); | ||||
|  | ||||
|     impl InterruptPender { | ||||
|         pub(crate) fn pend(self) { | ||||
|             // STIR is faster, but is only available in v7 and higher. | ||||
|             #[cfg(not(armv6m))] | ||||
|             { | ||||
|                 let mut nvic: cortex_m::peripheral::NVIC = unsafe { core::mem::transmute(()) }; | ||||
|                 nvic.request(self); | ||||
|             } | ||||
|  | ||||
|             #[cfg(armv6m)] | ||||
|             cortex_m::peripheral::NVIC::pend(self); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     unsafe impl cortex_m::interrupt::InterruptNumber for InterruptPender { | ||||
|         fn number(self) -> u16 { | ||||
|             self.0 | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     /// Interrupt mode executor. | ||||
|     /// | ||||
|     /// This executor runs tasks in interrupt mode. The interrupt handler is set up | ||||
|     /// to poll tasks, and when a task is woken the interrupt is pended from software. | ||||
|     /// | ||||
|     /// This allows running async tasks at a priority higher than thread mode. One | ||||
|     /// use case is to leave thread mode free for non-async tasks. Another use case is | ||||
|     /// to run multiple executors: one in thread mode for low priority tasks and another in | ||||
|     /// interrupt mode for higher priority tasks. Higher priority tasks will preempt lower | ||||
|     /// priority ones. | ||||
|     /// | ||||
|     /// It is even possible to run multiple interrupt mode executors at different priorities, | ||||
|     /// by assigning different priorities to the interrupts. For an example on how to do this, | ||||
|     /// See the 'multiprio' example for 'embassy-nrf'. | ||||
|     /// | ||||
|     /// To use it, you have to pick an interrupt that won't be used by the hardware. | ||||
|     /// Some chips reserve some interrupts for this purpose, sometimes named "software interrupts" (SWI). | ||||
|     /// If this is not the case, you may use an interrupt from any unused peripheral. | ||||
|     /// | ||||
|     /// It is somewhat more complex to use, it's recommended to use the thread-mode | ||||
|     /// [`Executor`] instead, if it works for your use case. | ||||
|     pub struct InterruptExecutor { | ||||
|         started: AtomicBool, | ||||
|         executor: UnsafeCell<MaybeUninit<raw::Executor>>, | ||||
|     } | ||||
|  | ||||
|     unsafe impl Send for InterruptExecutor {} | ||||
|     unsafe impl Sync for InterruptExecutor {} | ||||
|  | ||||
|     impl InterruptExecutor { | ||||
|         /// Create a new, not started `InterruptExecutor`. | ||||
|         #[inline] | ||||
|         pub const fn new() -> Self { | ||||
|             Self { | ||||
|                 started: AtomicBool::new(false), | ||||
|                 executor: UnsafeCell::new(MaybeUninit::uninit()), | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         /// Executor interrupt callback. | ||||
|         /// | ||||
|         /// # Safety | ||||
|         /// | ||||
|         /// You MUST call this from the interrupt handler, and from nowhere else. | ||||
|         pub unsafe fn on_interrupt(&'static self) { | ||||
|             let executor = unsafe { (&*self.executor.get()).assume_init_ref() }; | ||||
|             executor.poll(); | ||||
|         } | ||||
|  | ||||
|         /// Start the executor. | ||||
|         /// | ||||
|         /// This initializes the executor, enables the interrupt, and returns. | ||||
|         /// The executor keeps running in the background through the interrupt. | ||||
|         /// | ||||
|         /// This returns a [`SendSpawner`] you can use to spawn tasks on it. A [`SendSpawner`] | ||||
|         /// is returned instead of a [`Spawner`](embassy_executor::Spawner) because the executor effectively runs in a | ||||
|         /// different "thread" (the interrupt), so spawning tasks on it is effectively | ||||
|         /// sending them. | ||||
|         /// | ||||
|         /// To obtain a [`Spawner`](embassy_executor::Spawner) for this executor, use [`Spawner::for_current_executor()`](embassy_executor::Spawner::for_current_executor()) from | ||||
|         /// a task running in it. | ||||
|         /// | ||||
|         /// # Interrupt requirements | ||||
|         /// | ||||
|         /// You must write the interrupt handler yourself, and make it call [`on_interrupt()`](Self::on_interrupt). | ||||
|         /// | ||||
|         /// This method already enables (unmasks) the interrupt, you must NOT do it yourself. | ||||
|         /// | ||||
|         /// You must set the interrupt priority before calling this method. You MUST NOT | ||||
|         /// do it after. | ||||
|         /// | ||||
|         pub fn start(&'static self, irq: impl InterruptNumber) -> crate::SendSpawner { | ||||
|             if self | ||||
|                 .started | ||||
|                 .compare_exchange(false, true, Ordering::Acquire, Ordering::Relaxed) | ||||
|                 .is_err() | ||||
|             { | ||||
|                 panic!("InterruptExecutor::start() called multiple times on the same executor."); | ||||
|             } | ||||
|  | ||||
|         loop { | ||||
|             unsafe { | ||||
|                 self.inner.poll(); | ||||
|                 asm!("wfe"); | ||||
|             }; | ||||
|                 (&mut *self.executor.get()) | ||||
|                     .as_mut_ptr() | ||||
|                     .write(raw::Executor::new(Pender(PenderInner::Interrupt(InterruptPender( | ||||
|                         irq.number(), | ||||
|                     ))))) | ||||
|             } | ||||
|  | ||||
|             let executor = unsafe { (&*self.executor.get()).assume_init_ref() }; | ||||
|  | ||||
|             unsafe { NVIC::unmask(irq) } | ||||
|  | ||||
|             executor.spawner().make_send() | ||||
|         } | ||||
|     } | ||||
| } | ||||
|   | ||||
| @@ -1,73 +1,86 @@ | ||||
| use core::marker::PhantomData; | ||||
| use core::ptr; | ||||
| #[cfg(feature = "executor-interrupt")] | ||||
| compile_error!("`executor-interrupt` is not supported with `arch-riscv32`."); | ||||
|  | ||||
| use atomic_polyfill::{AtomicBool, Ordering}; | ||||
| #[cfg(feature = "executor-thread")] | ||||
| pub use thread::*; | ||||
| #[cfg(feature = "executor-thread")] | ||||
| mod thread { | ||||
|     use core::marker::PhantomData; | ||||
|     use core::sync::atomic::{AtomicBool, Ordering}; | ||||
|  | ||||
| use super::{raw, Spawner}; | ||||
|     #[cfg(feature = "nightly")] | ||||
|     pub use embassy_macros::main_riscv as main; | ||||
|  | ||||
| /// global atomic used to keep track of whether there is work to do since sev() is not available on RISCV | ||||
| /// | ||||
| static SIGNAL_WORK_THREAD_MODE: AtomicBool = AtomicBool::new(false); | ||||
|     use crate::raw::{Pender, PenderInner}; | ||||
|     use crate::{raw, Spawner}; | ||||
|  | ||||
| /// RISCV32 Executor | ||||
| pub struct Executor { | ||||
|     inner: raw::Executor, | ||||
|     not_send: PhantomData<*mut ()>, | ||||
| } | ||||
|     #[derive(Copy, Clone)] | ||||
|     pub(crate) struct ThreadPender; | ||||
|  | ||||
| impl Executor { | ||||
|     /// Create a new Executor. | ||||
|     pub fn new() -> Self { | ||||
|         Self { | ||||
|             // use Signal_Work_Thread_Mode as substitute for local interrupt register | ||||
|             inner: raw::Executor::new( | ||||
|                 |_| { | ||||
|                     SIGNAL_WORK_THREAD_MODE.store(true, Ordering::SeqCst); | ||||
|                 }, | ||||
|                 ptr::null_mut(), | ||||
|             ), | ||||
|             not_send: PhantomData, | ||||
|     impl ThreadPender { | ||||
|         #[allow(unused)] | ||||
|         pub(crate) fn pend(self) { | ||||
|             SIGNAL_WORK_THREAD_MODE.store(true, core::sync::atomic::Ordering::SeqCst); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     /// Run the executor. | ||||
|     /// | ||||
|     /// The `init` closure is called with a [`Spawner`] that spawns tasks on | ||||
|     /// this executor. Use it to spawn the initial task(s). After `init` returns, | ||||
|     /// the executor starts running the tasks. | ||||
|     /// | ||||
|     /// To spawn more tasks later, you may keep copies of the [`Spawner`] (it is `Copy`), | ||||
|     /// for example by passing it as an argument to the initial tasks. | ||||
|     /// | ||||
|     /// This function requires `&'static mut self`. This means you have to store the | ||||
|     /// Executor instance in a place where it'll live forever and grants you mutable | ||||
|     /// access. There's a few ways to do this: | ||||
|     /// | ||||
|     /// - a [StaticCell](https://docs.rs/static_cell/latest/static_cell/) (safe) | ||||
|     /// - a `static mut` (unsafe) | ||||
|     /// - a local variable in a function you know never returns (like `fn main() -> !`), upgrading its lifetime with `transmute`. (unsafe) | ||||
|     /// | ||||
|     /// This function never returns. | ||||
|     pub fn run(&'static mut self, init: impl FnOnce(Spawner)) -> ! { | ||||
|         init(self.inner.spawner()); | ||||
|     /// global atomic used to keep track of whether there is work to do since sev() is not available on RISCV | ||||
|     static SIGNAL_WORK_THREAD_MODE: AtomicBool = AtomicBool::new(false); | ||||
|  | ||||
|         loop { | ||||
|             unsafe { | ||||
|                 self.inner.poll(); | ||||
|                 // we do not care about race conditions between the load and store operations, interrupts | ||||
|                 //will only set this value to true. | ||||
|                 critical_section::with(|_| { | ||||
|                     // if there is work to do, loop back to polling | ||||
|                     // TODO can we relax this? | ||||
|                     if SIGNAL_WORK_THREAD_MODE.load(Ordering::SeqCst) { | ||||
|                         SIGNAL_WORK_THREAD_MODE.store(false, Ordering::SeqCst); | ||||
|                     } | ||||
|                     // if not, wait for interrupt | ||||
|                     else { | ||||
|                         core::arch::asm!("wfi"); | ||||
|                     } | ||||
|                 }); | ||||
|                 // if an interrupt occurred while waiting, it will be serviced here | ||||
|     /// RISCV32 Executor | ||||
|     pub struct Executor { | ||||
|         inner: raw::Executor, | ||||
|         not_send: PhantomData<*mut ()>, | ||||
|     } | ||||
|  | ||||
|     impl Executor { | ||||
|         /// Create a new Executor. | ||||
|         pub fn new() -> Self { | ||||
|             Self { | ||||
|                 inner: raw::Executor::new(Pender(PenderInner::Thread(ThreadPender))), | ||||
|                 not_send: PhantomData, | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         /// Run the executor. | ||||
|         /// | ||||
|         /// The `init` closure is called with a [`Spawner`] that spawns tasks on | ||||
|         /// this executor. Use it to spawn the initial task(s). After `init` returns, | ||||
|         /// the executor starts running the tasks. | ||||
|         /// | ||||
|         /// To spawn more tasks later, you may keep copies of the [`Spawner`] (it is `Copy`), | ||||
|         /// for example by passing it as an argument to the initial tasks. | ||||
|         /// | ||||
|         /// This function requires `&'static mut self`. This means you have to store the | ||||
|         /// Executor instance in a place where it'll live forever and grants you mutable | ||||
|         /// access. There's a few ways to do this: | ||||
|         /// | ||||
|         /// - a [StaticCell](https://docs.rs/static_cell/latest/static_cell/) (safe) | ||||
|         /// - a `static mut` (unsafe) | ||||
|         /// - a local variable in a function you know never returns (like `fn main() -> !`), upgrading its lifetime with `transmute`. (unsafe) | ||||
|         /// | ||||
|         /// This function never returns. | ||||
|         pub fn run(&'static mut self, init: impl FnOnce(Spawner)) -> ! { | ||||
|             init(self.inner.spawner()); | ||||
|  | ||||
|             loop { | ||||
|                 unsafe { | ||||
|                     self.inner.poll(); | ||||
|                     // we do not care about race conditions between the load and store operations, interrupts | ||||
|                     //will only set this value to true. | ||||
|                     critical_section::with(|_| { | ||||
|                         // if there is work to do, loop back to polling | ||||
|                         // TODO can we relax this? | ||||
|                         if SIGNAL_WORK_THREAD_MODE.load(Ordering::SeqCst) { | ||||
|                             SIGNAL_WORK_THREAD_MODE.store(false, Ordering::SeqCst); | ||||
|                         } | ||||
|                         // if not, wait for interrupt | ||||
|                         else { | ||||
|                             core::arch::asm!("wfi"); | ||||
|                         } | ||||
|                     }); | ||||
|                     // if an interrupt occurred while waiting, it will be serviced here | ||||
|                 } | ||||
|             } | ||||
|         } | ||||
|     } | ||||
|   | ||||
| @@ -1,84 +1,100 @@ | ||||
| use std::marker::PhantomData; | ||||
| use std::sync::{Condvar, Mutex}; | ||||
| #[cfg(feature = "executor-interrupt")] | ||||
| compile_error!("`executor-interrupt` is not supported with `arch-std`."); | ||||
|  | ||||
| use super::{raw, Spawner}; | ||||
| #[cfg(feature = "executor-thread")] | ||||
| pub use thread::*; | ||||
| #[cfg(feature = "executor-thread")] | ||||
| mod thread { | ||||
|     use std::marker::PhantomData; | ||||
|     use std::sync::{Condvar, Mutex}; | ||||
|  | ||||
| /// Single-threaded std-based executor. | ||||
| pub struct Executor { | ||||
|     inner: raw::Executor, | ||||
|     not_send: PhantomData<*mut ()>, | ||||
|     signaler: &'static Signaler, | ||||
| } | ||||
|     #[cfg(feature = "nightly")] | ||||
|     pub use embassy_macros::main_std as main; | ||||
|  | ||||
| impl Executor { | ||||
|     /// Create a new Executor. | ||||
|     pub fn new() -> Self { | ||||
|         let signaler = &*Box::leak(Box::new(Signaler::new())); | ||||
|         Self { | ||||
|             inner: raw::Executor::new( | ||||
|                 |p| unsafe { | ||||
|                     let s = &*(p as *const () as *const Signaler); | ||||
|                     s.signal() | ||||
|                 }, | ||||
|                 signaler as *const _ as _, | ||||
|             ), | ||||
|             not_send: PhantomData, | ||||
|             signaler, | ||||
|     use crate::raw::{Pender, PenderInner}; | ||||
|     use crate::{raw, Spawner}; | ||||
|  | ||||
|     #[derive(Copy, Clone)] | ||||
|     pub(crate) struct ThreadPender(&'static Signaler); | ||||
|  | ||||
|     impl ThreadPender { | ||||
|         #[allow(unused)] | ||||
|         pub(crate) fn pend(self) { | ||||
|             self.0.signal() | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     /// Run the executor. | ||||
|     /// | ||||
|     /// The `init` closure is called with a [`Spawner`] that spawns tasks on | ||||
|     /// this executor. Use it to spawn the initial task(s). After `init` returns, | ||||
|     /// the executor starts running the tasks. | ||||
|     /// | ||||
|     /// To spawn more tasks later, you may keep copies of the [`Spawner`] (it is `Copy`), | ||||
|     /// for example by passing it as an argument to the initial tasks. | ||||
|     /// | ||||
|     /// This function requires `&'static mut self`. This means you have to store the | ||||
|     /// Executor instance in a place where it'll live forever and grants you mutable | ||||
|     /// access. There's a few ways to do this: | ||||
|     /// | ||||
|     /// - a [StaticCell](https://docs.rs/static_cell/latest/static_cell/) (safe) | ||||
|     /// - a `static mut` (unsafe) | ||||
|     /// - a local variable in a function you know never returns (like `fn main() -> !`), upgrading its lifetime with `transmute`. (unsafe) | ||||
|     /// | ||||
|     /// This function never returns. | ||||
|     pub fn run(&'static mut self, init: impl FnOnce(Spawner)) -> ! { | ||||
|         init(self.inner.spawner()); | ||||
|  | ||||
|         loop { | ||||
|             unsafe { self.inner.poll() }; | ||||
|             self.signaler.wait() | ||||
|         } | ||||
|     /// Single-threaded std-based executor. | ||||
|     pub struct Executor { | ||||
|         inner: raw::Executor, | ||||
|         not_send: PhantomData<*mut ()>, | ||||
|         signaler: &'static Signaler, | ||||
|     } | ||||
| } | ||||
|  | ||||
| struct Signaler { | ||||
|     mutex: Mutex<bool>, | ||||
|     condvar: Condvar, | ||||
| } | ||||
|     impl Executor { | ||||
|         /// Create a new Executor. | ||||
|         pub fn new() -> Self { | ||||
|             let signaler = &*Box::leak(Box::new(Signaler::new())); | ||||
|             Self { | ||||
|                 inner: raw::Executor::new(Pender(PenderInner::Thread(ThreadPender(signaler)))), | ||||
|                 not_send: PhantomData, | ||||
|                 signaler, | ||||
|             } | ||||
|         } | ||||
|  | ||||
| impl Signaler { | ||||
|     fn new() -> Self { | ||||
|         Self { | ||||
|             mutex: Mutex::new(false), | ||||
|             condvar: Condvar::new(), | ||||
|         /// Run the executor. | ||||
|         /// | ||||
|         /// The `init` closure is called with a [`Spawner`] that spawns tasks on | ||||
|         /// this executor. Use it to spawn the initial task(s). After `init` returns, | ||||
|         /// the executor starts running the tasks. | ||||
|         /// | ||||
|         /// To spawn more tasks later, you may keep copies of the [`Spawner`] (it is `Copy`), | ||||
|         /// for example by passing it as an argument to the initial tasks. | ||||
|         /// | ||||
|         /// This function requires `&'static mut self`. This means you have to store the | ||||
|         /// Executor instance in a place where it'll live forever and grants you mutable | ||||
|         /// access. There's a few ways to do this: | ||||
|         /// | ||||
|         /// - a [StaticCell](https://docs.rs/static_cell/latest/static_cell/) (safe) | ||||
|         /// - a `static mut` (unsafe) | ||||
|         /// - a local variable in a function you know never returns (like `fn main() -> !`), upgrading its lifetime with `transmute`. (unsafe) | ||||
|         /// | ||||
|         /// This function never returns. | ||||
|         pub fn run(&'static mut self, init: impl FnOnce(Spawner)) -> ! { | ||||
|             init(self.inner.spawner()); | ||||
|  | ||||
|             loop { | ||||
|                 unsafe { self.inner.poll() }; | ||||
|                 self.signaler.wait() | ||||
|             } | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     fn wait(&self) { | ||||
|         let mut signaled = self.mutex.lock().unwrap(); | ||||
|         while !*signaled { | ||||
|             signaled = self.condvar.wait(signaled).unwrap(); | ||||
|         } | ||||
|         *signaled = false; | ||||
|     struct Signaler { | ||||
|         mutex: Mutex<bool>, | ||||
|         condvar: Condvar, | ||||
|     } | ||||
|  | ||||
|     fn signal(&self) { | ||||
|         let mut signaled = self.mutex.lock().unwrap(); | ||||
|         *signaled = true; | ||||
|         self.condvar.notify_one(); | ||||
|     impl Signaler { | ||||
|         fn new() -> Self { | ||||
|             Self { | ||||
|                 mutex: Mutex::new(false), | ||||
|                 condvar: Condvar::new(), | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         fn wait(&self) { | ||||
|             let mut signaled = self.mutex.lock().unwrap(); | ||||
|             while !*signaled { | ||||
|                 signaled = self.condvar.wait(signaled).unwrap(); | ||||
|             } | ||||
|             *signaled = false; | ||||
|         } | ||||
|  | ||||
|         fn signal(&self) { | ||||
|             let mut signaled = self.mutex.lock().unwrap(); | ||||
|             *signaled = true; | ||||
|             self.condvar.notify_one(); | ||||
|         } | ||||
|     } | ||||
| } | ||||
|   | ||||
| @@ -1,74 +1,88 @@ | ||||
| use core::marker::PhantomData; | ||||
| #[cfg(feature = "executor-interrupt")] | ||||
| compile_error!("`executor-interrupt` is not supported with `arch-wasm`."); | ||||
|  | ||||
| use js_sys::Promise; | ||||
| use wasm_bindgen::prelude::*; | ||||
| #[cfg(feature = "executor-thread")] | ||||
| pub use thread::*; | ||||
| #[cfg(feature = "executor-thread")] | ||||
| mod thread { | ||||
|  | ||||
| use super::raw::util::UninitCell; | ||||
| use super::raw::{self}; | ||||
| use super::Spawner; | ||||
|     use core::marker::PhantomData; | ||||
|  | ||||
| /// WASM executor, wasm_bindgen to schedule tasks on the JS event loop. | ||||
| pub struct Executor { | ||||
|     inner: raw::Executor, | ||||
|     ctx: &'static WasmContext, | ||||
|     not_send: PhantomData<*mut ()>, | ||||
| } | ||||
|     #[cfg(feature = "nightly")] | ||||
|     pub use embassy_macros::main_wasm as main; | ||||
|     use js_sys::Promise; | ||||
|     use wasm_bindgen::prelude::*; | ||||
|  | ||||
| pub(crate) struct WasmContext { | ||||
|     promise: Promise, | ||||
|     closure: UninitCell<Closure<dyn FnMut(JsValue)>>, | ||||
| } | ||||
|     use crate::raw::util::UninitCell; | ||||
|     use crate::raw::{Pender, PenderInner}; | ||||
|     use crate::{raw, Spawner}; | ||||
|  | ||||
| impl WasmContext { | ||||
|     pub fn new() -> Self { | ||||
|         Self { | ||||
|             promise: Promise::resolve(&JsValue::undefined()), | ||||
|             closure: UninitCell::uninit(), | ||||
|         } | ||||
|     /// WASM executor, wasm_bindgen to schedule tasks on the JS event loop. | ||||
|     pub struct Executor { | ||||
|         inner: raw::Executor, | ||||
|         ctx: &'static WasmContext, | ||||
|         not_send: PhantomData<*mut ()>, | ||||
|     } | ||||
| } | ||||
|  | ||||
| impl Executor { | ||||
|     /// Create a new Executor. | ||||
|     pub fn new() -> Self { | ||||
|         let ctx = &*Box::leak(Box::new(WasmContext::new())); | ||||
|         let inner = raw::Executor::new( | ||||
|             |p| unsafe { | ||||
|                 let ctx = &*(p as *const () as *const WasmContext); | ||||
|                 let _ = ctx.promise.then(ctx.closure.as_mut()); | ||||
|             }, | ||||
|             ctx as *const _ as _, | ||||
|         ); | ||||
|         Self { | ||||
|             inner, | ||||
|             not_send: PhantomData, | ||||
|             ctx, | ||||
|     pub(crate) struct WasmContext { | ||||
|         promise: Promise, | ||||
|         closure: UninitCell<Closure<dyn FnMut(JsValue)>>, | ||||
|     } | ||||
|  | ||||
|     #[derive(Copy, Clone)] | ||||
|     pub(crate) struct ThreadPender(&'static WasmContext); | ||||
|  | ||||
|     impl ThreadPender { | ||||
|         #[allow(unused)] | ||||
|         pub(crate) fn pend(self) { | ||||
|             let _ = self.0.promise.then(unsafe { self.0.closure.as_mut() }); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     /// Run the executor. | ||||
|     /// | ||||
|     /// The `init` closure is called with a [`Spawner`] that spawns tasks on | ||||
|     /// this executor. Use it to spawn the initial task(s). After `init` returns, | ||||
|     /// the executor starts running the tasks. | ||||
|     /// | ||||
|     /// To spawn more tasks later, you may keep copies of the [`Spawner`] (it is `Copy`), | ||||
|     /// for example by passing it as an argument to the initial tasks. | ||||
|     /// | ||||
|     /// This function requires `&'static mut self`. This means you have to store the | ||||
|     /// Executor instance in a place where it'll live forever and grants you mutable | ||||
|     /// access. There's a few ways to do this: | ||||
|     /// | ||||
|     /// - a [StaticCell](https://docs.rs/static_cell/latest/static_cell/) (safe) | ||||
|     /// - a `static mut` (unsafe) | ||||
|     /// - a local variable in a function you know never returns (like `fn main() -> !`), upgrading its lifetime with `transmute`. (unsafe) | ||||
|     pub fn start(&'static mut self, init: impl FnOnce(Spawner)) { | ||||
|         unsafe { | ||||
|             let executor = &self.inner; | ||||
|             self.ctx.closure.write(Closure::new(move |_| { | ||||
|                 executor.poll(); | ||||
|             })); | ||||
|             init(self.inner.spawner()); | ||||
|     impl WasmContext { | ||||
|         pub fn new() -> Self { | ||||
|             Self { | ||||
|                 promise: Promise::resolve(&JsValue::undefined()), | ||||
|                 closure: UninitCell::uninit(), | ||||
|             } | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     impl Executor { | ||||
|         /// Create a new Executor. | ||||
|         pub fn new() -> Self { | ||||
|             let ctx = &*Box::leak(Box::new(WasmContext::new())); | ||||
|             Self { | ||||
|                 inner: raw::Executor::new(Pender(PenderInner::Thread(ThreadPender(ctx)))), | ||||
|                 not_send: PhantomData, | ||||
|                 ctx, | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         /// Run the executor. | ||||
|         /// | ||||
|         /// The `init` closure is called with a [`Spawner`] that spawns tasks on | ||||
|         /// this executor. Use it to spawn the initial task(s). After `init` returns, | ||||
|         /// the executor starts running the tasks. | ||||
|         /// | ||||
|         /// To spawn more tasks later, you may keep copies of the [`Spawner`] (it is `Copy`), | ||||
|         /// for example by passing it as an argument to the initial tasks. | ||||
|         /// | ||||
|         /// This function requires `&'static mut self`. This means you have to store the | ||||
|         /// Executor instance in a place where it'll live forever and grants you mutable | ||||
|         /// access. There's a few ways to do this: | ||||
|         /// | ||||
|         /// - a [StaticCell](https://docs.rs/static_cell/latest/static_cell/) (safe) | ||||
|         /// - a `static mut` (unsafe) | ||||
|         /// - a local variable in a function you know never returns (like `fn main() -> !`), upgrading its lifetime with `transmute`. (unsafe) | ||||
|         pub fn start(&'static mut self, init: impl FnOnce(Spawner)) { | ||||
|             unsafe { | ||||
|                 let executor = &self.inner; | ||||
|                 self.ctx.closure.write(Closure::new(move |_| { | ||||
|                     executor.poll(); | ||||
|                 })); | ||||
|                 init(self.inner.spawner()); | ||||
|             } | ||||
|         } | ||||
|     } | ||||
| } | ||||
|   | ||||
| @@ -1,74 +1,84 @@ | ||||
| use core::marker::PhantomData; | ||||
| use core::ptr; | ||||
| #[cfg(feature = "executor-interrupt")] | ||||
| compile_error!("`executor-interrupt` is not supported with `arch-xtensa`."); | ||||
|  | ||||
| use atomic_polyfill::{AtomicBool, Ordering}; | ||||
| #[cfg(feature = "executor-thread")] | ||||
| pub use thread::*; | ||||
| #[cfg(feature = "executor-thread")] | ||||
| mod thread { | ||||
|     use core::marker::PhantomData; | ||||
|     use core::sync::atomic::{AtomicBool, Ordering}; | ||||
|  | ||||
| use super::{raw, Spawner}; | ||||
|     use crate::raw::{Pender, PenderInner}; | ||||
|     use crate::{raw, Spawner}; | ||||
|  | ||||
| /// global atomic used to keep track of whether there is work to do since sev() is not available on Xtensa | ||||
| /// | ||||
| static SIGNAL_WORK_THREAD_MODE: AtomicBool = AtomicBool::new(false); | ||||
|     #[derive(Copy, Clone)] | ||||
|     pub(crate) struct ThreadPender; | ||||
|  | ||||
| /// Xtensa Executor | ||||
| pub struct Executor { | ||||
|     inner: raw::Executor, | ||||
|     not_send: PhantomData<*mut ()>, | ||||
| } | ||||
|  | ||||
| impl Executor { | ||||
|     /// Create a new Executor. | ||||
|     pub fn new() -> Self { | ||||
|         Self { | ||||
|             // use Signal_Work_Thread_Mode as substitute for local interrupt register | ||||
|             inner: raw::Executor::new( | ||||
|                 |_| { | ||||
|                     SIGNAL_WORK_THREAD_MODE.store(true, Ordering::SeqCst); | ||||
|                 }, | ||||
|                 ptr::null_mut(), | ||||
|             ), | ||||
|             not_send: PhantomData, | ||||
|     impl ThreadPender { | ||||
|         #[allow(unused)] | ||||
|         pub(crate) fn pend(self) { | ||||
|             SIGNAL_WORK_THREAD_MODE.store(true, core::sync::atomic::Ordering::SeqCst); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     /// Run the executor. | ||||
|     /// | ||||
|     /// The `init` closure is called with a [`Spawner`] that spawns tasks on | ||||
|     /// this executor. Use it to spawn the initial task(s). After `init` returns, | ||||
|     /// the executor starts running the tasks. | ||||
|     /// | ||||
|     /// To spawn more tasks later, you may keep copies of the [`Spawner`] (it is `Copy`), | ||||
|     /// for example by passing it as an argument to the initial tasks. | ||||
|     /// | ||||
|     /// This function requires `&'static mut self`. This means you have to store the | ||||
|     /// Executor instance in a place where it'll live forever and grants you mutable | ||||
|     /// access. There's a few ways to do this: | ||||
|     /// | ||||
|     /// - a [StaticCell](https://docs.rs/static_cell/latest/static_cell/) (safe) | ||||
|     /// - a `static mut` (unsafe) | ||||
|     /// - a local variable in a function you know never returns (like `fn main() -> !`), upgrading its lifetime with `transmute`. (unsafe) | ||||
|     /// | ||||
|     /// This function never returns. | ||||
|     pub fn run(&'static mut self, init: impl FnOnce(Spawner)) -> ! { | ||||
|         init(self.inner.spawner()); | ||||
|     /// global atomic used to keep track of whether there is work to do since sev() is not available on Xtensa | ||||
|     static SIGNAL_WORK_THREAD_MODE: AtomicBool = AtomicBool::new(false); | ||||
|  | ||||
|         loop { | ||||
|             unsafe { | ||||
|                 self.inner.poll(); | ||||
|                 // we do not care about race conditions between the load and store operations, interrupts | ||||
|                 // will only set this value to true. | ||||
|                 // if there is work to do, loop back to polling | ||||
|                 // TODO can we relax this? | ||||
|                 critical_section::with(|_| { | ||||
|                     if SIGNAL_WORK_THREAD_MODE.load(Ordering::SeqCst) { | ||||
|                         SIGNAL_WORK_THREAD_MODE.store(false, Ordering::SeqCst); | ||||
|                     } else { | ||||
|                         // waiti sets the PS.INTLEVEL when slipping into sleep | ||||
|                         // because critical sections in Xtensa are implemented via increasing | ||||
|                         // PS.INTLEVEL the critical section ends here | ||||
|                         // take care not add code after `waiti` if it needs to be inside the CS | ||||
|                         core::arch::asm!("waiti 0"); // critical section ends here | ||||
|                     } | ||||
|                 }); | ||||
|     /// Xtensa Executor | ||||
|     pub struct Executor { | ||||
|         inner: raw::Executor, | ||||
|         not_send: PhantomData<*mut ()>, | ||||
|     } | ||||
|  | ||||
|     impl Executor { | ||||
|         /// Create a new Executor. | ||||
|         pub fn new() -> Self { | ||||
|             Self { | ||||
|                 inner: raw::Executor::new(Pender(PenderInner::Thread(ThreadPender))), | ||||
|                 not_send: PhantomData, | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         /// Run the executor. | ||||
|         /// | ||||
|         /// The `init` closure is called with a [`Spawner`] that spawns tasks on | ||||
|         /// this executor. Use it to spawn the initial task(s). After `init` returns, | ||||
|         /// the executor starts running the tasks. | ||||
|         /// | ||||
|         /// To spawn more tasks later, you may keep copies of the [`Spawner`] (it is `Copy`), | ||||
|         /// for example by passing it as an argument to the initial tasks. | ||||
|         /// | ||||
|         /// This function requires `&'static mut self`. This means you have to store the | ||||
|         /// Executor instance in a place where it'll live forever and grants you mutable | ||||
|         /// access. There's a few ways to do this: | ||||
|         /// | ||||
|         /// - a [StaticCell](https://docs.rs/static_cell/latest/static_cell/) (safe) | ||||
|         /// - a `static mut` (unsafe) | ||||
|         /// - a local variable in a function you know never returns (like `fn main() -> !`), upgrading its lifetime with `transmute`. (unsafe) | ||||
|         /// | ||||
|         /// This function never returns. | ||||
|         pub fn run(&'static mut self, init: impl FnOnce(Spawner)) -> ! { | ||||
|             init(self.inner.spawner()); | ||||
|  | ||||
|             loop { | ||||
|                 unsafe { | ||||
|                     self.inner.poll(); | ||||
|                     // we do not care about race conditions between the load and store operations, interrupts | ||||
|                     // will only set this value to true. | ||||
|                     // if there is work to do, loop back to polling | ||||
|                     // TODO can we relax this? | ||||
|                     critical_section::with(|_| { | ||||
|                         if SIGNAL_WORK_THREAD_MODE.load(Ordering::SeqCst) { | ||||
|                             SIGNAL_WORK_THREAD_MODE.store(false, Ordering::SeqCst); | ||||
|                         } else { | ||||
|                             // waiti sets the PS.INTLEVEL when slipping into sleep | ||||
|                             // because critical sections in Xtensa are implemented via increasing | ||||
|                             // PS.INTLEVEL the critical section ends here | ||||
|                             // take care not add code after `waiti` if it needs to be inside the CS | ||||
|                             core::arch::asm!("waiti 0"); // critical section ends here | ||||
|                         } | ||||
|                     }); | ||||
|                 } | ||||
|             } | ||||
|         } | ||||
|     } | ||||
|   | ||||
| @@ -1,5 +1,5 @@ | ||||
| #![cfg_attr(not(any(feature = "std", feature = "wasm")), no_std)] | ||||
| #![cfg_attr(all(feature = "nightly", target_arch = "xtensa"), feature(asm_experimental_arch))] | ||||
| #![cfg_attr(not(any(feature = "arch-std", feature = "arch-wasm")), no_std)] | ||||
| #![cfg_attr(all(feature = "nightly", feature = "arch-xtensa"), feature(asm_experimental_arch))] | ||||
| #![allow(clippy::new_without_default)] | ||||
| #![doc = include_str!("../README.md")] | ||||
| #![warn(missing_docs)] | ||||
| @@ -8,41 +8,45 @@ | ||||
| pub(crate) mod fmt; | ||||
|  | ||||
| #[cfg(feature = "nightly")] | ||||
| pub use embassy_macros::{main, task}; | ||||
| pub use embassy_macros::task; | ||||
|  | ||||
| cfg_if::cfg_if! { | ||||
|     if #[cfg(cortex_m)] { | ||||
|         #[path="arch/cortex_m.rs"] | ||||
|         mod arch; | ||||
|         pub use arch::*; | ||||
|     } | ||||
|     else if #[cfg(target_arch="riscv32")] { | ||||
|         #[path="arch/riscv32.rs"] | ||||
|         mod arch; | ||||
|         pub use arch::*; | ||||
|     } | ||||
|     else if #[cfg(all(target_arch="xtensa", feature = "nightly"))] { | ||||
|         #[path="arch/xtensa.rs"] | ||||
|         mod arch; | ||||
|         pub use arch::*; | ||||
|     } | ||||
|     else if #[cfg(feature="wasm")] { | ||||
|         #[path="arch/wasm.rs"] | ||||
|         mod arch; | ||||
|         pub use arch::*; | ||||
|     } | ||||
|     else if #[cfg(feature="std")] { | ||||
|         #[path="arch/std.rs"] | ||||
|         mod arch; | ||||
|         pub use arch::*; | ||||
|     } | ||||
| macro_rules! check_at_most_one { | ||||
|     (@amo [$($feats:literal)*] [] [$($res:tt)*]) => { | ||||
|         #[cfg(any($($res)*))] | ||||
|         compile_error!(concat!("At most one of these features can be enabled at the same time:", $(" `", $feats, "`",)*)); | ||||
|     }; | ||||
|     (@amo $feats:tt [$curr:literal $($rest:literal)*] [$($res:tt)*]) => { | ||||
|         check_at_most_one!(@amo $feats [$($rest)*] [$($res)* $(all(feature=$curr, feature=$rest),)*]); | ||||
|     }; | ||||
|     ($($f:literal),*$(,)?) => { | ||||
|         check_at_most_one!(@amo [$($f)*] [$($f)*] []); | ||||
|     }; | ||||
| } | ||||
| check_at_most_one!("arch-cortex-m", "arch-riscv32", "arch-xtensa", "arch-std", "arch-wasm",); | ||||
|  | ||||
| #[cfg(feature = "_arch")] | ||||
| #[cfg_attr(feature = "arch-cortex-m", path = "arch/cortex_m.rs")] | ||||
| #[cfg_attr(feature = "arch-riscv32", path = "arch/riscv32.rs")] | ||||
| #[cfg_attr(feature = "arch-xtensa", path = "arch/xtensa.rs")] | ||||
| #[cfg_attr(feature = "arch-std", path = "arch/std.rs")] | ||||
| #[cfg_attr(feature = "arch-wasm", path = "arch/wasm.rs")] | ||||
| mod arch; | ||||
|  | ||||
| #[cfg(feature = "_arch")] | ||||
| pub use arch::*; | ||||
|  | ||||
| pub mod raw; | ||||
|  | ||||
| mod spawner; | ||||
| pub use spawner::*; | ||||
|  | ||||
| /// Implementation details for embassy macros. | ||||
| /// Do not use. Used for macros and HALs only. Not covered by semver guarantees. | ||||
| #[doc(hidden)] | ||||
| /// Implementation details for embassy macros. DO NOT USE. | ||||
| pub mod export { | ||||
| pub mod _export { | ||||
|     #[cfg(feature = "rtos-trace")] | ||||
|     pub use rtos_trace::trace; | ||||
|     pub use static_cell::StaticCell; | ||||
|  | ||||
|     /// Expands the given block of code when `embassy-executor` is compiled with | ||||
|     /// the `rtos-trace-interrupt` feature. | ||||
| @@ -62,14 +66,3 @@ pub mod export { | ||||
|         ($($tt:tt)*) => {}; | ||||
|     } | ||||
| } | ||||
|  | ||||
| pub mod raw; | ||||
|  | ||||
| mod spawner; | ||||
| pub use spawner::*; | ||||
|  | ||||
| /// Do not use. Used for macros and HALs only. Not covered by semver guarantees. | ||||
| #[doc(hidden)] | ||||
| pub mod _export { | ||||
|     pub use static_cell::StaticCell; | ||||
| } | ||||
|   | ||||
| @@ -11,17 +11,17 @@ mod run_queue; | ||||
| #[cfg(feature = "integrated-timers")] | ||||
| mod timer_queue; | ||||
| pub(crate) mod util; | ||||
| #[cfg_attr(feature = "turbowakers", path = "waker_turbo.rs")] | ||||
| mod waker; | ||||
|  | ||||
| use core::cell::Cell; | ||||
| use core::future::Future; | ||||
| use core::marker::PhantomData; | ||||
| use core::mem; | ||||
| use core::pin::Pin; | ||||
| use core::ptr::NonNull; | ||||
| use core::task::{Context, Poll}; | ||||
| use core::{mem, ptr}; | ||||
|  | ||||
| use atomic_polyfill::{AtomicU32, Ordering}; | ||||
| use critical_section::CriticalSection; | ||||
| #[cfg(feature = "integrated-timers")] | ||||
| use embassy_time::driver::{self, AlarmHandle}; | ||||
| #[cfg(feature = "integrated-timers")] | ||||
| @@ -30,7 +30,7 @@ use embassy_time::Instant; | ||||
| use rtos_trace::trace; | ||||
|  | ||||
| use self::run_queue::{RunQueue, RunQueueItem}; | ||||
| use self::util::UninitCell; | ||||
| use self::util::{SyncUnsafeCell, UninitCell}; | ||||
| pub use self::waker::task_from_waker; | ||||
| use super::SpawnToken; | ||||
|  | ||||
| @@ -43,35 +43,49 @@ pub(crate) const STATE_RUN_QUEUED: u32 = 1 << 1; | ||||
| pub(crate) const STATE_TIMER_QUEUED: u32 = 1 << 2; | ||||
|  | ||||
| /// Raw task header for use in task pointers. | ||||
| /// | ||||
| /// This is an opaque struct, used for raw pointers to tasks, for use | ||||
| /// with funtions like [`wake_task`] and [`task_from_waker`]. | ||||
| pub struct TaskHeader { | ||||
| pub(crate) struct TaskHeader { | ||||
|     pub(crate) state: AtomicU32, | ||||
|     pub(crate) run_queue_item: RunQueueItem, | ||||
|     pub(crate) executor: Cell<*const Executor>, // Valid if state != 0 | ||||
|     pub(crate) poll_fn: UninitCell<unsafe fn(NonNull<TaskHeader>)>, // Valid if STATE_SPAWNED | ||||
|     pub(crate) executor: SyncUnsafeCell<Option<&'static SyncExecutor>>, | ||||
|     poll_fn: SyncUnsafeCell<Option<unsafe fn(TaskRef)>>, | ||||
|  | ||||
|     #[cfg(feature = "integrated-timers")] | ||||
|     pub(crate) expires_at: Cell<Instant>, | ||||
|     pub(crate) expires_at: SyncUnsafeCell<Instant>, | ||||
|     #[cfg(feature = "integrated-timers")] | ||||
|     pub(crate) timer_queue_item: timer_queue::TimerQueueItem, | ||||
| } | ||||
|  | ||||
| impl TaskHeader { | ||||
|     pub(crate) const fn new() -> Self { | ||||
|         Self { | ||||
|             state: AtomicU32::new(0), | ||||
|             run_queue_item: RunQueueItem::new(), | ||||
|             executor: Cell::new(ptr::null()), | ||||
|             poll_fn: UninitCell::uninit(), | ||||
| /// This is essentially a `&'static TaskStorage<F>` where the type of the future has been erased. | ||||
| #[derive(Clone, Copy)] | ||||
| pub struct TaskRef { | ||||
|     ptr: NonNull<TaskHeader>, | ||||
| } | ||||
|  | ||||
|             #[cfg(feature = "integrated-timers")] | ||||
|             expires_at: Cell::new(Instant::from_ticks(0)), | ||||
|             #[cfg(feature = "integrated-timers")] | ||||
|             timer_queue_item: timer_queue::TimerQueueItem::new(), | ||||
| unsafe impl Send for TaskRef where &'static TaskHeader: Send {} | ||||
| unsafe impl Sync for TaskRef where &'static TaskHeader: Sync {} | ||||
|  | ||||
| impl TaskRef { | ||||
|     fn new<F: Future + 'static>(task: &'static TaskStorage<F>) -> Self { | ||||
|         Self { | ||||
|             ptr: NonNull::from(task).cast(), | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     /// Safety: The pointer must have been obtained with `Task::as_ptr` | ||||
|     pub(crate) unsafe fn from_ptr(ptr: *const TaskHeader) -> Self { | ||||
|         Self { | ||||
|             ptr: NonNull::new_unchecked(ptr as *mut TaskHeader), | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     pub(crate) fn header(self) -> &'static TaskHeader { | ||||
|         unsafe { self.ptr.as_ref() } | ||||
|     } | ||||
|  | ||||
|     /// The returned pointer is valid for the entire TaskStorage. | ||||
|     pub(crate) fn as_ptr(self) -> *const TaskHeader { | ||||
|         self.ptr.as_ptr() | ||||
|     } | ||||
| } | ||||
|  | ||||
| /// Raw storage in which a task can be spawned. | ||||
| @@ -101,7 +115,18 @@ impl<F: Future + 'static> TaskStorage<F> { | ||||
|     /// Create a new TaskStorage, in not-spawned state. | ||||
|     pub const fn new() -> Self { | ||||
|         Self { | ||||
|             raw: TaskHeader::new(), | ||||
|             raw: TaskHeader { | ||||
|                 state: AtomicU32::new(0), | ||||
|                 run_queue_item: RunQueueItem::new(), | ||||
|                 executor: SyncUnsafeCell::new(None), | ||||
|                 // Note: this is lazily initialized so that a static `TaskStorage` will go in `.bss` | ||||
|                 poll_fn: SyncUnsafeCell::new(None), | ||||
|  | ||||
|                 #[cfg(feature = "integrated-timers")] | ||||
|                 expires_at: SyncUnsafeCell::new(Instant::from_ticks(0)), | ||||
|                 #[cfg(feature = "integrated-timers")] | ||||
|                 timer_queue_item: timer_queue::TimerQueueItem::new(), | ||||
|             }, | ||||
|             future: UninitCell::uninit(), | ||||
|         } | ||||
|     } | ||||
| @@ -120,29 +145,17 @@ impl<F: Future + 'static> TaskStorage<F> { | ||||
|     /// Once the task has finished running, you may spawn it again. It is allowed to spawn it | ||||
|     /// on a different executor. | ||||
|     pub fn spawn(&'static self, future: impl FnOnce() -> F) -> SpawnToken<impl Sized> { | ||||
|         if self.spawn_mark_used() { | ||||
|             return unsafe { SpawnToken::<F>::new(self.spawn_initialize(future)) }; | ||||
|         let task = AvailableTask::claim(self); | ||||
|         match task { | ||||
|             Some(task) => { | ||||
|                 let task = task.initialize(future); | ||||
|                 unsafe { SpawnToken::<F>::new(task) } | ||||
|             } | ||||
|             None => SpawnToken::new_failed(), | ||||
|         } | ||||
|  | ||||
|         SpawnToken::<F>::new_failed() | ||||
|     } | ||||
|  | ||||
|     fn spawn_mark_used(&'static self) -> bool { | ||||
|         let state = STATE_SPAWNED | STATE_RUN_QUEUED; | ||||
|         self.raw | ||||
|             .state | ||||
|             .compare_exchange(0, state, Ordering::AcqRel, Ordering::Acquire) | ||||
|             .is_ok() | ||||
|     } | ||||
|  | ||||
|     unsafe fn spawn_initialize(&'static self, future: impl FnOnce() -> F) -> NonNull<TaskHeader> { | ||||
|         // Initialize the task | ||||
|         self.raw.poll_fn.write(Self::poll); | ||||
|         self.future.write(future()); | ||||
|         NonNull::new_unchecked(self as *const TaskStorage<F> as *const TaskHeader as *mut TaskHeader) | ||||
|     } | ||||
|  | ||||
|     unsafe fn poll(p: NonNull<TaskHeader>) { | ||||
|     unsafe fn poll(p: TaskRef) { | ||||
|         let this = &*(p.as_ptr() as *const TaskStorage<F>); | ||||
|  | ||||
|         let future = Pin::new_unchecked(this.future.as_mut()); | ||||
| @@ -160,9 +173,37 @@ impl<F: Future + 'static> TaskStorage<F> { | ||||
|         // it's a noop for our waker. | ||||
|         mem::forget(waker); | ||||
|     } | ||||
|  | ||||
|     #[doc(hidden)] | ||||
|     #[allow(dead_code)] | ||||
|     fn _assert_sync(self) { | ||||
|         fn assert_sync<T: Sync>(_: T) {} | ||||
|  | ||||
|         assert_sync(self) | ||||
|     } | ||||
| } | ||||
|  | ||||
| unsafe impl<F: Future + 'static> Sync for TaskStorage<F> {} | ||||
| struct AvailableTask<F: Future + 'static> { | ||||
|     task: &'static TaskStorage<F>, | ||||
| } | ||||
|  | ||||
| impl<F: Future + 'static> AvailableTask<F> { | ||||
|     fn claim(task: &'static TaskStorage<F>) -> Option<Self> { | ||||
|         task.raw | ||||
|             .state | ||||
|             .compare_exchange(0, STATE_SPAWNED | STATE_RUN_QUEUED, Ordering::AcqRel, Ordering::Acquire) | ||||
|             .ok() | ||||
|             .map(|_| Self { task }) | ||||
|     } | ||||
|  | ||||
|     fn initialize(self, future: impl FnOnce() -> F) -> TaskRef { | ||||
|         unsafe { | ||||
|             self.task.raw.poll_fn.set(Some(TaskStorage::<F>::poll)); | ||||
|             self.task.future.write(future()); | ||||
|         } | ||||
|         TaskRef::new(self.task) | ||||
|     } | ||||
| } | ||||
|  | ||||
| /// Raw storage that can hold up to N tasks of the same type. | ||||
| /// | ||||
| @@ -187,13 +228,14 @@ impl<F: Future + 'static, const N: usize> TaskPool<F, N> { | ||||
|     /// is currently free. If none is free, a "poisoned" SpawnToken is returned, | ||||
|     /// which will cause [`Spawner::spawn()`](super::Spawner::spawn) to return the error. | ||||
|     pub fn spawn(&'static self, future: impl FnOnce() -> F) -> SpawnToken<impl Sized> { | ||||
|         for task in &self.pool { | ||||
|             if task.spawn_mark_used() { | ||||
|                 return unsafe { SpawnToken::<F>::new(task.spawn_initialize(future)) }; | ||||
|         let task = self.pool.iter().find_map(AvailableTask::claim); | ||||
|         match task { | ||||
|             Some(task) => { | ||||
|                 let task = task.initialize(future); | ||||
|                 unsafe { SpawnToken::<F>::new(task) } | ||||
|             } | ||||
|             None => SpawnToken::new_failed(), | ||||
|         } | ||||
|  | ||||
|         SpawnToken::<F>::new_failed() | ||||
|     } | ||||
|  | ||||
|     /// Like spawn(), but allows the task to be send-spawned if the args are Send even if | ||||
| @@ -235,39 +277,71 @@ impl<F: Future + 'static, const N: usize> TaskPool<F, N> { | ||||
|         // This ONLY holds for `async fn` futures. The other `spawn` methods can be called directly | ||||
|         // by the user, with arbitrary hand-implemented futures. This is why these return `SpawnToken<F>`. | ||||
|  | ||||
|         for task in &self.pool { | ||||
|             if task.spawn_mark_used() { | ||||
|                 return SpawnToken::<FutFn>::new(task.spawn_initialize(future)); | ||||
|         let task = self.pool.iter().find_map(AvailableTask::claim); | ||||
|         match task { | ||||
|             Some(task) => { | ||||
|                 let task = task.initialize(future); | ||||
|                 unsafe { SpawnToken::<FutFn>::new(task) } | ||||
|             } | ||||
|             None => SpawnToken::new_failed(), | ||||
|         } | ||||
|  | ||||
|         SpawnToken::<FutFn>::new_failed() | ||||
|     } | ||||
| } | ||||
|  | ||||
| /// Raw executor. | ||||
| #[derive(Clone, Copy)] | ||||
| pub(crate) enum PenderInner { | ||||
|     #[cfg(feature = "executor-thread")] | ||||
|     Thread(crate::arch::ThreadPender), | ||||
|     #[cfg(feature = "executor-interrupt")] | ||||
|     Interrupt(crate::arch::InterruptPender), | ||||
|     #[cfg(feature = "pender-callback")] | ||||
|     Callback { func: fn(*mut ()), context: *mut () }, | ||||
| } | ||||
|  | ||||
| unsafe impl Send for PenderInner {} | ||||
| unsafe impl Sync for PenderInner {} | ||||
|  | ||||
| /// Platform/architecture-specific action executed when an executor has pending work. | ||||
| /// | ||||
| /// This is the core of the Embassy executor. It is low-level, requiring manual | ||||
| /// handling of wakeups and task polling. If you can, prefer using one of the | ||||
| /// [higher level executors](crate::Executor). | ||||
| /// When a task within an executor is woken, the `Pender` is called. This does a | ||||
| /// platform/architecture-specific action to signal there is pending work in the executor. | ||||
| /// When this happens, you must arrange for [`Executor::poll`] to be called. | ||||
| /// | ||||
| /// The raw executor leaves it up to you to handle wakeups and scheduling: | ||||
| /// | ||||
| /// - To get the executor to do work, call `poll()`. This will poll all queued tasks (all tasks | ||||
| ///   that "want to run"). | ||||
| /// - You must supply a `signal_fn`. The executor will call it to notify you it has work | ||||
| ///   to do. You must arrange for `poll()` to be called as soon as possible. | ||||
| /// | ||||
| /// `signal_fn` can be called from *any* context: any thread, any interrupt priority | ||||
| /// level, etc. It may be called synchronously from any `Executor` method call as well. | ||||
| /// You must deal with this correctly. | ||||
| /// | ||||
| /// In particular, you must NOT call `poll` directly from `signal_fn`, as this violates | ||||
| /// the requirement for `poll` to not be called reentrantly. | ||||
| pub struct Executor { | ||||
| /// You can think of it as a waker, but for the whole executor. | ||||
| pub struct Pender(pub(crate) PenderInner); | ||||
|  | ||||
| impl Pender { | ||||
|     /// Create a `Pender` that will call an arbitrary function pointer. | ||||
|     /// | ||||
|     /// # Arguments | ||||
|     /// | ||||
|     /// - `func`: The function pointer to call. | ||||
|     /// - `context`: Opaque context pointer, that will be passed to the function pointer. | ||||
|     #[cfg(feature = "pender-callback")] | ||||
|     pub fn new_from_callback(func: fn(*mut ()), context: *mut ()) -> Self { | ||||
|         Self(PenderInner::Callback { | ||||
|             func, | ||||
|             context: context.into(), | ||||
|         }) | ||||
|     } | ||||
| } | ||||
|  | ||||
| impl Pender { | ||||
|     pub(crate) fn pend(&self) { | ||||
|         match self.0 { | ||||
|             #[cfg(feature = "executor-thread")] | ||||
|             PenderInner::Thread(x) => x.pend(), | ||||
|             #[cfg(feature = "executor-interrupt")] | ||||
|             PenderInner::Interrupt(x) => x.pend(), | ||||
|             #[cfg(feature = "pender-callback")] | ||||
|             PenderInner::Callback { func, context } => func(context), | ||||
|         } | ||||
|     } | ||||
| } | ||||
|  | ||||
| pub(crate) struct SyncExecutor { | ||||
|     run_queue: RunQueue, | ||||
|     signal_fn: fn(*mut ()), | ||||
|     signal_ctx: *mut (), | ||||
|     pender: Pender, | ||||
|  | ||||
|     #[cfg(feature = "integrated-timers")] | ||||
|     pub(crate) timer_queue: timer_queue::TimerQueue, | ||||
| @@ -275,23 +349,14 @@ pub struct Executor { | ||||
|     alarm: AlarmHandle, | ||||
| } | ||||
|  | ||||
| impl Executor { | ||||
|     /// Create a new executor. | ||||
|     /// | ||||
|     /// When the executor has work to do, it will call `signal_fn` with | ||||
|     /// `signal_ctx` as argument. | ||||
|     /// | ||||
|     /// See [`Executor`] docs for details on `signal_fn`. | ||||
|     pub fn new(signal_fn: fn(*mut ()), signal_ctx: *mut ()) -> Self { | ||||
| impl SyncExecutor { | ||||
|     pub(crate) fn new(pender: Pender) -> Self { | ||||
|         #[cfg(feature = "integrated-timers")] | ||||
|         let alarm = unsafe { unwrap!(driver::allocate_alarm()) }; | ||||
|         #[cfg(feature = "integrated-timers")] | ||||
|         driver::set_alarm_callback(alarm, signal_fn, signal_ctx); | ||||
|  | ||||
|         Self { | ||||
|             run_queue: RunQueue::new(), | ||||
|             signal_fn, | ||||
|             signal_ctx, | ||||
|             pender, | ||||
|  | ||||
|             #[cfg(feature = "integrated-timers")] | ||||
|             timer_queue: timer_queue::TimerQueue::new(), | ||||
| @@ -307,12 +372,133 @@ impl Executor { | ||||
|     /// - `task` must be set up to run in this executor. | ||||
|     /// - `task` must NOT be already enqueued (in this executor or another one). | ||||
|     #[inline(always)] | ||||
|     unsafe fn enqueue(&self, cs: CriticalSection, task: NonNull<TaskHeader>) { | ||||
|     unsafe fn enqueue(&self, task: TaskRef) { | ||||
|         #[cfg(feature = "rtos-trace")] | ||||
|         trace::task_ready_begin(task.as_ptr() as u32); | ||||
|  | ||||
|         if self.run_queue.enqueue(cs, task) { | ||||
|             (self.signal_fn)(self.signal_ctx) | ||||
|         if self.run_queue.enqueue(task) { | ||||
|             self.pender.pend(); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     #[cfg(feature = "integrated-timers")] | ||||
|     fn alarm_callback(ctx: *mut ()) { | ||||
|         let this: &Self = unsafe { &*(ctx as *const Self) }; | ||||
|         this.pender.pend(); | ||||
|     } | ||||
|  | ||||
|     pub(super) unsafe fn spawn(&'static self, task: TaskRef) { | ||||
|         task.header().executor.set(Some(self)); | ||||
|  | ||||
|         #[cfg(feature = "rtos-trace")] | ||||
|         trace::task_new(task.as_ptr() as u32); | ||||
|  | ||||
|         self.enqueue(task); | ||||
|     } | ||||
|  | ||||
|     /// # Safety | ||||
|     /// | ||||
|     /// Same as [`Executor::poll`], plus you must only call this on the thread this executor was created. | ||||
|     pub(crate) unsafe fn poll(&'static self) { | ||||
|         #[cfg(feature = "integrated-timers")] | ||||
|         driver::set_alarm_callback(self.alarm, Self::alarm_callback, self as *const _ as *mut ()); | ||||
|  | ||||
|         #[allow(clippy::never_loop)] | ||||
|         loop { | ||||
|             #[cfg(feature = "integrated-timers")] | ||||
|             self.timer_queue.dequeue_expired(Instant::now(), |task| wake_task(task)); | ||||
|  | ||||
|             self.run_queue.dequeue_all(|p| { | ||||
|                 let task = p.header(); | ||||
|  | ||||
|                 #[cfg(feature = "integrated-timers")] | ||||
|                 task.expires_at.set(Instant::MAX); | ||||
|  | ||||
|                 let state = task.state.fetch_and(!STATE_RUN_QUEUED, Ordering::AcqRel); | ||||
|                 if state & STATE_SPAWNED == 0 { | ||||
|                     // If task is not running, ignore it. This can happen in the following scenario: | ||||
|                     //   - Task gets dequeued, poll starts | ||||
|                     //   - While task is being polled, it gets woken. It gets placed in the queue. | ||||
|                     //   - Task poll finishes, returning done=true | ||||
|                     //   - RUNNING bit is cleared, but the task is already in the queue. | ||||
|                     return; | ||||
|                 } | ||||
|  | ||||
|                 #[cfg(feature = "rtos-trace")] | ||||
|                 trace::task_exec_begin(p.as_ptr() as u32); | ||||
|  | ||||
|                 // Run the task | ||||
|                 task.poll_fn.get().unwrap_unchecked()(p); | ||||
|  | ||||
|                 #[cfg(feature = "rtos-trace")] | ||||
|                 trace::task_exec_end(); | ||||
|  | ||||
|                 // Enqueue or update into timer_queue | ||||
|                 #[cfg(feature = "integrated-timers")] | ||||
|                 self.timer_queue.update(p); | ||||
|             }); | ||||
|  | ||||
|             #[cfg(feature = "integrated-timers")] | ||||
|             { | ||||
|                 // If this is already in the past, set_alarm might return false | ||||
|                 // In that case do another poll loop iteration. | ||||
|                 let next_expiration = self.timer_queue.next_expiration(); | ||||
|                 if driver::set_alarm(self.alarm, next_expiration.as_ticks()) { | ||||
|                     break; | ||||
|                 } | ||||
|             } | ||||
|  | ||||
|             #[cfg(not(feature = "integrated-timers"))] | ||||
|             { | ||||
|                 break; | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         #[cfg(feature = "rtos-trace")] | ||||
|         trace::system_idle(); | ||||
|     } | ||||
| } | ||||
|  | ||||
| /// Raw executor. | ||||
| /// | ||||
| /// This is the core of the Embassy executor. It is low-level, requiring manual | ||||
| /// handling of wakeups and task polling. If you can, prefer using one of the | ||||
| /// [higher level executors](crate::Executor). | ||||
| /// | ||||
| /// The raw executor leaves it up to you to handle wakeups and scheduling: | ||||
| /// | ||||
| /// - To get the executor to do work, call `poll()`. This will poll all queued tasks (all tasks | ||||
| ///   that "want to run"). | ||||
| /// - You must supply a [`Pender`]. The executor will call it to notify you it has work | ||||
| ///   to do. You must arrange for `poll()` to be called as soon as possible. | ||||
| /// | ||||
| /// The [`Pender`] can be called from *any* context: any thread, any interrupt priority | ||||
| /// level, etc. It may be called synchronously from any `Executor` method call as well. | ||||
| /// You must deal with this correctly. | ||||
| /// | ||||
| /// In particular, you must NOT call `poll` directly from the pender callback, as this violates | ||||
| /// the requirement for `poll` to not be called reentrantly. | ||||
| #[repr(transparent)] | ||||
| pub struct Executor { | ||||
|     pub(crate) inner: SyncExecutor, | ||||
|  | ||||
|     _not_sync: PhantomData<*mut ()>, | ||||
| } | ||||
|  | ||||
| impl Executor { | ||||
|     pub(crate) unsafe fn wrap(inner: &SyncExecutor) -> &Self { | ||||
|         mem::transmute(inner) | ||||
|     } | ||||
|  | ||||
|     /// Create a new executor. | ||||
|     /// | ||||
|     /// When the executor has work to do, it will call the [`Pender`]. | ||||
|     /// | ||||
|     /// See [`Executor`] docs for details on `Pender`. | ||||
|     pub fn new(pender: Pender) -> Self { | ||||
|         Self { | ||||
|             inner: SyncExecutor::new(pender), | ||||
|             _not_sync: PhantomData, | ||||
|         } | ||||
|     } | ||||
|  | ||||
| @@ -325,15 +511,8 @@ impl Executor { | ||||
|     /// It is OK to use `unsafe` to call this from a thread that's not the executor thread. | ||||
|     /// In this case, the task's Future must be Send. This is because this is effectively | ||||
|     /// sending the task to the executor thread. | ||||
|     pub(super) unsafe fn spawn(&'static self, task: NonNull<TaskHeader>) { | ||||
|         task.as_ref().executor.set(self); | ||||
|  | ||||
|         #[cfg(feature = "rtos-trace")] | ||||
|         trace::task_new(task.as_ptr() as u32); | ||||
|  | ||||
|         critical_section::with(|cs| { | ||||
|             self.enqueue(cs, task); | ||||
|         }) | ||||
|     pub(super) unsafe fn spawn(&'static self, task: TaskRef) { | ||||
|         self.inner.spawn(task) | ||||
|     } | ||||
|  | ||||
|     /// Poll all queued tasks in this executor. | ||||
| @@ -341,63 +520,20 @@ impl Executor { | ||||
|     /// This loops over all tasks that are queued to be polled (i.e. they're | ||||
|     /// freshly spawned or they've been woken). Other tasks are not polled. | ||||
|     /// | ||||
|     /// You must call `poll` after receiving a call to `signal_fn`. It is OK | ||||
|     /// to call `poll` even when not requested by `signal_fn`, but it wastes | ||||
|     /// You must call `poll` after receiving a call to the [`Pender`]. It is OK | ||||
|     /// to call `poll` even when not requested by the `Pender`, but it wastes | ||||
|     /// energy. | ||||
|     /// | ||||
|     /// # Safety | ||||
|     /// | ||||
|     /// You must NOT call `poll` reentrantly on the same executor. | ||||
|     /// | ||||
|     /// In particular, note that `poll` may call `signal_fn` synchronously. Therefore, you | ||||
|     /// must NOT directly call `poll()` from your `signal_fn`. Instead, `signal_fn` has to | ||||
|     /// In particular, note that `poll` may call the `Pender` synchronously. Therefore, you | ||||
|     /// must NOT directly call `poll()` from the `Pender` callback. Instead, the callback has to | ||||
|     /// somehow schedule for `poll()` to be called later, at a time you know for sure there's | ||||
|     /// no `poll()` already running. | ||||
|     pub unsafe fn poll(&'static self) { | ||||
|         #[cfg(feature = "integrated-timers")] | ||||
|         self.timer_queue.dequeue_expired(Instant::now(), |task| wake_task(task)); | ||||
|  | ||||
|         self.run_queue.dequeue_all(|p| { | ||||
|             let task = p.as_ref(); | ||||
|  | ||||
|             #[cfg(feature = "integrated-timers")] | ||||
|             task.expires_at.set(Instant::MAX); | ||||
|  | ||||
|             let state = task.state.fetch_and(!STATE_RUN_QUEUED, Ordering::AcqRel); | ||||
|             if state & STATE_SPAWNED == 0 { | ||||
|                 // If task is not running, ignore it. This can happen in the following scenario: | ||||
|                 //   - Task gets dequeued, poll starts | ||||
|                 //   - While task is being polled, it gets woken. It gets placed in the queue. | ||||
|                 //   - Task poll finishes, returning done=true | ||||
|                 //   - RUNNING bit is cleared, but the task is already in the queue. | ||||
|                 return; | ||||
|             } | ||||
|  | ||||
|             #[cfg(feature = "rtos-trace")] | ||||
|             trace::task_exec_begin(p.as_ptr() as u32); | ||||
|  | ||||
|             // Run the task | ||||
|             task.poll_fn.read()(p as _); | ||||
|  | ||||
|             #[cfg(feature = "rtos-trace")] | ||||
|             trace::task_exec_end(); | ||||
|  | ||||
|             // Enqueue or update into timer_queue | ||||
|             #[cfg(feature = "integrated-timers")] | ||||
|             self.timer_queue.update(p); | ||||
|         }); | ||||
|  | ||||
|         #[cfg(feature = "integrated-timers")] | ||||
|         { | ||||
|             // If this is already in the past, set_alarm will immediately trigger the alarm. | ||||
|             // This will cause `signal_fn` to be called, which will cause `poll()` to be called again, | ||||
|             // so we immediately do another poll loop iteration. | ||||
|             let next_expiration = self.timer_queue.next_expiration(); | ||||
|             driver::set_alarm(self.alarm, next_expiration.as_ticks()); | ||||
|         } | ||||
|  | ||||
|         #[cfg(feature = "rtos-trace")] | ||||
|         trace::system_idle(); | ||||
|         self.inner.poll() | ||||
|     } | ||||
|  | ||||
|     /// Get a spawner that spawns tasks in this executor. | ||||
| @@ -409,41 +545,49 @@ impl Executor { | ||||
|     } | ||||
| } | ||||
|  | ||||
| /// Wake a task by raw pointer. | ||||
| /// Wake a task by `TaskRef`. | ||||
| /// | ||||
| /// You can obtain task pointers from `Waker`s using [`task_from_waker`]. | ||||
| /// | ||||
| /// # Safety | ||||
| /// | ||||
| /// `task` must be a valid task pointer obtained from [`task_from_waker`]. | ||||
| pub unsafe fn wake_task(task: NonNull<TaskHeader>) { | ||||
|     critical_section::with(|cs| { | ||||
|         let header = task.as_ref(); | ||||
|         let state = header.state.load(Ordering::Relaxed); | ||||
| /// You can obtain a `TaskRef` from a `Waker` using [`task_from_waker`]. | ||||
| pub fn wake_task(task: TaskRef) { | ||||
|     let header = task.header(); | ||||
|  | ||||
|     let res = header.state.fetch_update(Ordering::SeqCst, Ordering::SeqCst, |state| { | ||||
|         // If already scheduled, or if not started, | ||||
|         if (state & STATE_RUN_QUEUED != 0) || (state & STATE_SPAWNED == 0) { | ||||
|             return; | ||||
|             None | ||||
|         } else { | ||||
|             // Mark it as scheduled | ||||
|             Some(state | STATE_RUN_QUEUED) | ||||
|         } | ||||
|     }); | ||||
|  | ||||
|         // Mark it as scheduled | ||||
|         header.state.store(state | STATE_RUN_QUEUED, Ordering::Relaxed); | ||||
|  | ||||
|     if res.is_ok() { | ||||
|         // We have just marked the task as scheduled, so enqueue it. | ||||
|         let executor = &*header.executor.get(); | ||||
|         executor.enqueue(cs, task); | ||||
|     }) | ||||
|         unsafe { | ||||
|             let executor = header.executor.get().unwrap_unchecked(); | ||||
|             executor.enqueue(task); | ||||
|         } | ||||
|     } | ||||
| } | ||||
|  | ||||
| #[cfg(feature = "integrated-timers")] | ||||
| #[no_mangle] | ||||
| unsafe fn _embassy_time_schedule_wake(at: Instant, waker: &core::task::Waker) { | ||||
|     let task = waker::task_from_waker(waker); | ||||
|     let task = task.as_ref(); | ||||
|     let expires_at = task.expires_at.get(); | ||||
|     task.expires_at.set(expires_at.min(at)); | ||||
| struct TimerQueue; | ||||
|  | ||||
| #[cfg(feature = "integrated-timers")] | ||||
| impl embassy_time::queue::TimerQueue for TimerQueue { | ||||
|     fn schedule_wake(&'static self, at: Instant, waker: &core::task::Waker) { | ||||
|         let task = waker::task_from_waker(waker); | ||||
|         let task = task.header(); | ||||
|         unsafe { | ||||
|             let expires_at = task.expires_at.get(); | ||||
|             task.expires_at.set(expires_at.min(at)); | ||||
|         } | ||||
|     } | ||||
| } | ||||
|  | ||||
| #[cfg(feature = "integrated-timers")] | ||||
| embassy_time::timer_queue_impl!(static TIMER_QUEUE: TimerQueue = TimerQueue); | ||||
|  | ||||
| #[cfg(feature = "rtos-trace")] | ||||
| impl rtos_trace::RtosTraceOSCallbacks for Executor { | ||||
|     fn task_list() { | ||||
|   | ||||
| @@ -2,18 +2,18 @@ use core::ptr; | ||||
| use core::ptr::NonNull; | ||||
|  | ||||
| use atomic_polyfill::{AtomicPtr, Ordering}; | ||||
| use critical_section::CriticalSection; | ||||
|  | ||||
| use super::TaskHeader; | ||||
| use super::{TaskHeader, TaskRef}; | ||||
| use crate::raw::util::SyncUnsafeCell; | ||||
|  | ||||
| pub(crate) struct RunQueueItem { | ||||
|     next: AtomicPtr<TaskHeader>, | ||||
|     next: SyncUnsafeCell<Option<TaskRef>>, | ||||
| } | ||||
|  | ||||
| impl RunQueueItem { | ||||
|     pub const fn new() -> Self { | ||||
|         Self { | ||||
|             next: AtomicPtr::new(ptr::null_mut()), | ||||
|             next: SyncUnsafeCell::new(None), | ||||
|         } | ||||
|     } | ||||
| } | ||||
| @@ -46,29 +46,43 @@ impl RunQueue { | ||||
|     /// | ||||
|     /// `item` must NOT be already enqueued in any queue. | ||||
|     #[inline(always)] | ||||
|     pub(crate) unsafe fn enqueue(&self, _cs: CriticalSection, task: NonNull<TaskHeader>) -> bool { | ||||
|         let prev = self.head.load(Ordering::Relaxed); | ||||
|         task.as_ref().run_queue_item.next.store(prev, Ordering::Relaxed); | ||||
|         self.head.store(task.as_ptr(), Ordering::Relaxed); | ||||
|         prev.is_null() | ||||
|     pub(crate) unsafe fn enqueue(&self, task: TaskRef) -> bool { | ||||
|         let mut was_empty = false; | ||||
|  | ||||
|         self.head | ||||
|             .fetch_update(Ordering::SeqCst, Ordering::SeqCst, |prev| { | ||||
|                 was_empty = prev.is_null(); | ||||
|                 unsafe { | ||||
|                     // safety: the pointer is either null or valid | ||||
|                     let prev = NonNull::new(prev).map(|ptr| TaskRef::from_ptr(ptr.as_ptr())); | ||||
|                     // safety: there are no concurrent accesses to `next` | ||||
|                     task.header().run_queue_item.next.set(prev); | ||||
|                 } | ||||
|                 Some(task.as_ptr() as *mut _) | ||||
|             }) | ||||
|             .ok(); | ||||
|  | ||||
|         was_empty | ||||
|     } | ||||
|  | ||||
|     /// Empty the queue, then call `on_task` for each task that was in the queue. | ||||
|     /// NOTE: It is OK for `on_task` to enqueue more tasks. In this case they're left in the queue | ||||
|     /// and will be processed by the *next* call to `dequeue_all`, *not* the current one. | ||||
|     pub(crate) fn dequeue_all(&self, on_task: impl Fn(NonNull<TaskHeader>)) { | ||||
|     pub(crate) fn dequeue_all(&self, on_task: impl Fn(TaskRef)) { | ||||
|         // Atomically empty the queue. | ||||
|         let mut ptr = self.head.swap(ptr::null_mut(), Ordering::AcqRel); | ||||
|         let ptr = self.head.swap(ptr::null_mut(), Ordering::AcqRel); | ||||
|  | ||||
|         // safety: the pointer is either null or valid | ||||
|         let mut next = unsafe { NonNull::new(ptr).map(|ptr| TaskRef::from_ptr(ptr.as_ptr())) }; | ||||
|  | ||||
|         // Iterate the linked list of tasks that were previously in the queue. | ||||
|         while let Some(task) = NonNull::new(ptr) { | ||||
|         while let Some(task) = next { | ||||
|             // If the task re-enqueues itself, the `next` pointer will get overwritten. | ||||
|             // Therefore, first read the next pointer, and only then process the task. | ||||
|             let next = unsafe { task.as_ref() }.run_queue_item.next.load(Ordering::Relaxed); | ||||
|             // safety: there are no concurrent accesses to `next` | ||||
|             next = unsafe { task.header().run_queue_item.next.get() }; | ||||
|  | ||||
|             on_task(task); | ||||
|  | ||||
|             ptr = next | ||||
|         } | ||||
|     } | ||||
| } | ||||
|   | ||||
| @@ -1,45 +1,43 @@ | ||||
| use core::cell::Cell; | ||||
| use core::cmp::min; | ||||
| use core::ptr; | ||||
| use core::ptr::NonNull; | ||||
|  | ||||
| use atomic_polyfill::Ordering; | ||||
| use embassy_time::Instant; | ||||
|  | ||||
| use super::{TaskHeader, STATE_TIMER_QUEUED}; | ||||
| use super::{TaskRef, STATE_TIMER_QUEUED}; | ||||
| use crate::raw::util::SyncUnsafeCell; | ||||
|  | ||||
| pub(crate) struct TimerQueueItem { | ||||
|     next: Cell<*mut TaskHeader>, | ||||
|     next: SyncUnsafeCell<Option<TaskRef>>, | ||||
| } | ||||
|  | ||||
| impl TimerQueueItem { | ||||
|     pub const fn new() -> Self { | ||||
|         Self { | ||||
|             next: Cell::new(ptr::null_mut()), | ||||
|             next: SyncUnsafeCell::new(None), | ||||
|         } | ||||
|     } | ||||
| } | ||||
|  | ||||
| pub(crate) struct TimerQueue { | ||||
|     head: Cell<*mut TaskHeader>, | ||||
|     head: SyncUnsafeCell<Option<TaskRef>>, | ||||
| } | ||||
|  | ||||
| impl TimerQueue { | ||||
|     pub const fn new() -> Self { | ||||
|         Self { | ||||
|             head: Cell::new(ptr::null_mut()), | ||||
|             head: SyncUnsafeCell::new(None), | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     pub(crate) unsafe fn update(&self, p: NonNull<TaskHeader>) { | ||||
|         let task = p.as_ref(); | ||||
|     pub(crate) unsafe fn update(&self, p: TaskRef) { | ||||
|         let task = p.header(); | ||||
|         if task.expires_at.get() != Instant::MAX { | ||||
|             let old_state = task.state.fetch_or(STATE_TIMER_QUEUED, Ordering::AcqRel); | ||||
|             let is_new = old_state & STATE_TIMER_QUEUED == 0; | ||||
|  | ||||
|             if is_new { | ||||
|                 task.timer_queue_item.next.set(self.head.get()); | ||||
|                 self.head.set(p.as_ptr()); | ||||
|                 self.head.set(Some(p)); | ||||
|             } | ||||
|         } | ||||
|     } | ||||
| @@ -47,7 +45,7 @@ impl TimerQueue { | ||||
|     pub(crate) unsafe fn next_expiration(&self) -> Instant { | ||||
|         let mut res = Instant::MAX; | ||||
|         self.retain(|p| { | ||||
|             let task = p.as_ref(); | ||||
|             let task = p.header(); | ||||
|             let expires = task.expires_at.get(); | ||||
|             res = min(res, expires); | ||||
|             expires != Instant::MAX | ||||
| @@ -55,9 +53,9 @@ impl TimerQueue { | ||||
|         res | ||||
|     } | ||||
|  | ||||
|     pub(crate) unsafe fn dequeue_expired(&self, now: Instant, on_task: impl Fn(NonNull<TaskHeader>)) { | ||||
|     pub(crate) unsafe fn dequeue_expired(&self, now: Instant, on_task: impl Fn(TaskRef)) { | ||||
|         self.retain(|p| { | ||||
|             let task = p.as_ref(); | ||||
|             let task = p.header(); | ||||
|             if task.expires_at.get() <= now { | ||||
|                 on_task(p); | ||||
|                 false | ||||
| @@ -67,11 +65,10 @@ impl TimerQueue { | ||||
|         }); | ||||
|     } | ||||
|  | ||||
|     pub(crate) unsafe fn retain(&self, mut f: impl FnMut(NonNull<TaskHeader>) -> bool) { | ||||
|     pub(crate) unsafe fn retain(&self, mut f: impl FnMut(TaskRef) -> bool) { | ||||
|         let mut prev = &self.head; | ||||
|         while !prev.get().is_null() { | ||||
|             let p = NonNull::new_unchecked(prev.get()); | ||||
|             let task = &*p.as_ptr(); | ||||
|         while let Some(p) = prev.get() { | ||||
|             let task = p.header(); | ||||
|             if f(p) { | ||||
|                 // Skip to next | ||||
|                 prev = &task.timer_queue_item.next; | ||||
|   | ||||
| @@ -26,8 +26,31 @@ impl<T> UninitCell<T> { | ||||
|     } | ||||
| } | ||||
|  | ||||
| impl<T: Copy> UninitCell<T> { | ||||
|     pub unsafe fn read(&self) -> T { | ||||
|         ptr::read(self.as_mut_ptr()) | ||||
| unsafe impl<T> Sync for UninitCell<T> {} | ||||
|  | ||||
| #[repr(transparent)] | ||||
| pub struct SyncUnsafeCell<T> { | ||||
|     value: UnsafeCell<T>, | ||||
| } | ||||
|  | ||||
| unsafe impl<T: Sync> Sync for SyncUnsafeCell<T> {} | ||||
|  | ||||
| impl<T> SyncUnsafeCell<T> { | ||||
|     #[inline] | ||||
|     pub const fn new(value: T) -> Self { | ||||
|         Self { | ||||
|             value: UnsafeCell::new(value), | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     pub unsafe fn set(&self, value: T) { | ||||
|         *self.value.get() = value; | ||||
|     } | ||||
|  | ||||
|     pub unsafe fn get(&self) -> T | ||||
|     where | ||||
|         T: Copy, | ||||
|     { | ||||
|         *self.value.get() | ||||
|     } | ||||
| } | ||||
|   | ||||
| @@ -1,8 +1,7 @@ | ||||
| use core::mem; | ||||
| use core::ptr::NonNull; | ||||
| use core::task::{RawWaker, RawWakerVTable, Waker}; | ||||
|  | ||||
| use super::{wake_task, TaskHeader}; | ||||
| use super::{wake_task, TaskHeader, TaskRef}; | ||||
|  | ||||
| const VTABLE: RawWakerVTable = RawWakerVTable::new(clone, wake, wake, drop); | ||||
|  | ||||
| @@ -11,14 +10,14 @@ unsafe fn clone(p: *const ()) -> RawWaker { | ||||
| } | ||||
|  | ||||
| unsafe fn wake(p: *const ()) { | ||||
|     wake_task(NonNull::new_unchecked(p as *mut TaskHeader)) | ||||
|     wake_task(TaskRef::from_ptr(p as *const TaskHeader)) | ||||
| } | ||||
|  | ||||
| unsafe fn drop(_: *const ()) { | ||||
|     // nop | ||||
| } | ||||
|  | ||||
| pub(crate) unsafe fn from_task(p: NonNull<TaskHeader>) -> Waker { | ||||
| pub(crate) unsafe fn from_task(p: TaskRef) -> Waker { | ||||
|     Waker::from_raw(RawWaker::new(p.as_ptr() as _, &VTABLE)) | ||||
| } | ||||
|  | ||||
| @@ -33,7 +32,7 @@ pub(crate) unsafe fn from_task(p: NonNull<TaskHeader>) -> Waker { | ||||
| /// # Panics | ||||
| /// | ||||
| /// Panics if the waker is not created by the Embassy executor. | ||||
| pub fn task_from_waker(waker: &Waker) -> NonNull<TaskHeader> { | ||||
| pub fn task_from_waker(waker: &Waker) -> TaskRef { | ||||
|     // safety: OK because WakerHack has the same layout as Waker. | ||||
|     // This is not really guaranteed because the structs are `repr(Rust)`, it is | ||||
|     // indeed the case in the current implementation. | ||||
| @@ -43,8 +42,8 @@ pub fn task_from_waker(waker: &Waker) -> NonNull<TaskHeader> { | ||||
|         panic!("Found waker not created by the Embassy executor. `embassy_time::Timer` only works with the Embassy executor.") | ||||
|     } | ||||
|  | ||||
|     // safety: we never create a waker with a null data pointer. | ||||
|     unsafe { NonNull::new_unchecked(hack.data as *mut TaskHeader) } | ||||
|     // safety: our wakers are always created with `TaskRef::as_ptr` | ||||
|     unsafe { TaskRef::from_ptr(hack.data as *const TaskHeader) } | ||||
| } | ||||
|  | ||||
| struct WakerHack { | ||||
|   | ||||
							
								
								
									
										34
									
								
								embassy-executor/src/raw/waker_turbo.rs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										34
									
								
								embassy-executor/src/raw/waker_turbo.rs
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,34 @@ | ||||
| use core::ptr::NonNull; | ||||
| use core::task::Waker; | ||||
|  | ||||
| use super::{wake_task, TaskHeader, TaskRef}; | ||||
|  | ||||
| pub(crate) unsafe fn from_task(p: TaskRef) -> Waker { | ||||
|     Waker::from_turbo_ptr(NonNull::new_unchecked(p.as_ptr() as _)) | ||||
| } | ||||
|  | ||||
| /// Get a task pointer from a waker. | ||||
| /// | ||||
| /// This can be used as an optimization in wait queues to store task pointers | ||||
| /// (1 word) instead of full Wakers (2 words). This saves a bit of RAM and helps | ||||
| /// avoid dynamic dispatch. | ||||
| /// | ||||
| /// You can use the returned task pointer to wake the task with [`wake_task`](super::wake_task). | ||||
| /// | ||||
| /// # Panics | ||||
| /// | ||||
| /// Panics if the waker is not created by the Embassy executor. | ||||
| pub fn task_from_waker(waker: &Waker) -> TaskRef { | ||||
|     let ptr = waker.as_turbo_ptr().as_ptr(); | ||||
|  | ||||
|     // safety: our wakers are always created with `TaskRef::as_ptr` | ||||
|     unsafe { TaskRef::from_ptr(ptr as *const TaskHeader) } | ||||
| } | ||||
|  | ||||
| #[inline(never)] | ||||
| #[no_mangle] | ||||
| fn _turbo_wake(ptr: NonNull<()>) { | ||||
|     // safety: our wakers are always created with `TaskRef::as_ptr` | ||||
|     let task = unsafe { TaskRef::from_ptr(ptr.as_ptr() as *const TaskHeader) }; | ||||
|     wake_task(task) | ||||
| } | ||||
| @@ -1,10 +1,8 @@ | ||||
| use core::future::poll_fn; | ||||
| use core::marker::PhantomData; | ||||
| use core::mem; | ||||
| use core::ptr::NonNull; | ||||
| use core::task::Poll; | ||||
|  | ||||
| use futures_util::future::poll_fn; | ||||
|  | ||||
| use super::raw; | ||||
|  | ||||
| /// Token to spawn a newly-created task in an executor. | ||||
| @@ -23,12 +21,12 @@ use super::raw; | ||||
| /// Once you've invoked a task function and obtained a SpawnToken, you *must* spawn it. | ||||
| #[must_use = "Calling a task function does nothing on its own. You must spawn the returned SpawnToken, typically with Spawner::spawn()"] | ||||
| pub struct SpawnToken<S> { | ||||
|     raw_task: Option<NonNull<raw::TaskHeader>>, | ||||
|     raw_task: Option<raw::TaskRef>, | ||||
|     phantom: PhantomData<*mut S>, | ||||
| } | ||||
|  | ||||
| impl<S> SpawnToken<S> { | ||||
|     pub(crate) unsafe fn new(raw_task: NonNull<raw::TaskHeader>) -> Self { | ||||
|     pub(crate) unsafe fn new(raw_task: raw::TaskRef) -> Self { | ||||
|         Self { | ||||
|             raw_task: Some(raw_task), | ||||
|             phantom: PhantomData, | ||||
| @@ -91,10 +89,11 @@ impl Spawner { | ||||
|     /// | ||||
|     /// Panics if the current executor is not an Embassy executor. | ||||
|     pub async fn for_current_executor() -> Self { | ||||
|         poll_fn(|cx| unsafe { | ||||
|         poll_fn(|cx| { | ||||
|             let task = raw::task_from_waker(cx.waker()); | ||||
|             let executor = (*task.as_ptr()).executor.get(); | ||||
|             Poll::Ready(Self::new(&*executor)) | ||||
|             let executor = unsafe { task.header().executor.get().unwrap_unchecked() }; | ||||
|             let executor = unsafe { raw::Executor::wrap(executor) }; | ||||
|             Poll::Ready(Self::new(executor)) | ||||
|         }) | ||||
|         .await | ||||
|     } | ||||
| @@ -132,9 +131,7 @@ impl Spawner { | ||||
|     /// spawner to other threads, but the spawner loses the ability to spawn | ||||
|     /// non-Send tasks. | ||||
|     pub fn make_send(&self) -> SendSpawner { | ||||
|         SendSpawner { | ||||
|             executor: self.executor, | ||||
|         } | ||||
|         SendSpawner::new(&self.executor.inner) | ||||
|     } | ||||
| } | ||||
|  | ||||
| @@ -147,14 +144,11 @@ impl Spawner { | ||||
| /// If you want to spawn non-Send tasks, use [Spawner]. | ||||
| #[derive(Copy, Clone)] | ||||
| pub struct SendSpawner { | ||||
|     executor: &'static raw::Executor, | ||||
|     executor: &'static raw::SyncExecutor, | ||||
| } | ||||
|  | ||||
| unsafe impl Send for SendSpawner {} | ||||
| unsafe impl Sync for SendSpawner {} | ||||
|  | ||||
| impl SendSpawner { | ||||
|     pub(crate) fn new(executor: &'static raw::Executor) -> Self { | ||||
|     pub(crate) fn new(executor: &'static raw::SyncExecutor) -> Self { | ||||
|         Self { executor } | ||||
|     } | ||||
|  | ||||
| @@ -167,10 +161,10 @@ impl SendSpawner { | ||||
|     /// | ||||
|     /// Panics if the current executor is not an Embassy executor. | ||||
|     pub async fn for_current_executor() -> Self { | ||||
|         poll_fn(|cx| unsafe { | ||||
|         poll_fn(|cx| { | ||||
|             let task = raw::task_from_waker(cx.waker()); | ||||
|             let executor = (*task.as_ptr()).executor.get(); | ||||
|             Poll::Ready(Self::new(&*executor)) | ||||
|             let executor = unsafe { task.header().executor.get().unwrap_unchecked() }; | ||||
|             Poll::Ready(Self::new(executor)) | ||||
|         }) | ||||
|         .await | ||||
|     } | ||||
|   | ||||
| @@ -24,6 +24,7 @@ pub fn yield_now() -> impl Future<Output = ()> { | ||||
|     YieldNowFuture { yielded: false } | ||||
| } | ||||
|  | ||||
| #[must_use = "futures do nothing unless you `.await` or poll them"] | ||||
| struct YieldNowFuture { | ||||
|     yielded: bool, | ||||
| } | ||||
|   | ||||
| @@ -2,6 +2,7 @@ | ||||
| name = "embassy-hal-common" | ||||
| version = "0.1.0" | ||||
| edition = "2021" | ||||
| license = "MIT OR Apache-2.0" | ||||
|  | ||||
| [features] | ||||
|  | ||||
|   | ||||
							
								
								
									
										558
									
								
								embassy-hal-common/src/atomic_ring_buffer.rs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										558
									
								
								embassy-hal-common/src/atomic_ring_buffer.rs
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,558 @@ | ||||
| use core::slice; | ||||
| use core::sync::atomic::{AtomicPtr, AtomicUsize, Ordering}; | ||||
|  | ||||
| /// Atomic reusable ringbuffer | ||||
| /// | ||||
| /// This ringbuffer implementation is designed to be stored in a `static`, | ||||
| /// therefore all methods take `&self` and not `&mut self`. | ||||
| /// | ||||
| /// It is "reusable": when created it has no backing buffer, you can give it | ||||
| /// one with `init` and take it back with `deinit`, and init it again in the | ||||
| /// future if needed. This is very non-idiomatic, but helps a lot when storing | ||||
| /// it in a `static`. | ||||
| /// | ||||
| /// One concurrent writer and one concurrent reader are supported, even at | ||||
| /// different execution priorities (like main and irq). | ||||
| pub struct RingBuffer { | ||||
|     pub buf: AtomicPtr<u8>, | ||||
|     pub len: AtomicUsize, | ||||
|  | ||||
|     // start and end wrap at len*2, not at len. | ||||
|     // This allows distinguishing "full" and "empty". | ||||
|     // full is when start+len == end (modulo len*2) | ||||
|     // empty is when start == end | ||||
|     // | ||||
|     // This avoids having to consider the ringbuffer "full" at len-1 instead of len. | ||||
|     // The usual solution is adding a "full" flag, but that can't be made atomic | ||||
|     pub start: AtomicUsize, | ||||
|     pub end: AtomicUsize, | ||||
| } | ||||
|  | ||||
| pub struct Reader<'a>(&'a RingBuffer); | ||||
| pub struct Writer<'a>(&'a RingBuffer); | ||||
|  | ||||
| impl RingBuffer { | ||||
|     /// Create a new empty ringbuffer. | ||||
|     pub const fn new() -> Self { | ||||
|         Self { | ||||
|             buf: AtomicPtr::new(core::ptr::null_mut()), | ||||
|             len: AtomicUsize::new(0), | ||||
|             start: AtomicUsize::new(0), | ||||
|             end: AtomicUsize::new(0), | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     /// Initialize the ring buffer with a buffer. | ||||
|     /// | ||||
|     /// # Safety | ||||
|     /// - The buffer (`buf .. buf+len`) must be valid memory until `deinit` is called. | ||||
|     /// - Must not be called concurrently with any other methods. | ||||
|     pub unsafe fn init(&self, buf: *mut u8, len: usize) { | ||||
|         // Ordering: it's OK to use `Relaxed` because this is not called | ||||
|         // concurrently with other methods. | ||||
|         self.buf.store(buf, Ordering::Relaxed); | ||||
|         self.len.store(len, Ordering::Relaxed); | ||||
|         self.start.store(0, Ordering::Relaxed); | ||||
|         self.end.store(0, Ordering::Relaxed); | ||||
|     } | ||||
|  | ||||
|     /// Deinitialize the ringbuffer. | ||||
|     /// | ||||
|     /// After calling this, the ringbuffer becomes empty, as if it was | ||||
|     /// just created with `new()`. | ||||
|     /// | ||||
|     /// # Safety | ||||
|     /// - Must not be called concurrently with any other methods. | ||||
|     pub unsafe fn deinit(&self) { | ||||
|         // Ordering: it's OK to use `Relaxed` because this is not called | ||||
|         // concurrently with other methods. | ||||
|         self.len.store(0, Ordering::Relaxed); | ||||
|         self.start.store(0, Ordering::Relaxed); | ||||
|         self.end.store(0, Ordering::Relaxed); | ||||
|     } | ||||
|  | ||||
|     /// Create a reader. | ||||
|     /// | ||||
|     /// # Safety | ||||
|     /// | ||||
|     /// Only one reader can exist at a time. | ||||
|     pub unsafe fn reader(&self) -> Reader<'_> { | ||||
|         Reader(self) | ||||
|     } | ||||
|  | ||||
|     /// Create a writer. | ||||
|     /// | ||||
|     /// # Safety | ||||
|     /// | ||||
|     /// Only one writer can exist at a time. | ||||
|     pub unsafe fn writer(&self) -> Writer<'_> { | ||||
|         Writer(self) | ||||
|     } | ||||
|  | ||||
|     pub fn len(&self) -> usize { | ||||
|         self.len.load(Ordering::Relaxed) | ||||
|     } | ||||
|  | ||||
|     pub fn is_full(&self) -> bool { | ||||
|         let len = self.len.load(Ordering::Relaxed); | ||||
|         let start = self.start.load(Ordering::Relaxed); | ||||
|         let end = self.end.load(Ordering::Relaxed); | ||||
|  | ||||
|         self.wrap(start + len) == end | ||||
|     } | ||||
|  | ||||
|     pub fn is_empty(&self) -> bool { | ||||
|         let start = self.start.load(Ordering::Relaxed); | ||||
|         let end = self.end.load(Ordering::Relaxed); | ||||
|  | ||||
|         start == end | ||||
|     } | ||||
|  | ||||
|     fn wrap(&self, mut n: usize) -> usize { | ||||
|         let len = self.len.load(Ordering::Relaxed); | ||||
|  | ||||
|         if n >= len * 2 { | ||||
|             n -= len * 2 | ||||
|         } | ||||
|         n | ||||
|     } | ||||
| } | ||||
|  | ||||
| impl<'a> Writer<'a> { | ||||
|     /// Push data into the buffer in-place. | ||||
|     /// | ||||
|     /// The closure `f` is called with a free part of the buffer, it must write | ||||
|     /// some data to it and return the amount of bytes written. | ||||
|     pub fn push(&mut self, f: impl FnOnce(&mut [u8]) -> usize) -> usize { | ||||
|         let (p, n) = self.push_buf(); | ||||
|         let buf = unsafe { slice::from_raw_parts_mut(p, n) }; | ||||
|         let n = f(buf); | ||||
|         self.push_done(n); | ||||
|         n | ||||
|     } | ||||
|  | ||||
|     /// Push one data byte. | ||||
|     /// | ||||
|     /// Returns true if pushed succesfully. | ||||
|     pub fn push_one(&mut self, val: u8) -> bool { | ||||
|         let n = self.push(|f| match f { | ||||
|             [] => 0, | ||||
|             [x, ..] => { | ||||
|                 *x = val; | ||||
|                 1 | ||||
|             } | ||||
|         }); | ||||
|         n != 0 | ||||
|     } | ||||
|  | ||||
|     /// Get a buffer where data can be pushed to. | ||||
|     /// | ||||
|     /// Equivalent to [`Self::push_buf`] but returns a slice. | ||||
|     pub fn push_slice(&mut self) -> &mut [u8] { | ||||
|         let (data, len) = self.push_buf(); | ||||
|         unsafe { slice::from_raw_parts_mut(data, len) } | ||||
|     } | ||||
|  | ||||
|     /// Get up to two buffers where data can be pushed to. | ||||
|     /// | ||||
|     /// Equivalent to [`Self::push_bufs`] but returns slices. | ||||
|     pub fn push_slices(&mut self) -> [&mut [u8]; 2] { | ||||
|         let [(d0, l0), (d1, l1)] = self.push_bufs(); | ||||
|         unsafe { [slice::from_raw_parts_mut(d0, l0), slice::from_raw_parts_mut(d1, l1)] } | ||||
|     } | ||||
|  | ||||
|     /// Get a buffer where data can be pushed to. | ||||
|     /// | ||||
|     /// Write data to the start of the buffer, then call `push_done` with | ||||
|     /// however many bytes you've pushed. | ||||
|     /// | ||||
|     /// The buffer is suitable to DMA to. | ||||
|     /// | ||||
|     /// If the ringbuf is full, size=0 will be returned. | ||||
|     /// | ||||
|     /// The buffer stays valid as long as no other `Writer` method is called | ||||
|     /// and `init`/`deinit` aren't called on the ringbuf. | ||||
|     pub fn push_buf(&mut self) -> (*mut u8, usize) { | ||||
|         // Ordering: popping writes `start` last, so we read `start` first. | ||||
|         // Read it with Acquire ordering, so that the next accesses can't be reordered up past it. | ||||
|         let mut start = self.0.start.load(Ordering::Acquire); | ||||
|         let buf = self.0.buf.load(Ordering::Relaxed); | ||||
|         let len = self.0.len.load(Ordering::Relaxed); | ||||
|         let mut end = self.0.end.load(Ordering::Relaxed); | ||||
|  | ||||
|         let empty = start == end; | ||||
|  | ||||
|         if start >= len { | ||||
|             start -= len | ||||
|         } | ||||
|         if end >= len { | ||||
|             end -= len | ||||
|         } | ||||
|  | ||||
|         if start == end && !empty { | ||||
|             // full | ||||
|             return (buf, 0); | ||||
|         } | ||||
|         let n = if start > end { start - end } else { len - end }; | ||||
|  | ||||
|         trace!("  ringbuf: push_buf {:?}..{:?}", end, end + n); | ||||
|         (unsafe { buf.add(end) }, n) | ||||
|     } | ||||
|  | ||||
|     /// Get up to two buffers where data can be pushed to. | ||||
|     /// | ||||
|     /// Write data starting at the beginning of the first buffer, then call | ||||
|     /// `push_done` with however many bytes you've pushed. | ||||
|     /// | ||||
|     /// The buffers are suitable to DMA to. | ||||
|     /// | ||||
|     /// If the ringbuf is full, both buffers will be zero length. | ||||
|     /// If there is only area available, the second buffer will be zero length. | ||||
|     /// | ||||
|     /// The buffer stays valid as long as no other `Writer` method is called | ||||
|     /// and `init`/`deinit` aren't called on the ringbuf. | ||||
|     pub fn push_bufs(&mut self) -> [(*mut u8, usize); 2] { | ||||
|         // Ordering: as per push_buf() | ||||
|         let mut start = self.0.start.load(Ordering::Acquire); | ||||
|         let buf = self.0.buf.load(Ordering::Relaxed); | ||||
|         let len = self.0.len.load(Ordering::Relaxed); | ||||
|         let mut end = self.0.end.load(Ordering::Relaxed); | ||||
|  | ||||
|         let empty = start == end; | ||||
|  | ||||
|         if start >= len { | ||||
|             start -= len | ||||
|         } | ||||
|         if end >= len { | ||||
|             end -= len | ||||
|         } | ||||
|  | ||||
|         if start == end && !empty { | ||||
|             // full | ||||
|             return [(buf, 0), (buf, 0)]; | ||||
|         } | ||||
|         let n0 = if start > end { start - end } else { len - end }; | ||||
|         let n1 = if start <= end { start } else { 0 }; | ||||
|  | ||||
|         trace!("  ringbuf: push_bufs [{:?}..{:?}, {:?}..{:?}]", end, end + n0, 0, n1); | ||||
|         [(unsafe { buf.add(end) }, n0), (buf, n1)] | ||||
|     } | ||||
|  | ||||
|     pub fn push_done(&mut self, n: usize) { | ||||
|         trace!("  ringbuf: push {:?}", n); | ||||
|         let end = self.0.end.load(Ordering::Relaxed); | ||||
|  | ||||
|         // Ordering: write `end` last, with Release ordering. | ||||
|         // The ordering ensures no preceding memory accesses (such as writing | ||||
|         // the actual data in the buffer) can be reordered down past it, which | ||||
|         // will guarantee the reader sees them after reading from `end`. | ||||
|         self.0.end.store(self.0.wrap(end + n), Ordering::Release); | ||||
|     } | ||||
| } | ||||
|  | ||||
| impl<'a> Reader<'a> { | ||||
|     /// Pop data from the buffer in-place. | ||||
|     /// | ||||
|     /// The closure `f` is called with the next data, it must process | ||||
|     /// some data from it and return the amount of bytes processed. | ||||
|     pub fn pop(&mut self, f: impl FnOnce(&[u8]) -> usize) -> usize { | ||||
|         let (p, n) = self.pop_buf(); | ||||
|         let buf = unsafe { slice::from_raw_parts(p, n) }; | ||||
|         let n = f(buf); | ||||
|         self.pop_done(n); | ||||
|         n | ||||
|     } | ||||
|  | ||||
|     /// Pop one data byte. | ||||
|     /// | ||||
|     /// Returns true if popped succesfully. | ||||
|     pub fn pop_one(&mut self) -> Option<u8> { | ||||
|         let mut res = None; | ||||
|         self.pop(|f| match f { | ||||
|             &[] => 0, | ||||
|             &[x, ..] => { | ||||
|                 res = Some(x); | ||||
|                 1 | ||||
|             } | ||||
|         }); | ||||
|         res | ||||
|     } | ||||
|  | ||||
|     /// Get a buffer where data can be popped from. | ||||
|     /// | ||||
|     /// Equivalent to [`Self::pop_buf`] but returns a slice. | ||||
|     pub fn pop_slice(&mut self) -> &mut [u8] { | ||||
|         let (data, len) = self.pop_buf(); | ||||
|         unsafe { slice::from_raw_parts_mut(data, len) } | ||||
|     } | ||||
|  | ||||
|     /// Get a buffer where data can be popped from. | ||||
|     /// | ||||
|     /// Read data from the start of the buffer, then call `pop_done` with | ||||
|     /// however many bytes you've processed. | ||||
|     /// | ||||
|     /// The buffer is suitable to DMA from. | ||||
|     /// | ||||
|     /// If the ringbuf is empty, size=0 will be returned. | ||||
|     /// | ||||
|     /// The buffer stays valid as long as no other `Reader` method is called | ||||
|     /// and `init`/`deinit` aren't called on the ringbuf. | ||||
|     pub fn pop_buf(&mut self) -> (*mut u8, usize) { | ||||
|         // Ordering: pushing writes `end` last, so we read `end` first. | ||||
|         // Read it with Acquire ordering, so that the next accesses can't be reordered up past it. | ||||
|         // This is needed to guarantee we "see" the data written by the writer. | ||||
|         let mut end = self.0.end.load(Ordering::Acquire); | ||||
|         let buf = self.0.buf.load(Ordering::Relaxed); | ||||
|         let len = self.0.len.load(Ordering::Relaxed); | ||||
|         let mut start = self.0.start.load(Ordering::Relaxed); | ||||
|  | ||||
|         if start == end { | ||||
|             return (buf, 0); | ||||
|         } | ||||
|  | ||||
|         if start >= len { | ||||
|             start -= len | ||||
|         } | ||||
|         if end >= len { | ||||
|             end -= len | ||||
|         } | ||||
|  | ||||
|         let n = if end > start { end - start } else { len - start }; | ||||
|  | ||||
|         trace!("  ringbuf: pop_buf {:?}..{:?}", start, start + n); | ||||
|         (unsafe { buf.add(start) }, n) | ||||
|     } | ||||
|  | ||||
|     pub fn pop_done(&mut self, n: usize) { | ||||
|         trace!("  ringbuf: pop {:?}", n); | ||||
|  | ||||
|         let start = self.0.start.load(Ordering::Relaxed); | ||||
|  | ||||
|         // Ordering: write `start` last, with Release ordering. | ||||
|         // The ordering ensures no preceding memory accesses (such as reading | ||||
|         // the actual data) can be reordered down past it. This is necessary | ||||
|         // because writing to `start` is effectively freeing the read part of the | ||||
|         // buffer, which "gives permission" to the writer to write to it again. | ||||
|         // Therefore, all buffer accesses must be completed before this. | ||||
|         self.0.start.store(self.0.wrap(start + n), Ordering::Release); | ||||
|     } | ||||
| } | ||||
|  | ||||
| #[cfg(test)] | ||||
| mod tests { | ||||
|     use super::*; | ||||
|  | ||||
|     #[test] | ||||
|     fn push_pop() { | ||||
|         let mut b = [0; 4]; | ||||
|         let rb = RingBuffer::new(); | ||||
|         unsafe { | ||||
|             rb.init(b.as_mut_ptr(), 4); | ||||
|  | ||||
|             assert_eq!(rb.is_empty(), true); | ||||
|             assert_eq!(rb.is_full(), false); | ||||
|  | ||||
|             rb.writer().push(|buf| { | ||||
|                 assert_eq!(4, buf.len()); | ||||
|                 buf[0] = 1; | ||||
|                 buf[1] = 2; | ||||
|                 buf[2] = 3; | ||||
|                 buf[3] = 4; | ||||
|                 4 | ||||
|             }); | ||||
|  | ||||
|             assert_eq!(rb.is_empty(), false); | ||||
|             assert_eq!(rb.is_full(), true); | ||||
|  | ||||
|             rb.writer().push(|buf| { | ||||
|                 // If it's full, we can push 0 bytes. | ||||
|                 assert_eq!(0, buf.len()); | ||||
|                 0 | ||||
|             }); | ||||
|  | ||||
|             assert_eq!(rb.is_empty(), false); | ||||
|             assert_eq!(rb.is_full(), true); | ||||
|  | ||||
|             rb.reader().pop(|buf| { | ||||
|                 assert_eq!(4, buf.len()); | ||||
|                 assert_eq!(1, buf[0]); | ||||
|                 1 | ||||
|             }); | ||||
|  | ||||
|             assert_eq!(rb.is_empty(), false); | ||||
|             assert_eq!(rb.is_full(), false); | ||||
|  | ||||
|             rb.reader().pop(|buf| { | ||||
|                 assert_eq!(3, buf.len()); | ||||
|                 0 | ||||
|             }); | ||||
|  | ||||
|             assert_eq!(rb.is_empty(), false); | ||||
|             assert_eq!(rb.is_full(), false); | ||||
|  | ||||
|             rb.reader().pop(|buf| { | ||||
|                 assert_eq!(3, buf.len()); | ||||
|                 assert_eq!(2, buf[0]); | ||||
|                 assert_eq!(3, buf[1]); | ||||
|                 2 | ||||
|             }); | ||||
|             rb.reader().pop(|buf| { | ||||
|                 assert_eq!(1, buf.len()); | ||||
|                 assert_eq!(4, buf[0]); | ||||
|                 1 | ||||
|             }); | ||||
|  | ||||
|             assert_eq!(rb.is_empty(), true); | ||||
|             assert_eq!(rb.is_full(), false); | ||||
|  | ||||
|             rb.reader().pop(|buf| { | ||||
|                 assert_eq!(0, buf.len()); | ||||
|                 0 | ||||
|             }); | ||||
|  | ||||
|             rb.writer().push(|buf| { | ||||
|                 assert_eq!(4, buf.len()); | ||||
|                 buf[0] = 10; | ||||
|                 1 | ||||
|             }); | ||||
|  | ||||
|             rb.writer().push(|buf| { | ||||
|                 assert_eq!(3, buf.len()); | ||||
|                 buf[0] = 11; | ||||
|                 buf[1] = 12; | ||||
|                 2 | ||||
|             }); | ||||
|  | ||||
|             assert_eq!(rb.is_empty(), false); | ||||
|             assert_eq!(rb.is_full(), false); | ||||
|  | ||||
|             rb.writer().push(|buf| { | ||||
|                 assert_eq!(1, buf.len()); | ||||
|                 buf[0] = 13; | ||||
|                 1 | ||||
|             }); | ||||
|  | ||||
|             assert_eq!(rb.is_empty(), false); | ||||
|             assert_eq!(rb.is_full(), true); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     #[test] | ||||
|     fn zero_len() { | ||||
|         let rb = RingBuffer::new(); | ||||
|         unsafe { | ||||
|             assert_eq!(rb.is_empty(), true); | ||||
|             assert_eq!(rb.is_full(), true); | ||||
|  | ||||
|             rb.writer().push(|buf| { | ||||
|                 assert_eq!(0, buf.len()); | ||||
|                 0 | ||||
|             }); | ||||
|  | ||||
|             rb.reader().pop(|buf| { | ||||
|                 assert_eq!(0, buf.len()); | ||||
|                 0 | ||||
|             }); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     #[test] | ||||
|     fn push_slices() { | ||||
|         init(); | ||||
|  | ||||
|         let mut b = [0; 4]; | ||||
|         let rb = RingBuffer::new(); | ||||
|         unsafe { | ||||
|             rb.init(b.as_mut_ptr(), 4); | ||||
|  | ||||
|             /* push 3 -> [1 2 3 x] */ | ||||
|             let mut w = rb.writer(); | ||||
|             let ps = w.push_slices(); | ||||
|             assert_eq!(4, ps[0].len()); | ||||
|             assert_eq!(0, ps[1].len()); | ||||
|             ps[0][0] = 1; | ||||
|             ps[0][1] = 2; | ||||
|             ps[0][2] = 3; | ||||
|             w.push_done(3); | ||||
|             drop(w); | ||||
|  | ||||
|             /* pop 2 -> [x x 3 x] */ | ||||
|             rb.reader().pop(|buf| { | ||||
|                 assert_eq!(3, buf.len()); | ||||
|                 assert_eq!(1, buf[0]); | ||||
|                 assert_eq!(2, buf[1]); | ||||
|                 assert_eq!(3, buf[2]); | ||||
|                 2 | ||||
|             }); | ||||
|  | ||||
|             /* push 3 -> [5 6 3 4] */ | ||||
|             let mut w = rb.writer(); | ||||
|             let ps = w.push_slices(); | ||||
|             assert_eq!(1, ps[0].len()); | ||||
|             assert_eq!(2, ps[1].len()); | ||||
|             ps[0][0] = 4; | ||||
|             ps[1][0] = 5; | ||||
|             ps[1][1] = 6; | ||||
|             w.push_done(3); | ||||
|             drop(w); | ||||
|  | ||||
|             /* buf is now full */ | ||||
|             let mut w = rb.writer(); | ||||
|             let ps = w.push_slices(); | ||||
|             assert_eq!(0, ps[0].len()); | ||||
|             assert_eq!(0, ps[1].len()); | ||||
|  | ||||
|             /* pop 2 -> [5 6 x x] */ | ||||
|             rb.reader().pop(|buf| { | ||||
|                 assert_eq!(2, buf.len()); | ||||
|                 assert_eq!(3, buf[0]); | ||||
|                 assert_eq!(4, buf[1]); | ||||
|                 2 | ||||
|             }); | ||||
|  | ||||
|             /* should now have one push slice again */ | ||||
|             let mut w = rb.writer(); | ||||
|             let ps = w.push_slices(); | ||||
|             assert_eq!(2, ps[0].len()); | ||||
|             assert_eq!(0, ps[1].len()); | ||||
|             drop(w); | ||||
|  | ||||
|             /* pop 2 -> [x x x x] */ | ||||
|             rb.reader().pop(|buf| { | ||||
|                 assert_eq!(2, buf.len()); | ||||
|                 assert_eq!(5, buf[0]); | ||||
|                 assert_eq!(6, buf[1]); | ||||
|                 2 | ||||
|             }); | ||||
|  | ||||
|             /* should now have two push slices */ | ||||
|             let mut w = rb.writer(); | ||||
|             let ps = w.push_slices(); | ||||
|             assert_eq!(2, ps[0].len()); | ||||
|             assert_eq!(2, ps[1].len()); | ||||
|             drop(w); | ||||
|  | ||||
|             /* make sure we exercise all wrap around cases properly */ | ||||
|             for _ in 0..10 { | ||||
|                 /* should be empty, push 1 */ | ||||
|                 let mut w = rb.writer(); | ||||
|                 let ps = w.push_slices(); | ||||
|                 assert_eq!(4, ps[0].len() + ps[1].len()); | ||||
|                 w.push_done(1); | ||||
|                 drop(w); | ||||
|  | ||||
|                 /* should have 1 element */ | ||||
|                 let mut w = rb.writer(); | ||||
|                 let ps = w.push_slices(); | ||||
|                 assert_eq!(3, ps[0].len() + ps[1].len()); | ||||
|                 drop(w); | ||||
|  | ||||
|                 /* pop 1 */ | ||||
|                 rb.reader().pop(|buf| { | ||||
|                     assert_eq!(1, buf.len()); | ||||
|                     1 | ||||
|                 }); | ||||
|             } | ||||
|         } | ||||
|     } | ||||
| } | ||||
| @@ -1,6 +1,7 @@ | ||||
| use core::mem; | ||||
| use core::mem::MaybeUninit; | ||||
|  | ||||
| #[must_use = "to delay the drop handler invokation to the end of the scope"] | ||||
| pub struct OnDrop<F: FnOnce()> { | ||||
|     f: MaybeUninit<F>, | ||||
| } | ||||
| @@ -27,6 +28,7 @@ impl<F: FnOnce()> Drop for OnDrop<F> { | ||||
| /// | ||||
| /// To correctly dispose of this device, call the [defuse](struct.DropBomb.html#method.defuse) | ||||
| /// method before this object is dropped. | ||||
| #[must_use = "to delay the drop bomb invokation to the end of the scope"] | ||||
| pub struct DropBomb { | ||||
|     _private: (), | ||||
| } | ||||
|   | ||||
| @@ -4,6 +4,7 @@ | ||||
| // This mod MUST go first, so that the others see its macros. | ||||
| pub(crate) mod fmt; | ||||
|  | ||||
| pub mod atomic_ring_buffer; | ||||
| pub mod drop; | ||||
| mod macros; | ||||
| mod peripheral; | ||||
|   | ||||
| @@ -1,10 +1,12 @@ | ||||
| #[macro_export] | ||||
| macro_rules! peripherals { | ||||
| macro_rules! peripherals_definition { | ||||
|     ($($(#[$cfg:meta])? $name:ident),*$(,)?) => { | ||||
|         /// Types for the peripheral singletons. | ||||
|         pub mod peripherals { | ||||
|             $( | ||||
|                 $(#[$cfg])? | ||||
|                 #[allow(non_camel_case_types)] | ||||
|                 #[doc = concat!(stringify!($name), " peripheral")] | ||||
|                 pub struct $name { _private: () } | ||||
|  | ||||
|                 $(#[$cfg])? | ||||
| @@ -24,10 +26,19 @@ macro_rules! peripherals { | ||||
|                 $crate::impl_peripheral!($name); | ||||
|             )* | ||||
|         } | ||||
|     }; | ||||
| } | ||||
|  | ||||
| #[macro_export] | ||||
| macro_rules! peripherals_struct { | ||||
|     ($($(#[$cfg:meta])? $name:ident),*$(,)?) => { | ||||
|         /// Struct containing all the peripheral singletons. | ||||
|         /// | ||||
|         /// To obtain the peripherals, you must initialize the HAL, by calling [`crate::init`]. | ||||
|         #[allow(non_snake_case)] | ||||
|         pub struct Peripherals { | ||||
|             $( | ||||
|                 #[doc = concat!(stringify!($name), " peripheral")] | ||||
|                 $(#[$cfg])? | ||||
|                 pub $name: peripherals::$name, | ||||
|             )* | ||||
| @@ -70,6 +81,24 @@ macro_rules! peripherals { | ||||
|     }; | ||||
| } | ||||
|  | ||||
| #[macro_export] | ||||
| macro_rules! peripherals { | ||||
|     ($($(#[$cfg:meta])? $name:ident),*$(,)?) => { | ||||
|         $crate::peripherals_definition!( | ||||
|             $( | ||||
|                 $(#[$cfg])? | ||||
|                 $name, | ||||
|             )* | ||||
|         ); | ||||
|         $crate::peripherals_struct!( | ||||
|             $( | ||||
|                 $(#[$cfg])? | ||||
|                 $name, | ||||
|             )* | ||||
|         ); | ||||
|     }; | ||||
| } | ||||
|  | ||||
| #[macro_export] | ||||
| macro_rules! into_ref { | ||||
|     ($($name:ident),*) => { | ||||
| @@ -86,7 +115,7 @@ macro_rules! impl_peripheral { | ||||
|             type P = $type; | ||||
|  | ||||
|             #[inline] | ||||
|             unsafe fn clone_unchecked(&mut self) -> Self::P { | ||||
|             unsafe fn clone_unchecked(&self) -> Self::P { | ||||
|                 $type { ..*self } | ||||
|             } | ||||
|         } | ||||
|   | ||||
| @@ -3,16 +3,17 @@ use core::ops::{Deref, DerefMut}; | ||||
|  | ||||
| /// An exclusive reference to a peripheral. | ||||
| /// | ||||
| /// This is functionally the same as a `&'a mut T`. The reason for having a | ||||
| /// dedicated struct is memory efficiency: | ||||
| /// This is functionally the same as a `&'a mut T`. There's a few advantages in having | ||||
| /// a dedicated struct instead: | ||||
| /// | ||||
| /// Peripheral singletons are typically either zero-sized (for concrete peripehrals | ||||
| /// like `PA9` or `Spi4`) or very small (for example `AnyPin` which is 1 byte). | ||||
| /// However `&mut T` is always 4 bytes for 32-bit targets, even if T is zero-sized. | ||||
| /// PeripheralRef stores a copy of `T` instead, so it's the same size. | ||||
| /// | ||||
| /// but it is the size of `T` not the size | ||||
| /// of a pointer. This is useful if T is a zero sized type. | ||||
| /// - Memory efficiency: Peripheral singletons are typically either zero-sized (for concrete | ||||
| ///   peripherals like `PA9` or `SPI4`) or very small (for example `AnyPin`, which is 1 byte). | ||||
| ///   However `&mut T` is always 4 bytes for 32-bit targets, even if T is zero-sized. | ||||
| ///   PeripheralRef stores a copy of `T` instead, so it's the same size. | ||||
| /// - Code size efficiency. If the user uses the same driver with both `SPI4` and `&mut SPI4`, | ||||
| ///   the driver code would be monomorphized two times. With PeripheralRef, the driver is generic | ||||
| ///   over a lifetime only. `SPI4` becomes `PeripheralRef<'static, SPI4>`, and `&mut SPI4` becomes | ||||
| ///   `PeripheralRef<'a, SPI4>`. Lifetimes don't cause monomorphization. | ||||
| pub struct PeripheralRef<'a, T> { | ||||
|     inner: T, | ||||
|     _lifetime: PhantomData<&'a mut T>, | ||||
| @@ -38,7 +39,7 @@ impl<'a, T> PeripheralRef<'a, T> { | ||||
|     /// You should strongly prefer using `reborrow()` instead. It returns a | ||||
|     /// `PeripheralRef` that borrows `self`, which allows the borrow checker | ||||
|     /// to enforce this at compile time. | ||||
|     pub unsafe fn clone_unchecked(&mut self) -> PeripheralRef<'a, T> | ||||
|     pub unsafe fn clone_unchecked(&self) -> PeripheralRef<'a, T> | ||||
|     where | ||||
|         T: Peripheral<P = T>, | ||||
|     { | ||||
| @@ -145,14 +146,14 @@ pub trait Peripheral: Sized { | ||||
|     /// | ||||
|     /// You should strongly prefer using `into_ref()` instead. It returns a | ||||
|     /// `PeripheralRef`, which allows the borrow checker to enforce this at compile time. | ||||
|     unsafe fn clone_unchecked(&mut self) -> Self::P; | ||||
|     unsafe fn clone_unchecked(&self) -> Self::P; | ||||
|  | ||||
|     /// Convert a value into a `PeripheralRef`. | ||||
|     /// | ||||
|     /// When called on an owned `T`, yields a `PeripheralRef<'static, T>`. | ||||
|     /// When called on an `&'a mut T`, yields a `PeripheralRef<'a, T>`. | ||||
|     #[inline] | ||||
|     fn into_ref<'a>(mut self) -> PeripheralRef<'a, Self::P> | ||||
|     fn into_ref<'a>(self) -> PeripheralRef<'a, Self::P> | ||||
|     where | ||||
|         Self: 'a, | ||||
|     { | ||||
| @@ -160,14 +161,14 @@ pub trait Peripheral: Sized { | ||||
|     } | ||||
| } | ||||
|  | ||||
| impl<'b, T: DerefMut> Peripheral for T | ||||
| impl<'b, T: Deref> Peripheral for T | ||||
| where | ||||
|     T::Target: Peripheral, | ||||
| { | ||||
|     type P = <T::Target as Peripheral>::P; | ||||
|  | ||||
|     #[inline] | ||||
|     unsafe fn clone_unchecked(&mut self) -> Self::P { | ||||
|         self.deref_mut().clone_unchecked() | ||||
|     unsafe fn clone_unchecked(&self) -> Self::P { | ||||
|         self.deref().clone_unchecked() | ||||
|     } | ||||
| } | ||||
|   | ||||
| @@ -2,22 +2,18 @@ | ||||
| name = "embassy-lora" | ||||
| version = "0.1.0" | ||||
| edition = "2021" | ||||
| license = "MIT OR Apache-2.0" | ||||
|  | ||||
| [package.metadata.embassy_docs] | ||||
| src_base = "https://github.com/embassy-rs/embassy/blob/embassy-lora-v$VERSION/embassy-lora/src/" | ||||
| src_base_git = "https://github.com/embassy-rs/embassy/blob/$COMMIT/embassy-lora/src/" | ||||
| features = ["time", "defmt"] | ||||
| flavors = [ | ||||
|     { name = "sx127x",  target = "thumbv7em-none-eabihf", features = ["sx127x", "embassy-stm32/stm32wl55jc-cm4", "embassy-stm32/time-driver-any", "embassy-time/tick-32768hz"] }, | ||||
|     { name = "stm32wl", target = "thumbv7em-none-eabihf", features = ["stm32wl", "embassy-stm32/stm32wl55jc-cm4", "embassy-stm32/time-driver-any", "embassy-time/tick-32768hz"] }, | ||||
| ] | ||||
|  | ||||
| [lib] | ||||
| features = ["stm32wl", "time", "defmt"] | ||||
| target = "thumbv7em-none-eabi" | ||||
|  | ||||
| [features] | ||||
| sx127x = [] | ||||
| stm32wl = ["embassy-stm32", "embassy-stm32/subghz"] | ||||
| stm32wl = ["dep:embassy-stm32"] | ||||
| time = [] | ||||
| defmt = ["dep:defmt", "lorawan-device/defmt"] | ||||
|  | ||||
| [dependencies] | ||||
|  | ||||
| @@ -25,14 +21,14 @@ defmt = { version = "0.3", optional = true } | ||||
| log = { version = "0.4.14", optional = true } | ||||
|  | ||||
| embassy-time = { version = "0.1.0", path = "../embassy-time" } | ||||
| embassy-sync = { version = "0.1.0", path = "../embassy-sync" } | ||||
| embassy-sync = { version = "0.2.0", path = "../embassy-sync" } | ||||
| embassy-stm32 = { version = "0.1.0", path = "../embassy-stm32", default-features = false, optional = true } | ||||
| embedded-hal-1 = { package = "embedded-hal", version = "1.0.0-alpha.8" } | ||||
| embedded-hal-async = { version = "0.1.0-alpha.1" } | ||||
| embedded-hal-1 = { package = "embedded-hal", version = "=1.0.0-alpha.10" } | ||||
| embedded-hal-async = { version = "=0.2.0-alpha.1" } | ||||
| embassy-hal-common = { version = "0.1.0", path = "../embassy-hal-common", default-features = false } | ||||
| futures = { version = "0.3.17", default-features = false, features = [ "async-await" ] } | ||||
| embedded-hal = { version = "0.2", features = ["unproven"] } | ||||
| bit_field = { version = "0.10" } | ||||
|  | ||||
| lorawan-device = { version = "0.7.1", default-features = false, features = ["async"] } | ||||
| lorawan = { version = "0.7.1", default-features = false } | ||||
| lora-phy = { version = "1" } | ||||
| lorawan-device = { version = "0.10.0", default-features = false, features = ["async"] } | ||||
|   | ||||
							
								
								
									
										325
									
								
								embassy-lora/src/iv.rs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										325
									
								
								embassy-lora/src/iv.rs
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,325 @@ | ||||
| #[cfg(feature = "stm32wl")] | ||||
| use embassy_stm32::interrupt::*; | ||||
| #[cfg(feature = "stm32wl")] | ||||
| use embassy_stm32::{pac, PeripheralRef}; | ||||
| #[cfg(feature = "stm32wl")] | ||||
| use embassy_sync::blocking_mutex::raw::CriticalSectionRawMutex; | ||||
| #[cfg(feature = "stm32wl")] | ||||
| use embassy_sync::signal::Signal; | ||||
| use embedded_hal::digital::v2::OutputPin; | ||||
| use embedded_hal_async::delay::DelayUs; | ||||
| use embedded_hal_async::digital::Wait; | ||||
| use lora_phy::mod_params::RadioError::*; | ||||
| use lora_phy::mod_params::{BoardType, RadioError}; | ||||
| use lora_phy::mod_traits::InterfaceVariant; | ||||
|  | ||||
| #[cfg(feature = "stm32wl")] | ||||
| static IRQ_SIGNAL: Signal<CriticalSectionRawMutex, ()> = Signal::new(); | ||||
|  | ||||
| #[cfg(feature = "stm32wl")] | ||||
| /// Base for the InterfaceVariant implementation for an stm32wl/sx1262 combination | ||||
| pub struct Stm32wlInterfaceVariant<'a, CTRL> { | ||||
|     board_type: BoardType, | ||||
|     irq: PeripheralRef<'a, SUBGHZ_RADIO>, | ||||
|     rf_switch_rx: Option<CTRL>, | ||||
|     rf_switch_tx: Option<CTRL>, | ||||
| } | ||||
|  | ||||
| #[cfg(feature = "stm32wl")] | ||||
| impl<'a, CTRL> Stm32wlInterfaceVariant<'a, CTRL> | ||||
| where | ||||
|     CTRL: OutputPin, | ||||
| { | ||||
|     /// Create an InterfaceVariant instance for an stm32wl/sx1262 combination | ||||
|     pub fn new( | ||||
|         irq: PeripheralRef<'a, SUBGHZ_RADIO>, | ||||
|         rf_switch_rx: Option<CTRL>, | ||||
|         rf_switch_tx: Option<CTRL>, | ||||
|     ) -> Result<Self, RadioError> { | ||||
|         irq.disable(); | ||||
|         irq.set_handler(Self::on_interrupt); | ||||
|         Ok(Self { | ||||
|             board_type: BoardType::Stm32wlSx1262, // updated when associated with a specific LoRa board | ||||
|             irq, | ||||
|             rf_switch_rx, | ||||
|             rf_switch_tx, | ||||
|         }) | ||||
|     } | ||||
|  | ||||
|     fn on_interrupt(_: *mut ()) { | ||||
|         unsafe { SUBGHZ_RADIO::steal() }.disable(); | ||||
|         IRQ_SIGNAL.signal(()); | ||||
|     } | ||||
| } | ||||
|  | ||||
| #[cfg(feature = "stm32wl")] | ||||
| impl<CTRL> InterfaceVariant for Stm32wlInterfaceVariant<'_, CTRL> | ||||
| where | ||||
|     CTRL: OutputPin, | ||||
| { | ||||
|     fn set_board_type(&mut self, board_type: BoardType) { | ||||
|         self.board_type = board_type; | ||||
|     } | ||||
|     async fn set_nss_low(&mut self) -> Result<(), RadioError> { | ||||
|         let pwr = pac::PWR; | ||||
|         unsafe { | ||||
|             pwr.subghzspicr().modify(|w| w.set_nss(pac::pwr::vals::Nss::LOW)); | ||||
|         } | ||||
|         Ok(()) | ||||
|     } | ||||
|     async fn set_nss_high(&mut self) -> Result<(), RadioError> { | ||||
|         let pwr = pac::PWR; | ||||
|         unsafe { | ||||
|             pwr.subghzspicr().modify(|w| w.set_nss(pac::pwr::vals::Nss::HIGH)); | ||||
|         } | ||||
|         Ok(()) | ||||
|     } | ||||
|     async fn reset(&mut self, _delay: &mut impl DelayUs) -> Result<(), RadioError> { | ||||
|         let rcc = pac::RCC; | ||||
|         unsafe { | ||||
|             rcc.csr().modify(|w| w.set_rfrst(true)); | ||||
|             rcc.csr().modify(|w| w.set_rfrst(false)); | ||||
|         } | ||||
|         Ok(()) | ||||
|     } | ||||
|     async fn wait_on_busy(&mut self) -> Result<(), RadioError> { | ||||
|         let pwr = pac::PWR; | ||||
|         while unsafe { pwr.sr2().read().rfbusys() == pac::pwr::vals::Rfbusys::BUSY } {} | ||||
|         Ok(()) | ||||
|     } | ||||
|  | ||||
|     async fn await_irq(&mut self) -> Result<(), RadioError> { | ||||
|         self.irq.enable(); | ||||
|         IRQ_SIGNAL.wait().await; | ||||
|         Ok(()) | ||||
|     } | ||||
|  | ||||
|     async fn enable_rf_switch_rx(&mut self) -> Result<(), RadioError> { | ||||
|         match &mut self.rf_switch_tx { | ||||
|             Some(pin) => pin.set_low().map_err(|_| RfSwitchTx)?, | ||||
|             None => (), | ||||
|         }; | ||||
|         match &mut self.rf_switch_rx { | ||||
|             Some(pin) => pin.set_high().map_err(|_| RfSwitchRx), | ||||
|             None => Ok(()), | ||||
|         } | ||||
|     } | ||||
|     async fn enable_rf_switch_tx(&mut self) -> Result<(), RadioError> { | ||||
|         match &mut self.rf_switch_rx { | ||||
|             Some(pin) => pin.set_low().map_err(|_| RfSwitchRx)?, | ||||
|             None => (), | ||||
|         }; | ||||
|         match &mut self.rf_switch_tx { | ||||
|             Some(pin) => pin.set_high().map_err(|_| RfSwitchTx), | ||||
|             None => Ok(()), | ||||
|         } | ||||
|     } | ||||
|     async fn disable_rf_switch(&mut self) -> Result<(), RadioError> { | ||||
|         match &mut self.rf_switch_rx { | ||||
|             Some(pin) => pin.set_low().map_err(|_| RfSwitchRx)?, | ||||
|             None => (), | ||||
|         }; | ||||
|         match &mut self.rf_switch_tx { | ||||
|             Some(pin) => pin.set_low().map_err(|_| RfSwitchTx), | ||||
|             None => Ok(()), | ||||
|         } | ||||
|     } | ||||
| } | ||||
|  | ||||
| /// Base for the InterfaceVariant implementation for an stm32l0/sx1276 combination | ||||
| pub struct Stm32l0InterfaceVariant<CTRL, WAIT> { | ||||
|     board_type: BoardType, | ||||
|     nss: CTRL, | ||||
|     reset: CTRL, | ||||
|     irq: WAIT, | ||||
|     rf_switch_rx: Option<CTRL>, | ||||
|     rf_switch_tx: Option<CTRL>, | ||||
| } | ||||
|  | ||||
| impl<CTRL, WAIT> Stm32l0InterfaceVariant<CTRL, WAIT> | ||||
| where | ||||
|     CTRL: OutputPin, | ||||
|     WAIT: Wait, | ||||
| { | ||||
|     /// Create an InterfaceVariant instance for an stm32l0/sx1276 combination | ||||
|     pub fn new( | ||||
|         nss: CTRL, | ||||
|         reset: CTRL, | ||||
|         irq: WAIT, | ||||
|         rf_switch_rx: Option<CTRL>, | ||||
|         rf_switch_tx: Option<CTRL>, | ||||
|     ) -> Result<Self, RadioError> { | ||||
|         Ok(Self { | ||||
|             board_type: BoardType::Stm32l0Sx1276, // updated when associated with a specific LoRa board | ||||
|             nss, | ||||
|             reset, | ||||
|             irq, | ||||
|             rf_switch_rx, | ||||
|             rf_switch_tx, | ||||
|         }) | ||||
|     } | ||||
| } | ||||
|  | ||||
| impl<CTRL, WAIT> InterfaceVariant for Stm32l0InterfaceVariant<CTRL, WAIT> | ||||
| where | ||||
|     CTRL: OutputPin, | ||||
|     WAIT: Wait, | ||||
| { | ||||
|     fn set_board_type(&mut self, board_type: BoardType) { | ||||
|         self.board_type = board_type; | ||||
|     } | ||||
|     async fn set_nss_low(&mut self) -> Result<(), RadioError> { | ||||
|         self.nss.set_low().map_err(|_| NSS) | ||||
|     } | ||||
|     async fn set_nss_high(&mut self) -> Result<(), RadioError> { | ||||
|         self.nss.set_high().map_err(|_| NSS) | ||||
|     } | ||||
|     async fn reset(&mut self, delay: &mut impl DelayUs) -> Result<(), RadioError> { | ||||
|         delay.delay_ms(10).await; | ||||
|         self.reset.set_low().map_err(|_| Reset)?; | ||||
|         delay.delay_ms(10).await; | ||||
|         self.reset.set_high().map_err(|_| Reset)?; | ||||
|         delay.delay_ms(10).await; | ||||
|         Ok(()) | ||||
|     } | ||||
|     async fn wait_on_busy(&mut self) -> Result<(), RadioError> { | ||||
|         Ok(()) | ||||
|     } | ||||
|     async fn await_irq(&mut self) -> Result<(), RadioError> { | ||||
|         self.irq.wait_for_high().await.map_err(|_| Irq) | ||||
|     } | ||||
|  | ||||
|     async fn enable_rf_switch_rx(&mut self) -> Result<(), RadioError> { | ||||
|         match &mut self.rf_switch_tx { | ||||
|             Some(pin) => pin.set_low().map_err(|_| RfSwitchTx)?, | ||||
|             None => (), | ||||
|         }; | ||||
|         match &mut self.rf_switch_rx { | ||||
|             Some(pin) => pin.set_high().map_err(|_| RfSwitchRx), | ||||
|             None => Ok(()), | ||||
|         } | ||||
|     } | ||||
|     async fn enable_rf_switch_tx(&mut self) -> Result<(), RadioError> { | ||||
|         match &mut self.rf_switch_rx { | ||||
|             Some(pin) => pin.set_low().map_err(|_| RfSwitchRx)?, | ||||
|             None => (), | ||||
|         }; | ||||
|         match &mut self.rf_switch_tx { | ||||
|             Some(pin) => pin.set_high().map_err(|_| RfSwitchTx), | ||||
|             None => Ok(()), | ||||
|         } | ||||
|     } | ||||
|     async fn disable_rf_switch(&mut self) -> Result<(), RadioError> { | ||||
|         match &mut self.rf_switch_rx { | ||||
|             Some(pin) => pin.set_low().map_err(|_| RfSwitchRx)?, | ||||
|             None => (), | ||||
|         }; | ||||
|         match &mut self.rf_switch_tx { | ||||
|             Some(pin) => pin.set_low().map_err(|_| RfSwitchTx), | ||||
|             None => Ok(()), | ||||
|         } | ||||
|     } | ||||
| } | ||||
|  | ||||
| /// Base for the InterfaceVariant implementation for a generic Sx126x LoRa board | ||||
| pub struct GenericSx126xInterfaceVariant<CTRL, WAIT> { | ||||
|     board_type: BoardType, | ||||
|     nss: CTRL, | ||||
|     reset: CTRL, | ||||
|     dio1: WAIT, | ||||
|     busy: WAIT, | ||||
|     rf_switch_rx: Option<CTRL>, | ||||
|     rf_switch_tx: Option<CTRL>, | ||||
| } | ||||
|  | ||||
| impl<CTRL, WAIT> GenericSx126xInterfaceVariant<CTRL, WAIT> | ||||
| where | ||||
|     CTRL: OutputPin, | ||||
|     WAIT: Wait, | ||||
| { | ||||
|     /// Create an InterfaceVariant instance for an nrf52840/sx1262 combination | ||||
|     pub fn new( | ||||
|         nss: CTRL, | ||||
|         reset: CTRL, | ||||
|         dio1: WAIT, | ||||
|         busy: WAIT, | ||||
|         rf_switch_rx: Option<CTRL>, | ||||
|         rf_switch_tx: Option<CTRL>, | ||||
|     ) -> Result<Self, RadioError> { | ||||
|         Ok(Self { | ||||
|             board_type: BoardType::Rak4631Sx1262, // updated when associated with a specific LoRa board | ||||
|             nss, | ||||
|             reset, | ||||
|             dio1, | ||||
|             busy, | ||||
|             rf_switch_rx, | ||||
|             rf_switch_tx, | ||||
|         }) | ||||
|     } | ||||
| } | ||||
|  | ||||
| impl<CTRL, WAIT> InterfaceVariant for GenericSx126xInterfaceVariant<CTRL, WAIT> | ||||
| where | ||||
|     CTRL: OutputPin, | ||||
|     WAIT: Wait, | ||||
| { | ||||
|     fn set_board_type(&mut self, board_type: BoardType) { | ||||
|         self.board_type = board_type; | ||||
|     } | ||||
|     async fn set_nss_low(&mut self) -> Result<(), RadioError> { | ||||
|         self.nss.set_low().map_err(|_| NSS) | ||||
|     } | ||||
|     async fn set_nss_high(&mut self) -> Result<(), RadioError> { | ||||
|         self.nss.set_high().map_err(|_| NSS) | ||||
|     } | ||||
|     async fn reset(&mut self, delay: &mut impl DelayUs) -> Result<(), RadioError> { | ||||
|         delay.delay_ms(10).await; | ||||
|         self.reset.set_low().map_err(|_| Reset)?; | ||||
|         delay.delay_ms(20).await; | ||||
|         self.reset.set_high().map_err(|_| Reset)?; | ||||
|         delay.delay_ms(10).await; | ||||
|         Ok(()) | ||||
|     } | ||||
|     async fn wait_on_busy(&mut self) -> Result<(), RadioError> { | ||||
|         self.busy.wait_for_low().await.map_err(|_| Busy) | ||||
|     } | ||||
|     async fn await_irq(&mut self) -> Result<(), RadioError> { | ||||
|         if self.board_type != BoardType::RpPicoWaveshareSx1262 { | ||||
|             self.dio1.wait_for_high().await.map_err(|_| DIO1)?; | ||||
|         } else { | ||||
|             self.dio1.wait_for_rising_edge().await.map_err(|_| DIO1)?; | ||||
|         } | ||||
|         Ok(()) | ||||
|     } | ||||
|  | ||||
|     async fn enable_rf_switch_rx(&mut self) -> Result<(), RadioError> { | ||||
|         match &mut self.rf_switch_tx { | ||||
|             Some(pin) => pin.set_low().map_err(|_| RfSwitchTx)?, | ||||
|             None => (), | ||||
|         }; | ||||
|         match &mut self.rf_switch_rx { | ||||
|             Some(pin) => pin.set_high().map_err(|_| RfSwitchRx), | ||||
|             None => Ok(()), | ||||
|         } | ||||
|     } | ||||
|     async fn enable_rf_switch_tx(&mut self) -> Result<(), RadioError> { | ||||
|         match &mut self.rf_switch_rx { | ||||
|             Some(pin) => pin.set_low().map_err(|_| RfSwitchRx)?, | ||||
|             None => (), | ||||
|         }; | ||||
|         match &mut self.rf_switch_tx { | ||||
|             Some(pin) => pin.set_high().map_err(|_| RfSwitchTx), | ||||
|             None => Ok(()), | ||||
|         } | ||||
|     } | ||||
|     async fn disable_rf_switch(&mut self) -> Result<(), RadioError> { | ||||
|         match &mut self.rf_switch_rx { | ||||
|             Some(pin) => pin.set_low().map_err(|_| RfSwitchRx)?, | ||||
|             None => (), | ||||
|         }; | ||||
|         match &mut self.rf_switch_tx { | ||||
|             Some(pin) => pin.set_low().map_err(|_| RfSwitchTx), | ||||
|             None => Ok(()), | ||||
|         } | ||||
|     } | ||||
| } | ||||
| @@ -1,23 +1,40 @@ | ||||
| #![no_std] | ||||
| #![feature(type_alias_impl_trait)] | ||||
| #![feature(generic_associated_types)] | ||||
| //! embassy-lora is a collection of async radio drivers that integrate with the lorawan-device | ||||
| //! crate's async LoRaWAN MAC implementation. | ||||
| #![feature(async_fn_in_trait, impl_trait_projections)] | ||||
| #![allow(incomplete_features)] | ||||
| //! embassy-lora holds LoRa-specific functionality. | ||||
|  | ||||
| pub(crate) mod fmt; | ||||
|  | ||||
| #[cfg(feature = "stm32wl")] | ||||
| pub mod stm32wl; | ||||
| #[cfg(feature = "sx127x")] | ||||
| pub mod sx127x; | ||||
| /// interface variants required by the external lora physical layer crate (lora-phy) | ||||
| pub mod iv; | ||||
|  | ||||
| #[cfg(feature = "time")] | ||||
| use embassy_time::{Duration, Instant, Timer}; | ||||
|  | ||||
| /// A convenience timer to use with the LoRaWAN crate | ||||
| pub struct LoraTimer; | ||||
| #[cfg(feature = "time")] | ||||
| pub struct LoraTimer { | ||||
|     start: Instant, | ||||
| } | ||||
|  | ||||
| #[cfg(feature = "time")] | ||||
| impl LoraTimer { | ||||
|     pub fn new() -> Self { | ||||
|         Self { start: Instant::now() } | ||||
|     } | ||||
| } | ||||
|  | ||||
| #[cfg(feature = "time")] | ||||
| impl lorawan_device::async_device::radio::Timer for LoraTimer { | ||||
|     type DelayFuture<'m> = impl core::future::Future<Output = ()> + 'm; | ||||
|     fn delay_ms<'m>(&'m mut self, millis: u64) -> Self::DelayFuture<'m> { | ||||
|         embassy_time::Timer::after(embassy_time::Duration::from_millis(millis)) | ||||
|     fn reset(&mut self) { | ||||
|         self.start = Instant::now(); | ||||
|     } | ||||
|  | ||||
|     async fn at(&mut self, millis: u64) { | ||||
|         Timer::at(self.start + Duration::from_millis(millis)).await | ||||
|     } | ||||
|  | ||||
|     async fn delay_ms(&mut self, millis: u64) { | ||||
|         Timer::after(Duration::from_millis(millis)).await | ||||
|     } | ||||
| } | ||||
|   | ||||
| @@ -1,339 +0,0 @@ | ||||
| //! A radio driver integration for the radio found on STM32WL family devices. | ||||
| use core::future::Future; | ||||
| use core::mem::MaybeUninit; | ||||
|  | ||||
| use embassy_hal_common::{into_ref, PeripheralRef}; | ||||
| use embassy_stm32::dma::NoDma; | ||||
| use embassy_stm32::gpio::{AnyPin, Output}; | ||||
| use embassy_stm32::interrupt::{InterruptExt, SUBGHZ_RADIO}; | ||||
| use embassy_stm32::subghz::{ | ||||
|     CalibrateImage, CfgIrq, CodingRate, Error, HeaderType, Irq, LoRaBandwidth, LoRaModParams, LoRaPacketParams, | ||||
|     LoRaSyncWord, Ocp, PaConfig, PaSel, PacketType, RampTime, RegMode, RfFreq, SpreadingFactor as SF, StandbyClk, | ||||
|     Status, SubGhz, TcxoMode, TcxoTrim, Timeout, TxParams, | ||||
| }; | ||||
| use embassy_stm32::Peripheral; | ||||
| use embassy_sync::signal::Signal; | ||||
| use lorawan_device::async_device::radio::{Bandwidth, PhyRxTx, RfConfig, RxQuality, SpreadingFactor, TxConfig}; | ||||
| use lorawan_device::async_device::Timings; | ||||
|  | ||||
| #[derive(Debug, Copy, Clone)] | ||||
| #[cfg_attr(feature = "defmt", derive(defmt::Format))] | ||||
| pub enum State { | ||||
|     Idle, | ||||
|     Txing, | ||||
|     Rxing, | ||||
| } | ||||
|  | ||||
| #[derive(Debug, Copy, Clone)] | ||||
| #[cfg_attr(feature = "defmt", derive(defmt::Format))] | ||||
| pub struct RadioError; | ||||
|  | ||||
| static IRQ: Signal<(Status, u16)> = Signal::new(); | ||||
|  | ||||
| struct StateInner<'d> { | ||||
|     radio: SubGhz<'d, NoDma, NoDma>, | ||||
|     switch: RadioSwitch<'d>, | ||||
| } | ||||
|  | ||||
| /// External state storage for the radio state | ||||
| pub struct SubGhzState<'a>(MaybeUninit<StateInner<'a>>); | ||||
| impl<'d> SubGhzState<'d> { | ||||
|     pub const fn new() -> Self { | ||||
|         Self(MaybeUninit::uninit()) | ||||
|     } | ||||
| } | ||||
|  | ||||
| /// The radio peripheral keeping the radio state and owning the radio IRQ. | ||||
| pub struct SubGhzRadio<'d> { | ||||
|     state: *mut StateInner<'d>, | ||||
|     _irq: PeripheralRef<'d, SUBGHZ_RADIO>, | ||||
| } | ||||
|  | ||||
| impl<'d> SubGhzRadio<'d> { | ||||
|     /// Create a new instance of a SubGhz radio for LoRaWAN. | ||||
|     /// | ||||
|     /// # Safety | ||||
|     /// Do not leak self or futures | ||||
|     pub unsafe fn new( | ||||
|         state: &'d mut SubGhzState<'d>, | ||||
|         radio: SubGhz<'d, NoDma, NoDma>, | ||||
|         switch: RadioSwitch<'d>, | ||||
|         irq: impl Peripheral<P = SUBGHZ_RADIO> + 'd, | ||||
|     ) -> Self { | ||||
|         into_ref!(irq); | ||||
|  | ||||
|         let mut inner = StateInner { radio, switch }; | ||||
|         inner.radio.reset(); | ||||
|  | ||||
|         let state_ptr = state.0.as_mut_ptr(); | ||||
|         state_ptr.write(inner); | ||||
|  | ||||
|         irq.disable(); | ||||
|         irq.set_handler(|p| { | ||||
|             // This is safe because we only get interrupts when configured for, so | ||||
|             // the radio will be awaiting on the signal at this point. If not, the ISR will | ||||
|             // anyway only adjust the state in the IRQ signal state. | ||||
|             let state = &mut *(p as *mut StateInner<'d>); | ||||
|             state.on_interrupt(); | ||||
|         }); | ||||
|         irq.set_handler_context(state_ptr as *mut ()); | ||||
|         irq.enable(); | ||||
|  | ||||
|         Self { | ||||
|             state: state_ptr, | ||||
|             _irq: irq, | ||||
|         } | ||||
|     } | ||||
| } | ||||
|  | ||||
| impl<'d> StateInner<'d> { | ||||
|     /// Configure radio settings in preparation for TX or RX | ||||
|     pub(crate) fn configure(&mut self) -> Result<(), RadioError> { | ||||
|         trace!("Configuring STM32WL SUBGHZ radio"); | ||||
|         self.radio.set_standby(StandbyClk::Rc)?; | ||||
|         let tcxo_mode = TcxoMode::new() | ||||
|             .set_txco_trim(TcxoTrim::Volts1pt7) | ||||
|             .set_timeout(Timeout::from_duration_sat(core::time::Duration::from_millis(40))); | ||||
|  | ||||
|         self.radio.set_tcxo_mode(&tcxo_mode)?; | ||||
|         self.radio.set_regulator_mode(RegMode::Ldo)?; | ||||
|  | ||||
|         self.radio.calibrate_image(CalibrateImage::ISM_863_870)?; | ||||
|  | ||||
|         self.radio.set_buffer_base_address(0, 0)?; | ||||
|  | ||||
|         self.radio | ||||
|             .set_pa_config(&PaConfig::new().set_pa_duty_cycle(0x1).set_hp_max(0x0).set_pa(PaSel::Lp))?; | ||||
|  | ||||
|         self.radio.set_pa_ocp(Ocp::Max140m)?; | ||||
|  | ||||
|         //        let tx_params = TxParams::LP_14.set_ramp_time(RampTime::Micros40); | ||||
|         self.radio | ||||
|             .set_tx_params(&TxParams::new().set_ramp_time(RampTime::Micros40).set_power(0x0A))?; | ||||
|  | ||||
|         self.radio.set_packet_type(PacketType::LoRa)?; | ||||
|         self.radio.set_lora_sync_word(LoRaSyncWord::Public)?; | ||||
|         trace!("Done initializing STM32WL SUBGHZ radio"); | ||||
|         Ok(()) | ||||
|     } | ||||
|  | ||||
|     /// Perform a transmission with the given parameters and payload. Returns any time adjustements needed form | ||||
|     /// the upcoming RX window start. | ||||
|     async fn do_tx(&mut self, config: TxConfig, buf: &[u8]) -> Result<u32, RadioError> { | ||||
|         //trace!("TX Request: {}", config); | ||||
|         trace!("TX START"); | ||||
|         self.switch.set_tx_lp(); | ||||
|         self.configure()?; | ||||
|  | ||||
|         self.radio | ||||
|             .set_rf_frequency(&RfFreq::from_frequency(config.rf.frequency))?; | ||||
|  | ||||
|         self.set_lora_mod_params(config.rf)?; | ||||
|  | ||||
|         let packet_params = LoRaPacketParams::new() | ||||
|             .set_preamble_len(8) | ||||
|             .set_header_type(HeaderType::Variable) | ||||
|             .set_payload_len(buf.len() as u8) | ||||
|             .set_crc_en(true) | ||||
|             .set_invert_iq(false); | ||||
|  | ||||
|         self.radio.set_lora_packet_params(&packet_params)?; | ||||
|  | ||||
|         let irq_cfg = CfgIrq::new() | ||||
|             .irq_enable_all(Irq::TxDone) | ||||
|             .irq_enable_all(Irq::RxDone) | ||||
|             .irq_enable_all(Irq::Timeout); | ||||
|         self.radio.set_irq_cfg(&irq_cfg)?; | ||||
|  | ||||
|         self.radio.set_buffer_base_address(0, 0)?; | ||||
|         self.radio.write_buffer(0, buf)?; | ||||
|  | ||||
|         self.radio.set_tx(Timeout::DISABLED)?; | ||||
|  | ||||
|         loop { | ||||
|             let (_status, irq_status) = IRQ.wait().await; | ||||
|             IRQ.reset(); | ||||
|  | ||||
|             if irq_status & Irq::TxDone.mask() != 0 { | ||||
|                 let stats = self.radio.lora_stats()?; | ||||
|                 let (status, error_mask) = self.radio.op_error()?; | ||||
|                 trace!( | ||||
|                     "TX done. Stats: {:?}. OP error: {:?}, mask {:?}", | ||||
|                     stats, | ||||
|                     status, | ||||
|                     error_mask | ||||
|                 ); | ||||
|  | ||||
|                 return Ok(0); | ||||
|             } else if irq_status & Irq::Timeout.mask() != 0 { | ||||
|                 trace!("TX timeout"); | ||||
|                 return Err(RadioError); | ||||
|             } | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     fn set_lora_mod_params(&mut self, config: RfConfig) -> Result<(), Error> { | ||||
|         let mod_params = LoRaModParams::new() | ||||
|             .set_sf(convert_spreading_factor(config.spreading_factor)) | ||||
|             .set_bw(convert_bandwidth(config.bandwidth)) | ||||
|             .set_cr(CodingRate::Cr45) | ||||
|             .set_ldro_en(true); | ||||
|         self.radio.set_lora_mod_params(&mod_params) | ||||
|     } | ||||
|  | ||||
|     /// Perform a radio receive operation with the radio config and receive buffer. The receive buffer must | ||||
|     /// be able to hold a single LoRaWAN packet. | ||||
|     async fn do_rx(&mut self, config: RfConfig, buf: &mut [u8]) -> Result<(usize, RxQuality), RadioError> { | ||||
|         assert!(buf.len() >= 255); | ||||
|         trace!("RX START"); | ||||
|         // trace!("Starting RX: {}", config); | ||||
|         self.switch.set_rx(); | ||||
|         self.configure()?; | ||||
|  | ||||
|         self.radio.set_rf_frequency(&RfFreq::from_frequency(config.frequency))?; | ||||
|  | ||||
|         self.set_lora_mod_params(config)?; | ||||
|  | ||||
|         let packet_params = LoRaPacketParams::new() | ||||
|             .set_preamble_len(8) | ||||
|             .set_header_type(HeaderType::Variable) | ||||
|             .set_payload_len(0xFF) | ||||
|             .set_crc_en(true) | ||||
|             .set_invert_iq(true); | ||||
|         self.radio.set_lora_packet_params(&packet_params)?; | ||||
|  | ||||
|         let irq_cfg = CfgIrq::new() | ||||
|             .irq_enable_all(Irq::RxDone) | ||||
|             .irq_enable_all(Irq::PreambleDetected) | ||||
|             .irq_enable_all(Irq::HeaderErr) | ||||
|             .irq_enable_all(Irq::Timeout) | ||||
|             .irq_enable_all(Irq::Err); | ||||
|         self.radio.set_irq_cfg(&irq_cfg)?; | ||||
|  | ||||
|         self.radio.set_rx(Timeout::DISABLED)?; | ||||
|         trace!("RX started"); | ||||
|  | ||||
|         loop { | ||||
|             let (status, irq_status) = IRQ.wait().await; | ||||
|             IRQ.reset(); | ||||
|             trace!("RX IRQ {:?}, {:?}", status, irq_status); | ||||
|             if irq_status & Irq::RxDone.mask() != 0 { | ||||
|                 let (status, len, ptr) = self.radio.rx_buffer_status()?; | ||||
|  | ||||
|                 let packet_status = self.radio.lora_packet_status()?; | ||||
|                 let rssi = packet_status.rssi_pkt().to_integer(); | ||||
|                 let snr = packet_status.snr_pkt().to_integer(); | ||||
|                 trace!( | ||||
|                     "RX done. Received {} bytes. RX status: {:?}. Pkt status: {:?}", | ||||
|                     len, | ||||
|                     status.cmd(), | ||||
|                     packet_status, | ||||
|                 ); | ||||
|                 self.radio.read_buffer(ptr, &mut buf[..len as usize])?; | ||||
|                 self.radio.set_standby(StandbyClk::Rc)?; | ||||
|                 return Ok((len as usize, RxQuality::new(rssi, snr as i8))); | ||||
|             } else if irq_status & (Irq::Timeout.mask() | Irq::TxDone.mask()) != 0 { | ||||
|                 return Err(RadioError); | ||||
|             } | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     /// Read interrupt status and store in global signal | ||||
|     fn on_interrupt(&mut self) { | ||||
|         let (status, irq_status) = self.radio.irq_status().expect("error getting irq status"); | ||||
|         self.radio | ||||
|             .clear_irq_status(irq_status) | ||||
|             .expect("error clearing irq status"); | ||||
|         if irq_status & Irq::PreambleDetected.mask() != 0 { | ||||
|             trace!("Preamble detected, ignoring"); | ||||
|         } else { | ||||
|             IRQ.signal((status, irq_status)); | ||||
|         } | ||||
|     } | ||||
| } | ||||
|  | ||||
| impl PhyRxTx for SubGhzRadio<'static> { | ||||
|     type PhyError = RadioError; | ||||
|  | ||||
|     type TxFuture<'m> = impl Future<Output = Result<u32, Self::PhyError>> + 'm; | ||||
|     fn tx<'m>(&'m mut self, config: TxConfig, buf: &'m [u8]) -> Self::TxFuture<'m> { | ||||
|         async move { | ||||
|             let inner = unsafe { &mut *self.state }; | ||||
|             inner.do_tx(config, buf).await | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     type RxFuture<'m> = impl Future<Output = Result<(usize, RxQuality), Self::PhyError>> + 'm; | ||||
|     fn rx<'m>(&'m mut self, config: RfConfig, buf: &'m mut [u8]) -> Self::RxFuture<'m> { | ||||
|         async move { | ||||
|             let inner = unsafe { &mut *self.state }; | ||||
|             inner.do_rx(config, buf).await | ||||
|         } | ||||
|     } | ||||
| } | ||||
|  | ||||
| impl From<embassy_stm32::spi::Error> for RadioError { | ||||
|     fn from(_: embassy_stm32::spi::Error) -> Self { | ||||
|         RadioError | ||||
|     } | ||||
| } | ||||
|  | ||||
| impl<'d> Timings for SubGhzRadio<'d> { | ||||
|     fn get_rx_window_offset_ms(&self) -> i32 { | ||||
|         -200 | ||||
|     } | ||||
|     fn get_rx_window_duration_ms(&self) -> u32 { | ||||
|         800 | ||||
|     } | ||||
| } | ||||
|  | ||||
| /// Represents the radio switch found on STM32WL based boards, used to control the radio for transmission or reception. | ||||
| pub struct RadioSwitch<'d> { | ||||
|     ctrl1: Output<'d, AnyPin>, | ||||
|     ctrl2: Output<'d, AnyPin>, | ||||
|     ctrl3: Output<'d, AnyPin>, | ||||
| } | ||||
|  | ||||
| impl<'d> RadioSwitch<'d> { | ||||
|     pub fn new(ctrl1: Output<'d, AnyPin>, ctrl2: Output<'d, AnyPin>, ctrl3: Output<'d, AnyPin>) -> Self { | ||||
|         Self { ctrl1, ctrl2, ctrl3 } | ||||
|     } | ||||
|  | ||||
|     pub(crate) fn set_rx(&mut self) { | ||||
|         self.ctrl1.set_high(); | ||||
|         self.ctrl2.set_low(); | ||||
|         self.ctrl3.set_high(); | ||||
|     } | ||||
|  | ||||
|     pub(crate) fn set_tx_lp(&mut self) { | ||||
|         self.ctrl1.set_high(); | ||||
|         self.ctrl2.set_high(); | ||||
|         self.ctrl3.set_high(); | ||||
|     } | ||||
|  | ||||
|     #[allow(dead_code)] | ||||
|     pub(crate) fn set_tx_hp(&mut self) { | ||||
|         self.ctrl2.set_high(); | ||||
|         self.ctrl1.set_low(); | ||||
|         self.ctrl3.set_high(); | ||||
|     } | ||||
| } | ||||
|  | ||||
| fn convert_spreading_factor(sf: SpreadingFactor) -> SF { | ||||
|     match sf { | ||||
|         SpreadingFactor::_7 => SF::Sf7, | ||||
|         SpreadingFactor::_8 => SF::Sf8, | ||||
|         SpreadingFactor::_9 => SF::Sf9, | ||||
|         SpreadingFactor::_10 => SF::Sf10, | ||||
|         SpreadingFactor::_11 => SF::Sf11, | ||||
|         SpreadingFactor::_12 => SF::Sf12, | ||||
|     } | ||||
| } | ||||
|  | ||||
| fn convert_bandwidth(bw: Bandwidth) -> LoRaBandwidth { | ||||
|     match bw { | ||||
|         Bandwidth::_125KHz => LoRaBandwidth::Bw125, | ||||
|         Bandwidth::_250KHz => LoRaBandwidth::Bw250, | ||||
|         Bandwidth::_500KHz => LoRaBandwidth::Bw500, | ||||
|     } | ||||
| } | ||||
| @@ -1,217 +0,0 @@ | ||||
| use core::future::Future; | ||||
|  | ||||
| use embedded_hal::digital::v2::OutputPin; | ||||
| use embedded_hal_async::digital::Wait; | ||||
| use embedded_hal_async::spi::*; | ||||
| use lorawan_device::async_device::radio::{Bandwidth, PhyRxTx, RfConfig, RxQuality, SpreadingFactor, TxConfig}; | ||||
| use lorawan_device::async_device::Timings; | ||||
|  | ||||
| mod sx127x_lora; | ||||
| use sx127x_lora::{Error as RadioError, LoRa, RadioMode, IRQ}; | ||||
|  | ||||
| /// Trait representing a radio switch for boards using the Sx127x radio. One some | ||||
| /// boards, this will be a dummy implementation that does nothing. | ||||
| pub trait RadioSwitch { | ||||
|     fn set_tx(&mut self); | ||||
|     fn set_rx(&mut self); | ||||
| } | ||||
|  | ||||
| /// Semtech Sx127x radio peripheral | ||||
| pub struct Sx127xRadio<SPI, CS, RESET, E, I, RFS> | ||||
| where | ||||
|     SPI: SpiBus<u8, Error = E> + 'static, | ||||
|     E: 'static, | ||||
|     CS: OutputPin + 'static, | ||||
|     RESET: OutputPin + 'static, | ||||
|     I: Wait + 'static, | ||||
|     RFS: RadioSwitch + 'static, | ||||
| { | ||||
|     radio: LoRa<SPI, CS, RESET>, | ||||
|     rfs: RFS, | ||||
|     irq: I, | ||||
| } | ||||
|  | ||||
| #[derive(Debug, Copy, Clone)] | ||||
| #[cfg_attr(feature = "defmt", derive(defmt::Format))] | ||||
| pub enum State { | ||||
|     Idle, | ||||
|     Txing, | ||||
|     Rxing, | ||||
| } | ||||
|  | ||||
| impl<SPI, CS, RESET, E, I, RFS> Sx127xRadio<SPI, CS, RESET, E, I, RFS> | ||||
| where | ||||
|     SPI: SpiBus<u8, Error = E> + 'static, | ||||
|     CS: OutputPin + 'static, | ||||
|     RESET: OutputPin + 'static, | ||||
|     I: Wait + 'static, | ||||
|     RFS: RadioSwitch + 'static, | ||||
|     E: 'static, | ||||
| { | ||||
|     pub async fn new( | ||||
|         spi: SPI, | ||||
|         cs: CS, | ||||
|         reset: RESET, | ||||
|         irq: I, | ||||
|         rfs: RFS, | ||||
|     ) -> Result<Self, RadioError<E, CS::Error, RESET::Error>> { | ||||
|         let mut radio = LoRa::new(spi, cs, reset); | ||||
|         radio.reset().await?; | ||||
|         Ok(Self { radio, irq, rfs }) | ||||
|     } | ||||
| } | ||||
|  | ||||
| impl<SPI, CS, RESET, E, I, RFS> Timings for Sx127xRadio<SPI, CS, RESET, E, I, RFS> | ||||
| where | ||||
|     SPI: SpiBus<u8, Error = E> + 'static, | ||||
|     CS: OutputPin + 'static, | ||||
|     RESET: OutputPin + 'static, | ||||
|     I: Wait + 'static, | ||||
|     RFS: RadioSwitch + 'static, | ||||
| { | ||||
|     fn get_rx_window_offset_ms(&self) -> i32 { | ||||
|         -500 | ||||
|     } | ||||
|     fn get_rx_window_duration_ms(&self) -> u32 { | ||||
|         800 | ||||
|     } | ||||
| } | ||||
|  | ||||
| impl<SPI, CS, RESET, E, I, RFS> PhyRxTx for Sx127xRadio<SPI, CS, RESET, E, I, RFS> | ||||
| where | ||||
|     SPI: SpiBus<u8, Error = E> + 'static, | ||||
|     CS: OutputPin + 'static, | ||||
|     E: 'static, | ||||
|     RESET: OutputPin + 'static, | ||||
|     I: Wait + 'static, | ||||
|     RFS: RadioSwitch + 'static, | ||||
| { | ||||
|     type PhyError = Sx127xError; | ||||
|  | ||||
|     type TxFuture<'m> = impl Future<Output = Result<u32, Self::PhyError>> + 'm | ||||
|     where | ||||
|         SPI: 'm, | ||||
|         CS: 'm, | ||||
|         RESET: 'm, | ||||
|         E: 'm, | ||||
|         I: 'm, | ||||
|         RFS: 'm; | ||||
|  | ||||
|     fn tx<'m>(&'m mut self, config: TxConfig, buf: &'m [u8]) -> Self::TxFuture<'m> { | ||||
|         trace!("TX START"); | ||||
|         async move { | ||||
|             self.radio.set_mode(RadioMode::Stdby).await.ok().unwrap(); | ||||
|             self.rfs.set_tx(); | ||||
|             self.radio.set_tx_power(14, 0).await?; | ||||
|             self.radio.set_frequency(config.rf.frequency).await?; | ||||
|             // TODO: Modify radio to support other coding rates | ||||
|             self.radio.set_coding_rate_4(5).await?; | ||||
|             self.radio | ||||
|                 .set_signal_bandwidth(bandwidth_to_i64(config.rf.bandwidth)) | ||||
|                 .await?; | ||||
|             self.radio | ||||
|                 .set_spreading_factor(spreading_factor_to_u8(config.rf.spreading_factor)) | ||||
|                 .await?; | ||||
|  | ||||
|             self.radio.set_preamble_length(8).await?; | ||||
|             self.radio.set_lora_pa_ramp().await?; | ||||
|             self.radio.set_lora_sync_word().await?; | ||||
|             self.radio.set_invert_iq(false).await?; | ||||
|             self.radio.set_crc(true).await?; | ||||
|  | ||||
|             self.radio.set_dio0_tx_done().await?; | ||||
|  | ||||
|             self.radio.transmit_start(buf).await?; | ||||
|  | ||||
|             loop { | ||||
|                 self.irq.wait_for_rising_edge().await.unwrap(); | ||||
|                 self.radio.set_mode(RadioMode::Stdby).await.ok().unwrap(); | ||||
|                 let irq = self.radio.clear_irq().await.ok().unwrap(); | ||||
|                 if (irq & IRQ::IrqTxDoneMask.addr()) != 0 { | ||||
|                     trace!("TX DONE"); | ||||
|                     return Ok(0); | ||||
|                 } | ||||
|             } | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     type RxFuture<'m> = impl Future<Output = Result<(usize, RxQuality), Self::PhyError>> + 'm | ||||
|     where | ||||
|         SPI: 'm, | ||||
|         CS: 'm, | ||||
|         RESET: 'm, | ||||
|         E: 'm, | ||||
|         I: 'm, | ||||
|         RFS: 'm; | ||||
|  | ||||
|     fn rx<'m>(&'m mut self, config: RfConfig, buf: &'m mut [u8]) -> Self::RxFuture<'m> { | ||||
|         trace!("RX START"); | ||||
|         async move { | ||||
|             self.rfs.set_rx(); | ||||
|             self.radio.reset_payload_length().await?; | ||||
|             self.radio.set_frequency(config.frequency).await?; | ||||
|             // TODO: Modify radio to support other coding rates | ||||
|             self.radio.set_coding_rate_4(5).await?; | ||||
|             self.radio | ||||
|                 .set_signal_bandwidth(bandwidth_to_i64(config.bandwidth)) | ||||
|                 .await?; | ||||
|             self.radio | ||||
|                 .set_spreading_factor(spreading_factor_to_u8(config.spreading_factor)) | ||||
|                 .await?; | ||||
|  | ||||
|             self.radio.set_preamble_length(8).await?; | ||||
|             self.radio.set_lora_sync_word().await?; | ||||
|             self.radio.set_invert_iq(true).await?; | ||||
|             self.radio.set_crc(true).await?; | ||||
|  | ||||
|             self.radio.set_dio0_rx_done().await?; | ||||
|             self.radio.set_mode(RadioMode::RxContinuous).await?; | ||||
|  | ||||
|             loop { | ||||
|                 self.irq.wait_for_rising_edge().await.unwrap(); | ||||
|                 self.radio.set_mode(RadioMode::Stdby).await.ok().unwrap(); | ||||
|                 let irq = self.radio.clear_irq().await.ok().unwrap(); | ||||
|                 if (irq & IRQ::IrqRxDoneMask.addr()) != 0 { | ||||
|                     let rssi = self.radio.get_packet_rssi().await.unwrap_or(0) as i16; | ||||
|                     let snr = self.radio.get_packet_snr().await.unwrap_or(0.0) as i8; | ||||
|                     let response = if let Ok(size) = self.radio.read_packet_size().await { | ||||
|                         self.radio.read_packet(buf).await?; | ||||
|                         Ok((size, RxQuality::new(rssi, snr))) | ||||
|                     } else { | ||||
|                         Ok((0, RxQuality::new(rssi, snr))) | ||||
|                     }; | ||||
|                     trace!("RX DONE"); | ||||
|                     return response; | ||||
|                 } | ||||
|             } | ||||
|         } | ||||
|     } | ||||
| } | ||||
|  | ||||
| #[cfg_attr(feature = "defmt", derive(defmt::Format))] | ||||
| pub struct Sx127xError; | ||||
|  | ||||
| impl<A, B, C> From<sx127x_lora::Error<A, B, C>> for Sx127xError { | ||||
|     fn from(_: sx127x_lora::Error<A, B, C>) -> Self { | ||||
|         Sx127xError | ||||
|     } | ||||
| } | ||||
|  | ||||
| fn spreading_factor_to_u8(sf: SpreadingFactor) -> u8 { | ||||
|     match sf { | ||||
|         SpreadingFactor::_7 => 7, | ||||
|         SpreadingFactor::_8 => 8, | ||||
|         SpreadingFactor::_9 => 9, | ||||
|         SpreadingFactor::_10 => 10, | ||||
|         SpreadingFactor::_11 => 11, | ||||
|         SpreadingFactor::_12 => 12, | ||||
|     } | ||||
| } | ||||
|  | ||||
| fn bandwidth_to_i64(bw: Bandwidth) -> i64 { | ||||
|     match bw { | ||||
|         Bandwidth::_125KHz => 125_000, | ||||
|         Bandwidth::_250KHz => 250_000, | ||||
|         Bandwidth::_500KHz => 500_000, | ||||
|     } | ||||
| } | ||||
| @@ -1,539 +0,0 @@ | ||||
| // Copyright Charles Wade (https://github.com/mr-glt/sx127x_lora). Licensed under the Apache 2.0 | ||||
| // license | ||||
| // | ||||
| // Modifications made to make the driver work with the rust-lorawan link layer. | ||||
|  | ||||
| #![allow(dead_code)] | ||||
|  | ||||
| use bit_field::BitField; | ||||
| use embassy_time::{Duration, Timer}; | ||||
| use embedded_hal::digital::v2::OutputPin; | ||||
| use embedded_hal_async::spi::SpiBus; | ||||
|  | ||||
| mod register; | ||||
| pub use self::register::IRQ; | ||||
| use self::register::{PaConfig, Register}; | ||||
|  | ||||
| /// Provides high-level access to Semtech SX1276/77/78/79 based boards connected to a Raspberry Pi | ||||
| pub struct LoRa<SPI, CS, RESET> { | ||||
|     spi: SPI, | ||||
|     cs: CS, | ||||
|     reset: RESET, | ||||
|     pub explicit_header: bool, | ||||
|     pub mode: RadioMode, | ||||
| } | ||||
|  | ||||
| #[allow(clippy::upper_case_acronyms)] | ||||
| #[derive(Debug)] | ||||
| #[cfg_attr(feature = "defmt", derive(defmt::Format))] | ||||
| pub enum Error<SPI, CS, RESET> { | ||||
|     Uninformative, | ||||
|     VersionMismatch(u8), | ||||
|     CS(CS), | ||||
|     Reset(RESET), | ||||
|     SPI(SPI), | ||||
|     Transmitting, | ||||
| } | ||||
|  | ||||
| use Error::*; | ||||
|  | ||||
| use super::sx127x_lora::register::{FskDataModulationShaping, FskRampUpRamDown}; | ||||
|  | ||||
| #[cfg(not(feature = "version_0x09"))] | ||||
| const VERSION_CHECK: u8 = 0x12; | ||||
|  | ||||
| #[cfg(feature = "version_0x09")] | ||||
| const VERSION_CHECK: u8 = 0x09; | ||||
|  | ||||
| impl<SPI, CS, RESET, E> LoRa<SPI, CS, RESET> | ||||
| where | ||||
|     SPI: SpiBus<u8, Error = E>, | ||||
|     CS: OutputPin, | ||||
|     RESET: OutputPin, | ||||
| { | ||||
|     /// Builds and returns a new instance of the radio. Only one instance of the radio should exist at a time. | ||||
|     /// This also preforms a hardware reset of the module and then puts it in standby. | ||||
|     pub fn new(spi: SPI, cs: CS, reset: RESET) -> Self { | ||||
|         Self { | ||||
|             spi, | ||||
|             cs, | ||||
|             reset, | ||||
|             explicit_header: true, | ||||
|             mode: RadioMode::Sleep, | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     pub async fn reset(&mut self) -> Result<(), Error<E, CS::Error, RESET::Error>> { | ||||
|         self.reset.set_low().map_err(Reset)?; | ||||
|         Timer::after(Duration::from_millis(10)).await; | ||||
|         self.reset.set_high().map_err(Reset)?; | ||||
|         Timer::after(Duration::from_millis(10)).await; | ||||
|         let version = self.read_register(Register::RegVersion.addr()).await?; | ||||
|         if version == VERSION_CHECK { | ||||
|             self.set_mode(RadioMode::Sleep).await?; | ||||
|             self.write_register(Register::RegFifoTxBaseAddr.addr(), 0).await?; | ||||
|             self.write_register(Register::RegFifoRxBaseAddr.addr(), 0).await?; | ||||
|             let lna = self.read_register(Register::RegLna.addr()).await?; | ||||
|             self.write_register(Register::RegLna.addr(), lna | 0x03).await?; | ||||
|             self.write_register(Register::RegModemConfig3.addr(), 0x04).await?; | ||||
|             self.set_tcxo(true).await?; | ||||
|             self.set_mode(RadioMode::Stdby).await?; | ||||
|             self.cs.set_high().map_err(CS)?; | ||||
|             Ok(()) | ||||
|         } else { | ||||
|             Err(Error::VersionMismatch(version)) | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     pub async fn set_dio0_tx_done(&mut self) -> Result<(), Error<E, CS::Error, RESET::Error>> { | ||||
|         self.write_register(Register::RegIrqFlagsMask.addr(), 0b1111_0111) | ||||
|             .await?; | ||||
|         let mapping = self.read_register(Register::RegDioMapping1.addr()).await?; | ||||
|         self.write_register(Register::RegDioMapping1.addr(), (mapping & 0x3F) | 0x40) | ||||
|             .await | ||||
|     } | ||||
|  | ||||
|     pub async fn set_dio0_rx_done(&mut self) -> Result<(), Error<E, CS::Error, RESET::Error>> { | ||||
|         self.write_register(Register::RegIrqFlagsMask.addr(), 0b0001_1111) | ||||
|             .await?; | ||||
|         let mapping = self.read_register(Register::RegDioMapping1.addr()).await?; | ||||
|         self.write_register(Register::RegDioMapping1.addr(), mapping & 0x3F) | ||||
|             .await | ||||
|     } | ||||
|  | ||||
|     pub async fn transmit_start(&mut self, buffer: &[u8]) -> Result<(), Error<E, CS::Error, RESET::Error>> { | ||||
|         assert!(buffer.len() < 255); | ||||
|         if self.transmitting().await? { | ||||
|             //trace!("ALREADY TRANSMNITTING"); | ||||
|             Err(Transmitting) | ||||
|         } else { | ||||
|             self.set_mode(RadioMode::Stdby).await?; | ||||
|             if self.explicit_header { | ||||
|                 self.set_explicit_header_mode().await?; | ||||
|             } else { | ||||
|                 self.set_implicit_header_mode().await?; | ||||
|             } | ||||
|  | ||||
|             self.write_register(Register::RegIrqFlags.addr(), 0).await?; | ||||
|             self.write_register(Register::RegFifoAddrPtr.addr(), 0).await?; | ||||
|             self.write_register(Register::RegPayloadLength.addr(), 0).await?; | ||||
|             for byte in buffer.iter() { | ||||
|                 self.write_register(Register::RegFifo.addr(), *byte).await?; | ||||
|             } | ||||
|             self.write_register(Register::RegPayloadLength.addr(), buffer.len() as u8) | ||||
|                 .await?; | ||||
|             self.set_mode(RadioMode::Tx).await?; | ||||
|             Ok(()) | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     pub async fn packet_ready(&mut self) -> Result<bool, Error<E, CS::Error, RESET::Error>> { | ||||
|         Ok(self.read_register(Register::RegIrqFlags.addr()).await?.get_bit(6)) | ||||
|     } | ||||
|  | ||||
|     pub async fn irq_flags_mask(&mut self) -> Result<u8, Error<E, CS::Error, RESET::Error>> { | ||||
|         Ok(self.read_register(Register::RegIrqFlagsMask.addr()).await? as u8) | ||||
|     } | ||||
|  | ||||
|     pub async fn irq_flags(&mut self) -> Result<u8, Error<E, CS::Error, RESET::Error>> { | ||||
|         Ok(self.read_register(Register::RegIrqFlags.addr()).await? as u8) | ||||
|     } | ||||
|  | ||||
|     pub async fn read_packet_size(&mut self) -> Result<usize, Error<E, CS::Error, RESET::Error>> { | ||||
|         let size = self.read_register(Register::RegRxNbBytes.addr()).await?; | ||||
|         Ok(size as usize) | ||||
|     } | ||||
|  | ||||
|     /// Returns the contents of the fifo as a fixed 255 u8 array. This should only be called is there is a | ||||
|     /// new packet ready to be read. | ||||
|     pub async fn read_packet(&mut self, buffer: &mut [u8]) -> Result<(), Error<E, CS::Error, RESET::Error>> { | ||||
|         self.clear_irq().await?; | ||||
|         let size = self.read_register(Register::RegRxNbBytes.addr()).await?; | ||||
|         assert!(size as usize <= buffer.len()); | ||||
|         let fifo_addr = self.read_register(Register::RegFifoRxCurrentAddr.addr()).await?; | ||||
|         self.write_register(Register::RegFifoAddrPtr.addr(), fifo_addr).await?; | ||||
|         for i in 0..size { | ||||
|             let byte = self.read_register(Register::RegFifo.addr()).await?; | ||||
|             buffer[i as usize] = byte; | ||||
|         } | ||||
|         self.write_register(Register::RegFifoAddrPtr.addr(), 0).await?; | ||||
|         Ok(()) | ||||
|     } | ||||
|  | ||||
|     /// Returns true if the radio is currently transmitting a packet. | ||||
|     pub async fn transmitting(&mut self) -> Result<bool, Error<E, CS::Error, RESET::Error>> { | ||||
|         if (self.read_register(Register::RegOpMode.addr()).await?) & RadioMode::Tx.addr() == RadioMode::Tx.addr() { | ||||
|             Ok(true) | ||||
|         } else { | ||||
|             if (self.read_register(Register::RegIrqFlags.addr()).await? & IRQ::IrqTxDoneMask.addr()) == 1 { | ||||
|                 self.write_register(Register::RegIrqFlags.addr(), IRQ::IrqTxDoneMask.addr()) | ||||
|                     .await?; | ||||
|             } | ||||
|             Ok(false) | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     /// Clears the radio's IRQ registers. | ||||
|     pub async fn clear_irq(&mut self) -> Result<u8, Error<E, CS::Error, RESET::Error>> { | ||||
|         let irq_flags = self.read_register(Register::RegIrqFlags.addr()).await?; | ||||
|         self.write_register(Register::RegIrqFlags.addr(), 0xFF).await?; | ||||
|         Ok(irq_flags) | ||||
|     } | ||||
|  | ||||
|     /// Sets the transmit power and pin. Levels can range from 0-14 when the output | ||||
|     /// pin = 0(RFO), and form 0-20 when output pin = 1(PaBoost). Power is in dB. | ||||
|     /// Default value is `17`. | ||||
|     pub async fn set_tx_power( | ||||
|         &mut self, | ||||
|         mut level: i32, | ||||
|         output_pin: u8, | ||||
|     ) -> Result<(), Error<E, CS::Error, RESET::Error>> { | ||||
|         if PaConfig::PaOutputRfoPin.addr() == output_pin { | ||||
|             // RFO | ||||
|             if level < 0 { | ||||
|                 level = 0; | ||||
|             } else if level > 14 { | ||||
|                 level = 14; | ||||
|             } | ||||
|             self.write_register(Register::RegPaConfig.addr(), (0x70 | level) as u8) | ||||
|                 .await | ||||
|         } else { | ||||
|             // PA BOOST | ||||
|             if level > 17 { | ||||
|                 if level > 20 { | ||||
|                     level = 20; | ||||
|                 } | ||||
|                 // subtract 3 from level, so 18 - 20 maps to 15 - 17 | ||||
|                 level -= 3; | ||||
|  | ||||
|                 // High Power +20 dBm Operation (Semtech SX1276/77/78/79 5.4.3.) | ||||
|                 self.write_register(Register::RegPaDac.addr(), 0x87).await?; | ||||
|                 self.set_ocp(140).await?; | ||||
|             } else { | ||||
|                 if level < 2 { | ||||
|                     level = 2; | ||||
|                 } | ||||
|                 //Default value PA_HF/LF or +17dBm | ||||
|                 self.write_register(Register::RegPaDac.addr(), 0x84).await?; | ||||
|                 self.set_ocp(100).await?; | ||||
|             } | ||||
|             level -= 2; | ||||
|             self.write_register(Register::RegPaConfig.addr(), PaConfig::PaBoost.addr() | level as u8) | ||||
|                 .await | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     pub async fn get_modem_stat(&mut self) -> Result<u8, Error<E, CS::Error, RESET::Error>> { | ||||
|         Ok(self.read_register(Register::RegModemStat.addr()).await? as u8) | ||||
|     } | ||||
|  | ||||
|     /// Sets the over current protection on the radio(mA). | ||||
|     pub async fn set_ocp(&mut self, ma: u8) -> Result<(), Error<E, CS::Error, RESET::Error>> { | ||||
|         let mut ocp_trim: u8 = 27; | ||||
|  | ||||
|         if ma <= 120 { | ||||
|             ocp_trim = (ma - 45) / 5; | ||||
|         } else if ma <= 240 { | ||||
|             ocp_trim = (ma + 30) / 10; | ||||
|         } | ||||
|         self.write_register(Register::RegOcp.addr(), 0x20 | (0x1F & ocp_trim)) | ||||
|             .await | ||||
|     } | ||||
|  | ||||
|     /// Sets the state of the radio. Default mode after initiation is `Standby`. | ||||
|     pub async fn set_mode(&mut self, mode: RadioMode) -> Result<(), Error<E, CS::Error, RESET::Error>> { | ||||
|         if self.explicit_header { | ||||
|             self.set_explicit_header_mode().await?; | ||||
|         } else { | ||||
|             self.set_implicit_header_mode().await?; | ||||
|         } | ||||
|         self.write_register( | ||||
|             Register::RegOpMode.addr(), | ||||
|             RadioMode::LongRangeMode.addr() | mode.addr(), | ||||
|         ) | ||||
|         .await?; | ||||
|  | ||||
|         self.mode = mode; | ||||
|         Ok(()) | ||||
|     } | ||||
|  | ||||
|     pub async fn reset_payload_length(&mut self) -> Result<(), Error<E, CS::Error, RESET::Error>> { | ||||
|         self.write_register(Register::RegPayloadLength.addr(), 0xFF).await | ||||
|     } | ||||
|  | ||||
|     /// Sets the frequency of the radio. Values are in megahertz. | ||||
|     /// I.E. 915 MHz must be used for North America. Check regulation for your area. | ||||
|     pub async fn set_frequency(&mut self, freq: u32) -> Result<(), Error<E, CS::Error, RESET::Error>> { | ||||
|         const FREQ_STEP: f64 = 61.03515625; | ||||
|         // calculate register values | ||||
|         let frf = (freq as f64 / FREQ_STEP) as u32; | ||||
|         // write registers | ||||
|         self.write_register(Register::RegFrfMsb.addr(), ((frf & 0x00FF_0000) >> 16) as u8) | ||||
|             .await?; | ||||
|         self.write_register(Register::RegFrfMid.addr(), ((frf & 0x0000_FF00) >> 8) as u8) | ||||
|             .await?; | ||||
|         self.write_register(Register::RegFrfLsb.addr(), (frf & 0x0000_00FF) as u8) | ||||
|             .await | ||||
|     } | ||||
|  | ||||
|     /// Sets the radio to use an explicit header. Default state is `ON`. | ||||
|     async fn set_explicit_header_mode(&mut self) -> Result<(), Error<E, CS::Error, RESET::Error>> { | ||||
|         let reg_modem_config_1 = self.read_register(Register::RegModemConfig1.addr()).await?; | ||||
|         self.write_register(Register::RegModemConfig1.addr(), reg_modem_config_1 & 0xfe) | ||||
|             .await?; | ||||
|         self.explicit_header = true; | ||||
|         Ok(()) | ||||
|     } | ||||
|  | ||||
|     /// Sets the radio to use an implicit header. Default state is `OFF`. | ||||
|     async fn set_implicit_header_mode(&mut self) -> Result<(), Error<E, CS::Error, RESET::Error>> { | ||||
|         let reg_modem_config_1 = self.read_register(Register::RegModemConfig1.addr()).await?; | ||||
|         self.write_register(Register::RegModemConfig1.addr(), reg_modem_config_1 & 0x01) | ||||
|             .await?; | ||||
|         self.explicit_header = false; | ||||
|         Ok(()) | ||||
|     } | ||||
|  | ||||
|     /// Sets the spreading factor of the radio. Supported values are between 6 and 12. | ||||
|     /// If a spreading factor of 6 is set, implicit header mode must be used to transmit | ||||
|     /// and receive packets. Default value is `7`. | ||||
|     pub async fn set_spreading_factor(&mut self, mut sf: u8) -> Result<(), Error<E, CS::Error, RESET::Error>> { | ||||
|         if sf < 6 { | ||||
|             sf = 6; | ||||
|         } else if sf > 12 { | ||||
|             sf = 12; | ||||
|         } | ||||
|  | ||||
|         if sf == 6 { | ||||
|             self.write_register(Register::RegDetectionOptimize.addr(), 0xc5).await?; | ||||
|             self.write_register(Register::RegDetectionThreshold.addr(), 0x0c) | ||||
|                 .await?; | ||||
|         } else { | ||||
|             self.write_register(Register::RegDetectionOptimize.addr(), 0xc3).await?; | ||||
|             self.write_register(Register::RegDetectionThreshold.addr(), 0x0a) | ||||
|                 .await?; | ||||
|         } | ||||
|         let modem_config_2 = self.read_register(Register::RegModemConfig2.addr()).await?; | ||||
|         self.write_register( | ||||
|             Register::RegModemConfig2.addr(), | ||||
|             (modem_config_2 & 0x0f) | ((sf << 4) & 0xf0), | ||||
|         ) | ||||
|         .await?; | ||||
|         self.set_ldo_flag().await?; | ||||
|  | ||||
|         self.write_register(Register::RegSymbTimeoutLsb.addr(), 0x05).await?; | ||||
|  | ||||
|         Ok(()) | ||||
|     } | ||||
|  | ||||
|     pub async fn set_tcxo(&mut self, external: bool) -> Result<(), Error<E, CS::Error, RESET::Error>> { | ||||
|         if external { | ||||
|             self.write_register(Register::RegTcxo.addr(), 0x10).await | ||||
|         } else { | ||||
|             self.write_register(Register::RegTcxo.addr(), 0x00).await | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     /// Sets the signal bandwidth of the radio. Supported values are: `7800 Hz`, `10400 Hz`, | ||||
|     /// `15600 Hz`, `20800 Hz`, `31250 Hz`,`41700 Hz` ,`62500 Hz`,`125000 Hz` and `250000 Hz` | ||||
|     /// Default value is `125000 Hz` | ||||
|     pub async fn set_signal_bandwidth(&mut self, sbw: i64) -> Result<(), Error<E, CS::Error, RESET::Error>> { | ||||
|         let bw: i64 = match sbw { | ||||
|             7_800 => 0, | ||||
|             10_400 => 1, | ||||
|             15_600 => 2, | ||||
|             20_800 => 3, | ||||
|             31_250 => 4, | ||||
|             41_700 => 5, | ||||
|             62_500 => 6, | ||||
|             125_000 => 7, | ||||
|             250_000 => 8, | ||||
|             _ => 9, | ||||
|         }; | ||||
|         let modem_config_1 = self.read_register(Register::RegModemConfig1.addr()).await?; | ||||
|         self.write_register( | ||||
|             Register::RegModemConfig1.addr(), | ||||
|             (modem_config_1 & 0x0f) | ((bw << 4) as u8), | ||||
|         ) | ||||
|         .await?; | ||||
|         self.set_ldo_flag().await?; | ||||
|         Ok(()) | ||||
|     } | ||||
|  | ||||
|     /// Sets the coding rate of the radio with the numerator fixed at 4. Supported values | ||||
|     /// are between `5` and `8`, these correspond to coding rates of `4/5` and `4/8`. | ||||
|     /// Default value is `5`. | ||||
|     pub async fn set_coding_rate_4(&mut self, mut denominator: u8) -> Result<(), Error<E, CS::Error, RESET::Error>> { | ||||
|         if denominator < 5 { | ||||
|             denominator = 5; | ||||
|         } else if denominator > 8 { | ||||
|             denominator = 8; | ||||
|         } | ||||
|         let cr = denominator - 4; | ||||
|         let modem_config_1 = self.read_register(Register::RegModemConfig1.addr()).await?; | ||||
|         self.write_register(Register::RegModemConfig1.addr(), (modem_config_1 & 0xf1) | (cr << 1)) | ||||
|             .await | ||||
|     } | ||||
|  | ||||
|     /// Sets the preamble length of the radio. Values are between 6 and 65535. | ||||
|     /// Default value is `8`. | ||||
|     pub async fn set_preamble_length(&mut self, length: i64) -> Result<(), Error<E, CS::Error, RESET::Error>> { | ||||
|         self.write_register(Register::RegPreambleMsb.addr(), (length >> 8) as u8) | ||||
|             .await?; | ||||
|         self.write_register(Register::RegPreambleLsb.addr(), length as u8).await | ||||
|     } | ||||
|  | ||||
|     /// Enables are disables the radio's CRC check. Default value is `false`. | ||||
|     pub async fn set_crc(&mut self, value: bool) -> Result<(), Error<E, CS::Error, RESET::Error>> { | ||||
|         let modem_config_2 = self.read_register(Register::RegModemConfig2.addr()).await?; | ||||
|         if value { | ||||
|             self.write_register(Register::RegModemConfig2.addr(), modem_config_2 | 0x04) | ||||
|                 .await | ||||
|         } else { | ||||
|             self.write_register(Register::RegModemConfig2.addr(), modem_config_2 & 0xfb) | ||||
|                 .await | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     /// Inverts the radio's IQ signals. Default value is `false`. | ||||
|     pub async fn set_invert_iq(&mut self, value: bool) -> Result<(), Error<E, CS::Error, RESET::Error>> { | ||||
|         if value { | ||||
|             self.write_register(Register::RegInvertiq.addr(), 0x66).await?; | ||||
|             self.write_register(Register::RegInvertiq2.addr(), 0x19).await | ||||
|         } else { | ||||
|             self.write_register(Register::RegInvertiq.addr(), 0x27).await?; | ||||
|             self.write_register(Register::RegInvertiq2.addr(), 0x1d).await | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     /// Returns the spreading factor of the radio. | ||||
|     pub async fn get_spreading_factor(&mut self) -> Result<u8, Error<E, CS::Error, RESET::Error>> { | ||||
|         Ok(self.read_register(Register::RegModemConfig2.addr()).await? >> 4) | ||||
|     } | ||||
|  | ||||
|     /// Returns the signal bandwidth of the radio. | ||||
|     pub async fn get_signal_bandwidth(&mut self) -> Result<i64, Error<E, CS::Error, RESET::Error>> { | ||||
|         let bw = self.read_register(Register::RegModemConfig1.addr()).await? >> 4; | ||||
|         let bw = match bw { | ||||
|             0 => 7_800, | ||||
|             1 => 10_400, | ||||
|             2 => 15_600, | ||||
|             3 => 20_800, | ||||
|             4 => 31_250, | ||||
|             5 => 41_700, | ||||
|             6 => 62_500, | ||||
|             7 => 125_000, | ||||
|             8 => 250_000, | ||||
|             9 => 500_000, | ||||
|             _ => -1, | ||||
|         }; | ||||
|         Ok(bw) | ||||
|     } | ||||
|  | ||||
|     /// Returns the RSSI of the last received packet. | ||||
|     pub async fn get_packet_rssi(&mut self) -> Result<i32, Error<E, CS::Error, RESET::Error>> { | ||||
|         Ok(i32::from(self.read_register(Register::RegPktRssiValue.addr()).await?) - 157) | ||||
|     } | ||||
|  | ||||
|     /// Returns the signal to noise radio of the the last received packet. | ||||
|     pub async fn get_packet_snr(&mut self) -> Result<f64, Error<E, CS::Error, RESET::Error>> { | ||||
|         Ok(f64::from(self.read_register(Register::RegPktSnrValue.addr()).await?)) | ||||
|     } | ||||
|  | ||||
|     /// Returns the frequency error of the last received packet in Hz. | ||||
|     pub async fn get_packet_frequency_error(&mut self) -> Result<i64, Error<E, CS::Error, RESET::Error>> { | ||||
|         let mut freq_error: i32; | ||||
|         freq_error = i32::from(self.read_register(Register::RegFreqErrorMsb.addr()).await? & 0x7); | ||||
|         freq_error <<= 8i64; | ||||
|         freq_error += i32::from(self.read_register(Register::RegFreqErrorMid.addr()).await?); | ||||
|         freq_error <<= 8i64; | ||||
|         freq_error += i32::from(self.read_register(Register::RegFreqErrorLsb.addr()).await?); | ||||
|  | ||||
|         let f_xtal = 32_000_000; // FXOSC: crystal oscillator (XTAL) frequency (2.5. Chip Specification, p. 14) | ||||
|         let f_error = ((f64::from(freq_error) * (1i64 << 24) as f64) / f64::from(f_xtal)) | ||||
|             * (self.get_signal_bandwidth().await? as f64 / 500_000.0f64); // p. 37 | ||||
|         Ok(f_error as i64) | ||||
|     } | ||||
|  | ||||
|     async fn set_ldo_flag(&mut self) -> Result<(), Error<E, CS::Error, RESET::Error>> { | ||||
|         let sw = self.get_signal_bandwidth().await?; | ||||
|         // Section 4.1.1.5 | ||||
|         let symbol_duration = 1000 / (sw / ((1_i64) << self.get_spreading_factor().await?)); | ||||
|  | ||||
|         // Section 4.1.1.6 | ||||
|         let ldo_on = symbol_duration > 16; | ||||
|  | ||||
|         let mut config_3 = self.read_register(Register::RegModemConfig3.addr()).await?; | ||||
|         config_3.set_bit(3, ldo_on); | ||||
|         //config_3.set_bit(2, true); | ||||
|         self.write_register(Register::RegModemConfig3.addr(), config_3).await | ||||
|     } | ||||
|  | ||||
|     async fn read_register(&mut self, reg: u8) -> Result<u8, Error<E, CS::Error, RESET::Error>> { | ||||
|         let mut buffer = [reg & 0x7f, 0]; | ||||
|         self.cs.set_low().map_err(CS)?; | ||||
|  | ||||
|         let _ = self.spi.transfer(&mut buffer, &[reg & 0x7f, 0]).await.map_err(SPI)?; | ||||
|  | ||||
|         self.cs.set_high().map_err(CS)?; | ||||
|         Ok(buffer[1]) | ||||
|     } | ||||
|  | ||||
|     async fn write_register(&mut self, reg: u8, byte: u8) -> Result<(), Error<E, CS::Error, RESET::Error>> { | ||||
|         self.cs.set_low().map_err(CS)?; | ||||
|         let buffer = [reg | 0x80, byte]; | ||||
|         self.spi.write(&buffer).await.map_err(SPI)?; | ||||
|         self.cs.set_high().map_err(CS)?; | ||||
|         Ok(()) | ||||
|     } | ||||
|  | ||||
|     pub async fn put_in_fsk_mode(&mut self) -> Result<(), Error<E, CS::Error, RESET::Error>> { | ||||
|         // Put in FSK mode | ||||
|         let mut op_mode = 0; | ||||
|         op_mode | ||||
|             .set_bit(7, false) // FSK mode | ||||
|             .set_bits(5..6, 0x00) // FSK modulation | ||||
|             .set_bit(3, false) //Low freq registers | ||||
|             .set_bits(0..2, 0b011); // Mode | ||||
|  | ||||
|         self.write_register(Register::RegOpMode as u8, op_mode).await | ||||
|     } | ||||
|  | ||||
|     pub async fn set_fsk_pa_ramp( | ||||
|         &mut self, | ||||
|         modulation_shaping: FskDataModulationShaping, | ||||
|         ramp: FskRampUpRamDown, | ||||
|     ) -> Result<(), Error<E, CS::Error, RESET::Error>> { | ||||
|         let mut pa_ramp = 0; | ||||
|         pa_ramp | ||||
|             .set_bits(5..6, modulation_shaping as u8) | ||||
|             .set_bits(0..3, ramp as u8); | ||||
|  | ||||
|         self.write_register(Register::RegPaRamp as u8, pa_ramp).await | ||||
|     } | ||||
|  | ||||
|     pub async fn set_lora_pa_ramp(&mut self) -> Result<(), Error<E, CS::Error, RESET::Error>> { | ||||
|         self.write_register(Register::RegPaRamp as u8, 0b1000).await | ||||
|     } | ||||
|  | ||||
|     pub async fn set_lora_sync_word(&mut self) -> Result<(), Error<E, CS::Error, RESET::Error>> { | ||||
|         self.write_register(Register::RegSyncWord as u8, 0x34).await | ||||
|     } | ||||
| } | ||||
| /// Modes of the radio and their corresponding register values. | ||||
| #[derive(Clone, Copy)] | ||||
| pub enum RadioMode { | ||||
|     LongRangeMode = 0x80, | ||||
|     Sleep = 0x00, | ||||
|     Stdby = 0x01, | ||||
|     Tx = 0x03, | ||||
|     RxContinuous = 0x05, | ||||
|     RxSingle = 0x06, | ||||
| } | ||||
|  | ||||
| impl RadioMode { | ||||
|     /// Returns the address of the mode. | ||||
|     pub fn addr(self) -> u8 { | ||||
|         self as u8 | ||||
|     } | ||||
| } | ||||
| @@ -1,107 +0,0 @@ | ||||
| // Copyright Charles Wade (https://github.com/mr-glt/sx127x_lora). Licensed under the Apache 2.0 | ||||
| // license | ||||
| // | ||||
| // Modifications made to make the driver work with the rust-lorawan link layer. | ||||
| #![allow(dead_code, clippy::enum_variant_names)] | ||||
|  | ||||
| #[derive(Clone, Copy)] | ||||
| pub enum Register { | ||||
|     RegFifo = 0x00, | ||||
|     RegOpMode = 0x01, | ||||
|     RegFrfMsb = 0x06, | ||||
|     RegFrfMid = 0x07, | ||||
|     RegFrfLsb = 0x08, | ||||
|     RegPaConfig = 0x09, | ||||
|     RegPaRamp = 0x0a, | ||||
|     RegOcp = 0x0b, | ||||
|     RegLna = 0x0c, | ||||
|     RegFifoAddrPtr = 0x0d, | ||||
|     RegFifoTxBaseAddr = 0x0e, | ||||
|     RegFifoRxBaseAddr = 0x0f, | ||||
|     RegFifoRxCurrentAddr = 0x10, | ||||
|     RegIrqFlagsMask = 0x11, | ||||
|     RegIrqFlags = 0x12, | ||||
|     RegRxNbBytes = 0x13, | ||||
|     RegPktSnrValue = 0x19, | ||||
|     RegModemStat = 0x18, | ||||
|     RegPktRssiValue = 0x1a, | ||||
|     RegModemConfig1 = 0x1d, | ||||
|     RegModemConfig2 = 0x1e, | ||||
|     RegSymbTimeoutLsb = 0x1f, | ||||
|     RegPreambleMsb = 0x20, | ||||
|     RegPreambleLsb = 0x21, | ||||
|     RegPayloadLength = 0x22, | ||||
|     RegMaxPayloadLength = 0x23, | ||||
|     RegModemConfig3 = 0x26, | ||||
|     RegFreqErrorMsb = 0x28, | ||||
|     RegFreqErrorMid = 0x29, | ||||
|     RegFreqErrorLsb = 0x2a, | ||||
|     RegRssiWideband = 0x2c, | ||||
|     RegDetectionOptimize = 0x31, | ||||
|     RegInvertiq = 0x33, | ||||
|     RegDetectionThreshold = 0x37, | ||||
|     RegSyncWord = 0x39, | ||||
|     RegInvertiq2 = 0x3b, | ||||
|     RegDioMapping1 = 0x40, | ||||
|     RegVersion = 0x42, | ||||
|     RegTcxo = 0x4b, | ||||
|     RegPaDac = 0x4d, | ||||
| } | ||||
| #[derive(Clone, Copy)] | ||||
| pub enum PaConfig { | ||||
|     PaBoost = 0x80, | ||||
|     PaOutputRfoPin = 0, | ||||
| } | ||||
|  | ||||
| #[derive(Clone, Copy)] | ||||
| pub enum IRQ { | ||||
|     IrqTxDoneMask = 0x08, | ||||
|     IrqPayloadCrcErrorMask = 0x20, | ||||
|     IrqRxDoneMask = 0x40, | ||||
| } | ||||
|  | ||||
| impl Register { | ||||
|     pub fn addr(self) -> u8 { | ||||
|         self as u8 | ||||
|     } | ||||
| } | ||||
|  | ||||
| impl PaConfig { | ||||
|     pub fn addr(self) -> u8 { | ||||
|         self as u8 | ||||
|     } | ||||
| } | ||||
|  | ||||
| impl IRQ { | ||||
|     pub fn addr(self) -> u8 { | ||||
|         self as u8 | ||||
|     } | ||||
| } | ||||
|  | ||||
| #[derive(Clone, Copy)] | ||||
| pub enum FskDataModulationShaping { | ||||
|     None = 1, | ||||
|     GaussianBt1d0 = 2, | ||||
|     GaussianBt0d5 = 10, | ||||
|     GaussianBt0d3 = 11, | ||||
| } | ||||
|  | ||||
| #[derive(Clone, Copy)] | ||||
| pub enum FskRampUpRamDown { | ||||
|     _3d4ms = 0b000, | ||||
|     _2ms = 0b0001, | ||||
|     _1ms = 0b0010, | ||||
|     _500us = 0b0011, | ||||
|     _250us = 0b0100, | ||||
|     _125us = 0b0101, | ||||
|     _100us = 0b0110, | ||||
|     _62us = 0b0111, | ||||
|     _50us = 0b1000, | ||||
|     _40us = 0b1001, | ||||
|     _31us = 0b1010, | ||||
|     _25us = 0b1011, | ||||
|     _20us = 0b1100, | ||||
|     _15us = 0b1101, | ||||
|     _12us = 0b1110, | ||||
|     _10us = 0b1111, | ||||
| } | ||||
| @@ -1,7 +1,15 @@ | ||||
| [package] | ||||
| name = "embassy-macros" | ||||
| version = "0.1.0" | ||||
| version = "0.2.0" | ||||
| edition = "2021" | ||||
| license = "MIT OR Apache-2.0" | ||||
| description = "macros for creating the entry point and tasks for embassy-executor" | ||||
| repository = "https://github.com/embassy-rs/embassy" | ||||
| categories = [ | ||||
|     "embedded", | ||||
|     "no-std", | ||||
|     "asynchronous", | ||||
| ] | ||||
|  | ||||
| [dependencies] | ||||
| syn = { version = "1.0.76", features = ["full", "extra-traits"] } | ||||
| @@ -13,8 +21,5 @@ proc-macro2 = "1.0.29" | ||||
| proc-macro = true | ||||
|  | ||||
| [features] | ||||
| std = [] | ||||
| wasm = [] | ||||
|  | ||||
| # Enabling this cause interrupt::take! to require embassy-executor | ||||
| rtos-trace-interrupt = [] | ||||
|   | ||||
							
								
								
									
										21
									
								
								embassy-macros/README.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										21
									
								
								embassy-macros/README.md
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,21 @@ | ||||
| # embassy-macros | ||||
|  | ||||
| An [Embassy](https://embassy.dev) project. | ||||
|  | ||||
| Macros for creating the main entry point and tasks that can be spawned by `embassy-executor`.  | ||||
|  | ||||
| NOTE: The macros are re-exported by the `embassy-executor` crate which should be used instead of adding a direct dependency on the `embassy-macros` crate. | ||||
|  | ||||
| ## Minimum supported Rust version (MSRV) | ||||
|  | ||||
| The `task` and `main` macros require the type alias impl trait (TAIT) nightly feature in order to compile. | ||||
|  | ||||
| ## License | ||||
|  | ||||
| This work is licensed under either of | ||||
|  | ||||
| - Apache License, Version 2.0 ([LICENSE-APACHE](LICENSE-APACHE) or | ||||
|   <http://www.apache.org/licenses/LICENSE-2.0>) | ||||
| - MIT license ([LICENSE-MIT](LICENSE-MIT) or <http://opensource.org/licenses/MIT>) | ||||
|  | ||||
| at your option. | ||||
| @@ -1,3 +1,4 @@ | ||||
| #![doc = include_str!("../README.md")] | ||||
| extern crate proc_macro; | ||||
|  | ||||
| use proc_macro::TokenStream; | ||||
| @@ -6,6 +7,36 @@ mod macros; | ||||
| mod util; | ||||
| use macros::*; | ||||
|  | ||||
| /// Declares an async task that can be run by `embassy-executor`. The optional `pool_size` parameter can be used to specify how | ||||
| /// many concurrent tasks can be spawned (default is 1) for the function. | ||||
| /// | ||||
| /// | ||||
| /// The following restrictions apply: | ||||
| /// | ||||
| /// * The function must be declared `async`. | ||||
| /// * The function must not use generics. | ||||
| /// * The optional `pool_size` attribute must be 1 or greater. | ||||
| /// | ||||
| /// | ||||
| /// ## Examples | ||||
| /// | ||||
| /// Declaring a task taking no arguments: | ||||
| /// | ||||
| /// ``` rust | ||||
| /// #[embassy_executor::task] | ||||
| /// async fn mytask() { | ||||
| ///     // Function body | ||||
| /// } | ||||
| /// ``` | ||||
| /// | ||||
| /// Declaring a task with a given pool size: | ||||
| /// | ||||
| /// ``` rust | ||||
| /// #[embassy_executor::task(pool_size = 4)] | ||||
| /// async fn mytask() { | ||||
| ///     // Function body | ||||
| /// } | ||||
| /// ``` | ||||
| #[proc_macro_attribute] | ||||
| pub fn task(args: TokenStream, item: TokenStream) -> TokenStream { | ||||
|     let args = syn::parse_macro_input!(args as syn::AttributeArgs); | ||||
| @@ -14,11 +45,104 @@ pub fn task(args: TokenStream, item: TokenStream) -> TokenStream { | ||||
|     task::run(args, f).unwrap_or_else(|x| x).into() | ||||
| } | ||||
|  | ||||
| /// Creates a new `executor` instance and declares an application entry point for Cortex-M spawning the corresponding function body as an async task. | ||||
| /// | ||||
| /// The following restrictions apply: | ||||
| /// | ||||
| /// * The function must accept exactly 1 parameter, an `embassy_executor::Spawner` handle that it can use to spawn additional tasks. | ||||
| /// * The function must be declared `async`. | ||||
| /// * The function must not use generics. | ||||
| /// * Only a single `main` task may be declared. | ||||
| /// | ||||
| /// ## Examples | ||||
| /// Spawning a task: | ||||
| /// | ||||
| /// ``` rust | ||||
| /// #[embassy_executor::main] | ||||
| /// async fn main(_s: embassy_executor::Spawner) { | ||||
| ///     // Function body | ||||
| /// } | ||||
| /// ``` | ||||
| #[proc_macro_attribute] | ||||
| pub fn main(args: TokenStream, item: TokenStream) -> TokenStream { | ||||
| pub fn main_cortex_m(args: TokenStream, item: TokenStream) -> TokenStream { | ||||
|     let args = syn::parse_macro_input!(args as syn::AttributeArgs); | ||||
|     let f = syn::parse_macro_input!(item as syn::ItemFn); | ||||
|     main::run(args, f).unwrap_or_else(|x| x).into() | ||||
|     main::run(args, f, main::cortex_m()).unwrap_or_else(|x| x).into() | ||||
| } | ||||
|  | ||||
| /// Creates a new `executor` instance and declares an application entry point for RISC-V spawning the corresponding function body as an async task. | ||||
| /// | ||||
| /// The following restrictions apply: | ||||
| /// | ||||
| /// * The function must accept exactly 1 parameter, an `embassy_executor::Spawner` handle that it can use to spawn additional tasks. | ||||
| /// * The function must be declared `async`. | ||||
| /// * The function must not use generics. | ||||
| /// * Only a single `main` task may be declared. | ||||
| /// | ||||
| /// ## Examples | ||||
| /// Spawning a task: | ||||
| /// | ||||
| /// ``` rust | ||||
| /// #[embassy_executor::main] | ||||
| /// async fn main(_s: embassy_executor::Spawner) { | ||||
| ///     // Function body | ||||
| /// } | ||||
| /// ``` | ||||
| #[proc_macro_attribute] | ||||
| pub fn main_riscv(args: TokenStream, item: TokenStream) -> TokenStream { | ||||
|     let args = syn::parse_macro_input!(args as syn::AttributeArgs); | ||||
|     let f = syn::parse_macro_input!(item as syn::ItemFn); | ||||
|     main::run(args, f, main::riscv()).unwrap_or_else(|x| x).into() | ||||
| } | ||||
|  | ||||
| /// Creates a new `executor` instance and declares an application entry point for STD spawning the corresponding function body as an async task. | ||||
| /// | ||||
| /// The following restrictions apply: | ||||
| /// | ||||
| /// * The function must accept exactly 1 parameter, an `embassy_executor::Spawner` handle that it can use to spawn additional tasks. | ||||
| /// * The function must be declared `async`. | ||||
| /// * The function must not use generics. | ||||
| /// * Only a single `main` task may be declared. | ||||
| /// | ||||
| /// ## Examples | ||||
| /// Spawning a task: | ||||
| /// | ||||
| /// ``` rust | ||||
| /// #[embassy_executor::main] | ||||
| /// async fn main(_s: embassy_executor::Spawner) { | ||||
| ///     // Function body | ||||
| /// } | ||||
| /// ``` | ||||
| #[proc_macro_attribute] | ||||
| pub fn main_std(args: TokenStream, item: TokenStream) -> TokenStream { | ||||
|     let args = syn::parse_macro_input!(args as syn::AttributeArgs); | ||||
|     let f = syn::parse_macro_input!(item as syn::ItemFn); | ||||
|     main::run(args, f, main::std()).unwrap_or_else(|x| x).into() | ||||
| } | ||||
|  | ||||
| /// Creates a new `executor` instance and declares an application entry point for WASM spawning the corresponding function body as an async task. | ||||
| /// | ||||
| /// The following restrictions apply: | ||||
| /// | ||||
| /// * The function must accept exactly 1 parameter, an `embassy_executor::Spawner` handle that it can use to spawn additional tasks. | ||||
| /// * The function must be declared `async`. | ||||
| /// * The function must not use generics. | ||||
| /// * Only a single `main` task may be declared. | ||||
| /// | ||||
| /// ## Examples | ||||
| /// Spawning a task: | ||||
| /// | ||||
| /// ``` rust | ||||
| /// #[embassy_executor::main] | ||||
| /// async fn main(_s: embassy_executor::Spawner) { | ||||
| ///     // Function body | ||||
| /// } | ||||
| /// ``` | ||||
| #[proc_macro_attribute] | ||||
| pub fn main_wasm(args: TokenStream, item: TokenStream) -> TokenStream { | ||||
|     let args = syn::parse_macro_input!(args as syn::AttributeArgs); | ||||
|     let f = syn::parse_macro_input!(item as syn::ItemFn); | ||||
|     main::run(args, f, main::wasm()).unwrap_or_else(|x| x).into() | ||||
| } | ||||
|  | ||||
| #[proc_macro_attribute] | ||||
|   | ||||
| @@ -6,7 +6,10 @@ pub fn run(name: syn::Ident) -> Result<TokenStream, TokenStream> { | ||||
|     let name_interrupt = format_ident!("{}", name); | ||||
|     let name_handler = format!("__EMBASSY_{}_HANDLER", name); | ||||
|  | ||||
|     let doc = format!("{} interrupt singleton.", name); | ||||
|  | ||||
|     let result = quote! { | ||||
|         #[doc = #doc] | ||||
|         #[allow(non_camel_case_types)] | ||||
|         pub struct #name_interrupt(()); | ||||
|         unsafe impl ::embassy_cortex_m::interrupt::Interrupt for #name_interrupt { | ||||
| @@ -18,9 +21,9 @@ pub fn run(name: syn::Ident) -> Result<TokenStream, TokenStream> { | ||||
|             unsafe fn steal() -> Self { | ||||
|                 Self(()) | ||||
|             } | ||||
|             unsafe fn __handler(&self) -> &'static ::embassy_cortex_m::interrupt::Handler { | ||||
|             unsafe fn __handler(&self) -> &'static ::embassy_cortex_m::interrupt::DynHandler { | ||||
|                 #[export_name = #name_handler] | ||||
|                 static HANDLER: ::embassy_cortex_m::interrupt::Handler = ::embassy_cortex_m::interrupt::Handler::new(); | ||||
|                 static HANDLER: ::embassy_cortex_m::interrupt::DynHandler = ::embassy_cortex_m::interrupt::DynHandler::new(); | ||||
|                 &HANDLER | ||||
|             } | ||||
|         } | ||||
|   | ||||
| @@ -10,12 +10,12 @@ pub fn run(name: syn::Ident) -> Result<TokenStream, TokenStream> { | ||||
|     let (isr_enter, isr_exit) = ( | ||||
|         quote! { | ||||
|             ::embassy_executor::rtos_trace_interrupt! { | ||||
|                 ::embassy_executor::export::trace::isr_enter(); | ||||
|                 ::embassy_executor::_export::trace::isr_enter(); | ||||
|             } | ||||
|         }, | ||||
|         quote! { | ||||
|             ::embassy_executor::rtos_trace_interrupt! { | ||||
|                 ::embassy_executor::export::trace::isr_exit(); | ||||
|                 ::embassy_executor::_export::trace::isr_exit(); | ||||
|             } | ||||
|         }, | ||||
|     ); | ||||
| @@ -30,7 +30,7 @@ pub fn run(name: syn::Ident) -> Result<TokenStream, TokenStream> { | ||||
|             pub unsafe extern "C" fn trampoline() { | ||||
|                 extern "C" { | ||||
|                     #[link_name = #name_handler] | ||||
|                     static HANDLER: interrupt::Handler; | ||||
|                     static HANDLER: interrupt::DynHandler; | ||||
|                 } | ||||
|  | ||||
|                 let func = HANDLER.func.load(interrupt::_export::atomic::Ordering::Relaxed); | ||||
|   | ||||
| @@ -1,13 +1,69 @@ | ||||
| use darling::FromMeta; | ||||
| use proc_macro2::TokenStream; | ||||
| use quote::quote; | ||||
| use syn::{ReturnType, Type}; | ||||
|  | ||||
| use crate::util::ctxt::Ctxt; | ||||
|  | ||||
| #[derive(Debug, FromMeta)] | ||||
| struct Args {} | ||||
|  | ||||
| pub fn run(args: syn::AttributeArgs, f: syn::ItemFn) -> Result<TokenStream, TokenStream> { | ||||
| pub fn riscv() -> TokenStream { | ||||
|     quote! { | ||||
|         #[riscv_rt::entry] | ||||
|         fn main() -> ! { | ||||
|             let mut executor = ::embassy_executor::Executor::new(); | ||||
|             let executor = unsafe { __make_static(&mut executor) }; | ||||
|             executor.run(|spawner| { | ||||
|                 spawner.must_spawn(__embassy_main(spawner)); | ||||
|             }) | ||||
|         } | ||||
|     } | ||||
| } | ||||
|  | ||||
| pub fn cortex_m() -> TokenStream { | ||||
|     quote! { | ||||
|         #[cortex_m_rt::entry] | ||||
|         fn main() -> ! { | ||||
|             let mut executor = ::embassy_executor::Executor::new(); | ||||
|             let executor = unsafe { __make_static(&mut executor) }; | ||||
|             executor.run(|spawner| { | ||||
|                 spawner.must_spawn(__embassy_main(spawner)); | ||||
|             }) | ||||
|         } | ||||
|     } | ||||
| } | ||||
|  | ||||
| pub fn wasm() -> TokenStream { | ||||
|     quote! { | ||||
|         #[wasm_bindgen::prelude::wasm_bindgen(start)] | ||||
|         pub fn main() -> Result<(), wasm_bindgen::JsValue> { | ||||
|             static EXECUTOR: ::embassy_executor::_export::StaticCell<::embassy_executor::Executor> = ::embassy_executor::_export::StaticCell::new(); | ||||
|             let executor = EXECUTOR.init(::embassy_executor::Executor::new()); | ||||
|  | ||||
|             executor.start(|spawner| { | ||||
|                 spawner.spawn(__embassy_main(spawner)).unwrap(); | ||||
|             }); | ||||
|  | ||||
|             Ok(()) | ||||
|         } | ||||
|     } | ||||
| } | ||||
|  | ||||
| pub fn std() -> TokenStream { | ||||
|     quote! { | ||||
|         fn main() -> ! { | ||||
|             let mut executor = ::embassy_executor::Executor::new(); | ||||
|             let executor = unsafe { __make_static(&mut executor) }; | ||||
|  | ||||
|             executor.run(|spawner| { | ||||
|                 spawner.must_spawn(__embassy_main(spawner)); | ||||
|             }) | ||||
|         } | ||||
|     } | ||||
| } | ||||
|  | ||||
| pub fn run(args: syn::AttributeArgs, f: syn::ItemFn, main: TokenStream) -> Result<TokenStream, TokenStream> { | ||||
|     #[allow(unused_variables)] | ||||
|     let args = Args::from_list(&args).map_err(|e| e.write_errors())?; | ||||
|  | ||||
| @@ -21,6 +77,26 @@ pub fn run(args: syn::AttributeArgs, f: syn::ItemFn) -> Result<TokenStream, Toke | ||||
|     if !f.sig.generics.params.is_empty() { | ||||
|         ctxt.error_spanned_by(&f.sig, "main function must not be generic"); | ||||
|     } | ||||
|     if !f.sig.generics.where_clause.is_none() { | ||||
|         ctxt.error_spanned_by(&f.sig, "main function must not have `where` clauses"); | ||||
|     } | ||||
|     if !f.sig.abi.is_none() { | ||||
|         ctxt.error_spanned_by(&f.sig, "main function must not have an ABI qualifier"); | ||||
|     } | ||||
|     if !f.sig.variadic.is_none() { | ||||
|         ctxt.error_spanned_by(&f.sig, "main function must not be variadic"); | ||||
|     } | ||||
|     match &f.sig.output { | ||||
|         ReturnType::Default => {} | ||||
|         ReturnType::Type(_, ty) => match &**ty { | ||||
|             Type::Tuple(tuple) if tuple.elems.is_empty() => {} | ||||
|             Type::Never(_) => {} | ||||
|             _ => ctxt.error_spanned_by( | ||||
|                 &f.sig, | ||||
|                 "main function must either not return a value, return `()` or return `!`", | ||||
|             ), | ||||
|         }, | ||||
|     } | ||||
|  | ||||
|     if fargs.len() != 1 { | ||||
|         ctxt.error_spanned_by(&f.sig, "main function must have 1 argument: the spawner."); | ||||
| @@ -29,50 +105,11 @@ pub fn run(args: syn::AttributeArgs, f: syn::ItemFn) -> Result<TokenStream, Toke | ||||
|     ctxt.check()?; | ||||
|  | ||||
|     let f_body = f.block; | ||||
|  | ||||
|     #[cfg(feature = "wasm")] | ||||
|     let main = quote! { | ||||
|         #[wasm_bindgen::prelude::wasm_bindgen(start)] | ||||
|         pub fn main() -> Result<(), wasm_bindgen::JsValue> { | ||||
|             static EXECUTOR: ::embassy_executor::_export::StaticCell<::embassy_executor::Executor> = ::embassy_executor::_export::StaticCell::new(); | ||||
|             let executor = EXECUTOR.init(::embassy_executor::Executor::new()); | ||||
|  | ||||
|             executor.start(|spawner| { | ||||
|                 spawner.spawn(__embassy_main(spawner)).unwrap(); | ||||
|             }); | ||||
|  | ||||
|             Ok(()) | ||||
|         } | ||||
|     }; | ||||
|  | ||||
|     #[cfg(all(feature = "std", not(feature = "wasm")))] | ||||
|     let main = quote! { | ||||
|         fn main() -> ! { | ||||
|             let mut executor = ::embassy_executor::Executor::new(); | ||||
|             let executor = unsafe { __make_static(&mut executor) }; | ||||
|  | ||||
|             executor.run(|spawner| { | ||||
|                 spawner.must_spawn(__embassy_main(spawner)); | ||||
|             }) | ||||
|         } | ||||
|     }; | ||||
|  | ||||
|     #[cfg(all(not(feature = "std"), not(feature = "wasm")))] | ||||
|     let main = quote! { | ||||
|         #[cortex_m_rt::entry] | ||||
|         fn main() -> ! { | ||||
|             let mut executor = ::embassy_executor::Executor::new(); | ||||
|             let executor = unsafe { __make_static(&mut executor) }; | ||||
|  | ||||
|             executor.run(|spawner| { | ||||
|                 spawner.must_spawn(__embassy_main(spawner)); | ||||
|             }) | ||||
|         } | ||||
|     }; | ||||
|     let out = &f.sig.output; | ||||
|  | ||||
|     let result = quote! { | ||||
|         #[::embassy_executor::task()] | ||||
|         async fn __embassy_main(#fargs) { | ||||
|         async fn __embassy_main(#fargs) #out { | ||||
|             #f_body | ||||
|         } | ||||
|  | ||||
|   | ||||
| @@ -1,6 +1,7 @@ | ||||
| use darling::FromMeta; | ||||
| use proc_macro2::TokenStream; | ||||
| use quote::{format_ident, quote}; | ||||
| use syn::{parse_quote, ItemFn, ReturnType, Type}; | ||||
|  | ||||
| use crate::util::ctxt::Ctxt; | ||||
|  | ||||
| @@ -23,6 +24,27 @@ pub fn run(args: syn::AttributeArgs, f: syn::ItemFn) -> Result<TokenStream, Toke | ||||
|     if !f.sig.generics.params.is_empty() { | ||||
|         ctxt.error_spanned_by(&f.sig, "task functions must not be generic"); | ||||
|     } | ||||
|     if !f.sig.generics.where_clause.is_none() { | ||||
|         ctxt.error_spanned_by(&f.sig, "task functions must not have `where` clauses"); | ||||
|     } | ||||
|     if !f.sig.abi.is_none() { | ||||
|         ctxt.error_spanned_by(&f.sig, "task functions must not have an ABI qualifier"); | ||||
|     } | ||||
|     if !f.sig.variadic.is_none() { | ||||
|         ctxt.error_spanned_by(&f.sig, "task functions must not be variadic"); | ||||
|     } | ||||
|     match &f.sig.output { | ||||
|         ReturnType::Default => {} | ||||
|         ReturnType::Type(_, ty) => match &**ty { | ||||
|             Type::Tuple(tuple) if tuple.elems.is_empty() => {} | ||||
|             Type::Never(_) => {} | ||||
|             _ => ctxt.error_spanned_by( | ||||
|                 &f.sig, | ||||
|                 "task functions must either not return a value, return `()` or return `!`", | ||||
|             ), | ||||
|         }, | ||||
|     } | ||||
|  | ||||
|     if pool_size < 1 { | ||||
|         ctxt.error_spanned_by(&f.sig, "pool_size must be 1 or greater"); | ||||
|     } | ||||
| @@ -57,13 +79,7 @@ pub fn run(args: syn::AttributeArgs, f: syn::ItemFn) -> Result<TokenStream, Toke | ||||
|     task_inner.vis = syn::Visibility::Inherited; | ||||
|     task_inner.sig.ident = task_inner_ident.clone(); | ||||
|  | ||||
|     let result = quote! { | ||||
|         // This is the user's task function, renamed. | ||||
|         // We put it outside the #task_ident fn below, because otherwise | ||||
|         // the items defined there (such as POOL) would be in scope | ||||
|         // in the user's code. | ||||
|         #task_inner | ||||
|  | ||||
|     let mut task_outer: ItemFn = parse_quote! { | ||||
|         #visibility fn #task_ident(#fargs) -> ::embassy_executor::SpawnToken<impl Sized> { | ||||
|             type Fut = impl ::core::future::Future + 'static; | ||||
|             static POOL: ::embassy_executor::raw::TaskPool<Fut, #pool_size> = ::embassy_executor::raw::TaskPool::new(); | ||||
| @@ -71,5 +87,18 @@ pub fn run(args: syn::AttributeArgs, f: syn::ItemFn) -> Result<TokenStream, Toke | ||||
|         } | ||||
|     }; | ||||
|  | ||||
|     task_outer.attrs.append(&mut task_inner.attrs.clone()); | ||||
|  | ||||
|     let result = quote! { | ||||
|         // This is the user's task function, renamed. | ||||
|         // We put it outside the #task_ident fn below, because otherwise | ||||
|         // the items defined there (such as POOL) would be in scope | ||||
|         // in the user's code. | ||||
|         #[doc(hidden)] | ||||
|         #task_inner | ||||
|  | ||||
|         #task_outer | ||||
|     }; | ||||
|  | ||||
|     Ok(result) | ||||
| } | ||||
|   | ||||
							
								
								
									
										18
									
								
								embassy-net-driver-channel/Cargo.toml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										18
									
								
								embassy-net-driver-channel/Cargo.toml
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,18 @@ | ||||
| [package] | ||||
| name = "embassy-net-driver-channel" | ||||
| version = "0.1.0" | ||||
| edition = "2021" | ||||
|  | ||||
| [package.metadata.embassy_docs] | ||||
| src_base = "https://github.com/embassy-rs/embassy/blob/embassy-net-driver-channel-v$VERSION/embassy-net-driver-channel/src/" | ||||
| src_base_git = "https://github.com/embassy-rs/embassy/blob/$COMMIT/embassy-net-driver-channel/src/" | ||||
| features = ["defmt"] | ||||
| target = "thumbv7em-none-eabi" | ||||
|  | ||||
| [dependencies] | ||||
| defmt = { version = "0.3", optional = true } | ||||
| log = { version = "0.4.14", optional = true } | ||||
|  | ||||
| embassy-sync = { version = "0.2.0", path = "../embassy-sync" } | ||||
| embassy-futures = { version = "0.1.0", path = "../embassy-futures" } | ||||
| embassy-net-driver = { version = "0.1.0", path = "../embassy-net-driver" } | ||||
							
								
								
									
										549
									
								
								embassy-net-driver-channel/src/lib.rs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										549
									
								
								embassy-net-driver-channel/src/lib.rs
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,549 @@ | ||||
| #![no_std] | ||||
|  | ||||
| // must go first! | ||||
| mod fmt; | ||||
|  | ||||
| use core::cell::RefCell; | ||||
| use core::mem::MaybeUninit; | ||||
| use core::task::{Context, Poll}; | ||||
|  | ||||
| pub use embassy_net_driver as driver; | ||||
| use embassy_net_driver::{Capabilities, LinkState, Medium}; | ||||
| use embassy_sync::blocking_mutex::raw::NoopRawMutex; | ||||
| use embassy_sync::blocking_mutex::Mutex; | ||||
| use embassy_sync::waitqueue::WakerRegistration; | ||||
|  | ||||
| pub struct State<const MTU: usize, const N_RX: usize, const N_TX: usize> { | ||||
|     rx: [PacketBuf<MTU>; N_RX], | ||||
|     tx: [PacketBuf<MTU>; N_TX], | ||||
|     inner: MaybeUninit<StateInner<'static, MTU>>, | ||||
| } | ||||
|  | ||||
| impl<const MTU: usize, const N_RX: usize, const N_TX: usize> State<MTU, N_RX, N_TX> { | ||||
|     const NEW_PACKET: PacketBuf<MTU> = PacketBuf::new(); | ||||
|  | ||||
|     pub const fn new() -> Self { | ||||
|         Self { | ||||
|             rx: [Self::NEW_PACKET; N_RX], | ||||
|             tx: [Self::NEW_PACKET; N_TX], | ||||
|             inner: MaybeUninit::uninit(), | ||||
|         } | ||||
|     } | ||||
| } | ||||
|  | ||||
| struct StateInner<'d, const MTU: usize> { | ||||
|     rx: zerocopy_channel::Channel<'d, NoopRawMutex, PacketBuf<MTU>>, | ||||
|     tx: zerocopy_channel::Channel<'d, NoopRawMutex, PacketBuf<MTU>>, | ||||
|     shared: Mutex<NoopRawMutex, RefCell<Shared>>, | ||||
| } | ||||
|  | ||||
| /// State of the LinkState | ||||
| struct Shared { | ||||
|     link_state: LinkState, | ||||
|     waker: WakerRegistration, | ||||
|     ethernet_address: [u8; 6], | ||||
| } | ||||
|  | ||||
| pub struct Runner<'d, const MTU: usize> { | ||||
|     tx_chan: zerocopy_channel::Receiver<'d, NoopRawMutex, PacketBuf<MTU>>, | ||||
|     rx_chan: zerocopy_channel::Sender<'d, NoopRawMutex, PacketBuf<MTU>>, | ||||
|     shared: &'d Mutex<NoopRawMutex, RefCell<Shared>>, | ||||
| } | ||||
|  | ||||
| #[derive(Clone, Copy)] | ||||
| pub struct StateRunner<'d> { | ||||
|     shared: &'d Mutex<NoopRawMutex, RefCell<Shared>>, | ||||
| } | ||||
|  | ||||
| pub struct RxRunner<'d, const MTU: usize> { | ||||
|     rx_chan: zerocopy_channel::Sender<'d, NoopRawMutex, PacketBuf<MTU>>, | ||||
| } | ||||
|  | ||||
| pub struct TxRunner<'d, const MTU: usize> { | ||||
|     tx_chan: zerocopy_channel::Receiver<'d, NoopRawMutex, PacketBuf<MTU>>, | ||||
| } | ||||
|  | ||||
| impl<'d, const MTU: usize> Runner<'d, MTU> { | ||||
|     pub fn split(self) -> (StateRunner<'d>, RxRunner<'d, MTU>, TxRunner<'d, MTU>) { | ||||
|         ( | ||||
|             StateRunner { shared: self.shared }, | ||||
|             RxRunner { rx_chan: self.rx_chan }, | ||||
|             TxRunner { tx_chan: self.tx_chan }, | ||||
|         ) | ||||
|     } | ||||
|  | ||||
|     pub fn state_runner(&self) -> StateRunner<'d> { | ||||
|         StateRunner { shared: self.shared } | ||||
|     } | ||||
|  | ||||
|     pub fn set_link_state(&mut self, state: LinkState) { | ||||
|         self.shared.lock(|s| { | ||||
|             let s = &mut *s.borrow_mut(); | ||||
|             s.link_state = state; | ||||
|             s.waker.wake(); | ||||
|         }); | ||||
|     } | ||||
|  | ||||
|     pub fn set_ethernet_address(&mut self, address: [u8; 6]) { | ||||
|         self.shared.lock(|s| { | ||||
|             let s = &mut *s.borrow_mut(); | ||||
|             s.ethernet_address = address; | ||||
|             s.waker.wake(); | ||||
|         }); | ||||
|     } | ||||
|  | ||||
|     pub async fn rx_buf(&mut self) -> &mut [u8] { | ||||
|         let p = self.rx_chan.send().await; | ||||
|         &mut p.buf | ||||
|     } | ||||
|  | ||||
|     pub fn try_rx_buf(&mut self) -> Option<&mut [u8]> { | ||||
|         let p = self.rx_chan.try_send()?; | ||||
|         Some(&mut p.buf) | ||||
|     } | ||||
|  | ||||
|     pub fn poll_rx_buf(&mut self, cx: &mut Context) -> Poll<&mut [u8]> { | ||||
|         match self.rx_chan.poll_send(cx) { | ||||
|             Poll::Ready(p) => Poll::Ready(&mut p.buf), | ||||
|             Poll::Pending => Poll::Pending, | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     pub fn rx_done(&mut self, len: usize) { | ||||
|         let p = self.rx_chan.try_send().unwrap(); | ||||
|         p.len = len; | ||||
|         self.rx_chan.send_done(); | ||||
|     } | ||||
|  | ||||
|     pub async fn tx_buf(&mut self) -> &mut [u8] { | ||||
|         let p = self.tx_chan.recv().await; | ||||
|         &mut p.buf[..p.len] | ||||
|     } | ||||
|  | ||||
|     pub fn try_tx_buf(&mut self) -> Option<&mut [u8]> { | ||||
|         let p = self.tx_chan.try_recv()?; | ||||
|         Some(&mut p.buf[..p.len]) | ||||
|     } | ||||
|  | ||||
|     pub fn poll_tx_buf(&mut self, cx: &mut Context) -> Poll<&mut [u8]> { | ||||
|         match self.tx_chan.poll_recv(cx) { | ||||
|             Poll::Ready(p) => Poll::Ready(&mut p.buf[..p.len]), | ||||
|             Poll::Pending => Poll::Pending, | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     pub fn tx_done(&mut self) { | ||||
|         self.tx_chan.recv_done(); | ||||
|     } | ||||
| } | ||||
|  | ||||
| impl<'d> StateRunner<'d> { | ||||
|     pub fn set_link_state(&self, state: LinkState) { | ||||
|         self.shared.lock(|s| { | ||||
|             let s = &mut *s.borrow_mut(); | ||||
|             s.link_state = state; | ||||
|             s.waker.wake(); | ||||
|         }); | ||||
|     } | ||||
|  | ||||
|     pub fn set_ethernet_address(&self, address: [u8; 6]) { | ||||
|         self.shared.lock(|s| { | ||||
|             let s = &mut *s.borrow_mut(); | ||||
|             s.ethernet_address = address; | ||||
|             s.waker.wake(); | ||||
|         }); | ||||
|     } | ||||
| } | ||||
|  | ||||
| impl<'d, const MTU: usize> RxRunner<'d, MTU> { | ||||
|     pub async fn rx_buf(&mut self) -> &mut [u8] { | ||||
|         let p = self.rx_chan.send().await; | ||||
|         &mut p.buf | ||||
|     } | ||||
|  | ||||
|     pub fn try_rx_buf(&mut self) -> Option<&mut [u8]> { | ||||
|         let p = self.rx_chan.try_send()?; | ||||
|         Some(&mut p.buf) | ||||
|     } | ||||
|  | ||||
|     pub fn poll_rx_buf(&mut self, cx: &mut Context) -> Poll<&mut [u8]> { | ||||
|         match self.rx_chan.poll_send(cx) { | ||||
|             Poll::Ready(p) => Poll::Ready(&mut p.buf), | ||||
|             Poll::Pending => Poll::Pending, | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     pub fn rx_done(&mut self, len: usize) { | ||||
|         let p = self.rx_chan.try_send().unwrap(); | ||||
|         p.len = len; | ||||
|         self.rx_chan.send_done(); | ||||
|     } | ||||
| } | ||||
|  | ||||
| impl<'d, const MTU: usize> TxRunner<'d, MTU> { | ||||
|     pub async fn tx_buf(&mut self) -> &mut [u8] { | ||||
|         let p = self.tx_chan.recv().await; | ||||
|         &mut p.buf[..p.len] | ||||
|     } | ||||
|  | ||||
|     pub fn try_tx_buf(&mut self) -> Option<&mut [u8]> { | ||||
|         let p = self.tx_chan.try_recv()?; | ||||
|         Some(&mut p.buf[..p.len]) | ||||
|     } | ||||
|  | ||||
|     pub fn poll_tx_buf(&mut self, cx: &mut Context) -> Poll<&mut [u8]> { | ||||
|         match self.tx_chan.poll_recv(cx) { | ||||
|             Poll::Ready(p) => Poll::Ready(&mut p.buf[..p.len]), | ||||
|             Poll::Pending => Poll::Pending, | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     pub fn tx_done(&mut self) { | ||||
|         self.tx_chan.recv_done(); | ||||
|     } | ||||
| } | ||||
|  | ||||
| pub fn new<'d, const MTU: usize, const N_RX: usize, const N_TX: usize>( | ||||
|     state: &'d mut State<MTU, N_RX, N_TX>, | ||||
|     ethernet_address: [u8; 6], | ||||
| ) -> (Runner<'d, MTU>, Device<'d, MTU>) { | ||||
|     let mut caps = Capabilities::default(); | ||||
|     caps.max_transmission_unit = MTU; | ||||
|     caps.medium = Medium::Ethernet; | ||||
|  | ||||
|     // safety: this is a self-referential struct, however: | ||||
|     // - it can't move while the `'d` borrow is active. | ||||
|     // - when the borrow ends, the dangling references inside the MaybeUninit will never be used again. | ||||
|     let state_uninit: *mut MaybeUninit<StateInner<'d, MTU>> = | ||||
|         (&mut state.inner as *mut MaybeUninit<StateInner<'static, MTU>>).cast(); | ||||
|     let state = unsafe { &mut *state_uninit }.write(StateInner { | ||||
|         rx: zerocopy_channel::Channel::new(&mut state.rx[..]), | ||||
|         tx: zerocopy_channel::Channel::new(&mut state.tx[..]), | ||||
|         shared: Mutex::new(RefCell::new(Shared { | ||||
|             link_state: LinkState::Down, | ||||
|             ethernet_address, | ||||
|             waker: WakerRegistration::new(), | ||||
|         })), | ||||
|     }); | ||||
|  | ||||
|     let (rx_sender, rx_receiver) = state.rx.split(); | ||||
|     let (tx_sender, tx_receiver) = state.tx.split(); | ||||
|  | ||||
|     ( | ||||
|         Runner { | ||||
|             tx_chan: tx_receiver, | ||||
|             rx_chan: rx_sender, | ||||
|             shared: &state.shared, | ||||
|         }, | ||||
|         Device { | ||||
|             caps, | ||||
|             shared: &state.shared, | ||||
|             rx: rx_receiver, | ||||
|             tx: tx_sender, | ||||
|         }, | ||||
|     ) | ||||
| } | ||||
|  | ||||
| pub struct PacketBuf<const MTU: usize> { | ||||
|     len: usize, | ||||
|     buf: [u8; MTU], | ||||
| } | ||||
|  | ||||
| impl<const MTU: usize> PacketBuf<MTU> { | ||||
|     pub const fn new() -> Self { | ||||
|         Self { len: 0, buf: [0; MTU] } | ||||
|     } | ||||
| } | ||||
|  | ||||
| pub struct Device<'d, const MTU: usize> { | ||||
|     rx: zerocopy_channel::Receiver<'d, NoopRawMutex, PacketBuf<MTU>>, | ||||
|     tx: zerocopy_channel::Sender<'d, NoopRawMutex, PacketBuf<MTU>>, | ||||
|     shared: &'d Mutex<NoopRawMutex, RefCell<Shared>>, | ||||
|     caps: Capabilities, | ||||
| } | ||||
|  | ||||
| impl<'d, const MTU: usize> embassy_net_driver::Driver for Device<'d, MTU> { | ||||
|     type RxToken<'a> = RxToken<'a, MTU> where Self: 'a ; | ||||
|     type TxToken<'a> = TxToken<'a, MTU> where Self: 'a ; | ||||
|  | ||||
|     fn receive(&mut self, cx: &mut Context) -> Option<(Self::RxToken<'_>, Self::TxToken<'_>)> { | ||||
|         if self.rx.poll_recv(cx).is_ready() && self.tx.poll_send(cx).is_ready() { | ||||
|             Some((RxToken { rx: self.rx.borrow() }, TxToken { tx: self.tx.borrow() })) | ||||
|         } else { | ||||
|             None | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     /// Construct a transmit token. | ||||
|     fn transmit(&mut self, cx: &mut Context) -> Option<Self::TxToken<'_>> { | ||||
|         if self.tx.poll_send(cx).is_ready() { | ||||
|             Some(TxToken { tx: self.tx.borrow() }) | ||||
|         } else { | ||||
|             None | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     /// Get a description of device capabilities. | ||||
|     fn capabilities(&self) -> Capabilities { | ||||
|         self.caps.clone() | ||||
|     } | ||||
|  | ||||
|     fn ethernet_address(&self) -> [u8; 6] { | ||||
|         self.shared.lock(|s| s.borrow().ethernet_address) | ||||
|     } | ||||
|  | ||||
|     fn link_state(&mut self, cx: &mut Context) -> LinkState { | ||||
|         self.shared.lock(|s| { | ||||
|             let s = &mut *s.borrow_mut(); | ||||
|             s.waker.register(cx.waker()); | ||||
|             s.link_state | ||||
|         }) | ||||
|     } | ||||
| } | ||||
|  | ||||
| pub struct RxToken<'a, const MTU: usize> { | ||||
|     rx: zerocopy_channel::Receiver<'a, NoopRawMutex, PacketBuf<MTU>>, | ||||
| } | ||||
|  | ||||
| impl<'a, const MTU: usize> embassy_net_driver::RxToken for RxToken<'a, MTU> { | ||||
|     fn consume<R, F>(mut self, f: F) -> R | ||||
|     where | ||||
|         F: FnOnce(&mut [u8]) -> R, | ||||
|     { | ||||
|         // NOTE(unwrap): we checked the queue wasn't full when creating the token. | ||||
|         let pkt = unwrap!(self.rx.try_recv()); | ||||
|         let r = f(&mut pkt.buf[..pkt.len]); | ||||
|         self.rx.recv_done(); | ||||
|         r | ||||
|     } | ||||
| } | ||||
|  | ||||
| pub struct TxToken<'a, const MTU: usize> { | ||||
|     tx: zerocopy_channel::Sender<'a, NoopRawMutex, PacketBuf<MTU>>, | ||||
| } | ||||
|  | ||||
| impl<'a, const MTU: usize> embassy_net_driver::TxToken for TxToken<'a, MTU> { | ||||
|     fn consume<R, F>(mut self, len: usize, f: F) -> R | ||||
|     where | ||||
|         F: FnOnce(&mut [u8]) -> R, | ||||
|     { | ||||
|         // NOTE(unwrap): we checked the queue wasn't full when creating the token. | ||||
|         let pkt = unwrap!(self.tx.try_send()); | ||||
|         let r = f(&mut pkt.buf[..len]); | ||||
|         pkt.len = len; | ||||
|         self.tx.send_done(); | ||||
|         r | ||||
|     } | ||||
| } | ||||
|  | ||||
| mod zerocopy_channel { | ||||
|     use core::cell::RefCell; | ||||
|     use core::future::poll_fn; | ||||
|     use core::marker::PhantomData; | ||||
|     use core::task::{Context, Poll}; | ||||
|  | ||||
|     use embassy_sync::blocking_mutex::raw::RawMutex; | ||||
|     use embassy_sync::blocking_mutex::Mutex; | ||||
|     use embassy_sync::waitqueue::WakerRegistration; | ||||
|  | ||||
|     pub struct Channel<'a, M: RawMutex, T> { | ||||
|         buf: *mut T, | ||||
|         phantom: PhantomData<&'a mut T>, | ||||
|         state: Mutex<M, RefCell<State>>, | ||||
|     } | ||||
|  | ||||
|     impl<'a, M: RawMutex, T> Channel<'a, M, T> { | ||||
|         pub fn new(buf: &'a mut [T]) -> Self { | ||||
|             let len = buf.len(); | ||||
|             assert!(len != 0); | ||||
|  | ||||
|             Self { | ||||
|                 buf: buf.as_mut_ptr(), | ||||
|                 phantom: PhantomData, | ||||
|                 state: Mutex::new(RefCell::new(State { | ||||
|                     len, | ||||
|                     front: 0, | ||||
|                     back: 0, | ||||
|                     full: false, | ||||
|                     send_waker: WakerRegistration::new(), | ||||
|                     recv_waker: WakerRegistration::new(), | ||||
|                 })), | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         pub fn split(&mut self) -> (Sender<'_, M, T>, Receiver<'_, M, T>) { | ||||
|             (Sender { channel: self }, Receiver { channel: self }) | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     pub struct Sender<'a, M: RawMutex, T> { | ||||
|         channel: &'a Channel<'a, M, T>, | ||||
|     } | ||||
|  | ||||
|     impl<'a, M: RawMutex, T> Sender<'a, M, T> { | ||||
|         pub fn borrow(&mut self) -> Sender<'_, M, T> { | ||||
|             Sender { channel: self.channel } | ||||
|         } | ||||
|  | ||||
|         pub fn try_send(&mut self) -> Option<&mut T> { | ||||
|             self.channel.state.lock(|s| { | ||||
|                 let s = &mut *s.borrow_mut(); | ||||
|                 match s.push_index() { | ||||
|                     Some(i) => Some(unsafe { &mut *self.channel.buf.add(i) }), | ||||
|                     None => None, | ||||
|                 } | ||||
|             }) | ||||
|         } | ||||
|  | ||||
|         pub fn poll_send(&mut self, cx: &mut Context) -> Poll<&mut T> { | ||||
|             self.channel.state.lock(|s| { | ||||
|                 let s = &mut *s.borrow_mut(); | ||||
|                 match s.push_index() { | ||||
|                     Some(i) => Poll::Ready(unsafe { &mut *self.channel.buf.add(i) }), | ||||
|                     None => { | ||||
|                         s.recv_waker.register(cx.waker()); | ||||
|                         Poll::Pending | ||||
|                     } | ||||
|                 } | ||||
|             }) | ||||
|         } | ||||
|  | ||||
|         pub async fn send(&mut self) -> &mut T { | ||||
|             let i = poll_fn(|cx| { | ||||
|                 self.channel.state.lock(|s| { | ||||
|                     let s = &mut *s.borrow_mut(); | ||||
|                     match s.push_index() { | ||||
|                         Some(i) => Poll::Ready(i), | ||||
|                         None => { | ||||
|                             s.recv_waker.register(cx.waker()); | ||||
|                             Poll::Pending | ||||
|                         } | ||||
|                     } | ||||
|                 }) | ||||
|             }) | ||||
|             .await; | ||||
|             unsafe { &mut *self.channel.buf.add(i) } | ||||
|         } | ||||
|  | ||||
|         pub fn send_done(&mut self) { | ||||
|             self.channel.state.lock(|s| s.borrow_mut().push_done()) | ||||
|         } | ||||
|     } | ||||
|     pub struct Receiver<'a, M: RawMutex, T> { | ||||
|         channel: &'a Channel<'a, M, T>, | ||||
|     } | ||||
|  | ||||
|     impl<'a, M: RawMutex, T> Receiver<'a, M, T> { | ||||
|         pub fn borrow(&mut self) -> Receiver<'_, M, T> { | ||||
|             Receiver { channel: self.channel } | ||||
|         } | ||||
|  | ||||
|         pub fn try_recv(&mut self) -> Option<&mut T> { | ||||
|             self.channel.state.lock(|s| { | ||||
|                 let s = &mut *s.borrow_mut(); | ||||
|                 match s.pop_index() { | ||||
|                     Some(i) => Some(unsafe { &mut *self.channel.buf.add(i) }), | ||||
|                     None => None, | ||||
|                 } | ||||
|             }) | ||||
|         } | ||||
|  | ||||
|         pub fn poll_recv(&mut self, cx: &mut Context) -> Poll<&mut T> { | ||||
|             self.channel.state.lock(|s| { | ||||
|                 let s = &mut *s.borrow_mut(); | ||||
|                 match s.pop_index() { | ||||
|                     Some(i) => Poll::Ready(unsafe { &mut *self.channel.buf.add(i) }), | ||||
|                     None => { | ||||
|                         s.send_waker.register(cx.waker()); | ||||
|                         Poll::Pending | ||||
|                     } | ||||
|                 } | ||||
|             }) | ||||
|         } | ||||
|  | ||||
|         pub async fn recv(&mut self) -> &mut T { | ||||
|             let i = poll_fn(|cx| { | ||||
|                 self.channel.state.lock(|s| { | ||||
|                     let s = &mut *s.borrow_mut(); | ||||
|                     match s.pop_index() { | ||||
|                         Some(i) => Poll::Ready(i), | ||||
|                         None => { | ||||
|                             s.send_waker.register(cx.waker()); | ||||
|                             Poll::Pending | ||||
|                         } | ||||
|                     } | ||||
|                 }) | ||||
|             }) | ||||
|             .await; | ||||
|             unsafe { &mut *self.channel.buf.add(i) } | ||||
|         } | ||||
|  | ||||
|         pub fn recv_done(&mut self) { | ||||
|             self.channel.state.lock(|s| s.borrow_mut().pop_done()) | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     struct State { | ||||
|         len: usize, | ||||
|  | ||||
|         /// Front index. Always 0..=(N-1) | ||||
|         front: usize, | ||||
|         /// Back index. Always 0..=(N-1). | ||||
|         back: usize, | ||||
|  | ||||
|         /// Used to distinguish "empty" and "full" cases when `front == back`. | ||||
|         /// May only be `true` if `front == back`, always `false` otherwise. | ||||
|         full: bool, | ||||
|  | ||||
|         send_waker: WakerRegistration, | ||||
|         recv_waker: WakerRegistration, | ||||
|     } | ||||
|  | ||||
|     impl State { | ||||
|         fn increment(&self, i: usize) -> usize { | ||||
|             if i + 1 == self.len { | ||||
|                 0 | ||||
|             } else { | ||||
|                 i + 1 | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         fn is_full(&self) -> bool { | ||||
|             self.full | ||||
|         } | ||||
|  | ||||
|         fn is_empty(&self) -> bool { | ||||
|             self.front == self.back && !self.full | ||||
|         } | ||||
|  | ||||
|         fn push_index(&mut self) -> Option<usize> { | ||||
|             match self.is_full() { | ||||
|                 true => None, | ||||
|                 false => Some(self.back), | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         fn push_done(&mut self) { | ||||
|             assert!(!self.is_full()); | ||||
|             self.back = self.increment(self.back); | ||||
|             if self.back == self.front { | ||||
|                 self.full = true; | ||||
|             } | ||||
|             self.send_waker.wake(); | ||||
|         } | ||||
|  | ||||
|         fn pop_index(&mut self) -> Option<usize> { | ||||
|             match self.is_empty() { | ||||
|                 true => None, | ||||
|                 false => Some(self.front), | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         fn pop_done(&mut self) { | ||||
|             assert!(!self.is_empty()); | ||||
|             self.front = self.increment(self.front); | ||||
|             self.full = false; | ||||
|             self.recv_waker.wake(); | ||||
|         } | ||||
|     } | ||||
| } | ||||
							
								
								
									
										15
									
								
								embassy-net-driver/Cargo.toml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										15
									
								
								embassy-net-driver/Cargo.toml
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,15 @@ | ||||
| [package] | ||||
| name = "embassy-net-driver" | ||||
| version = "0.1.0" | ||||
| edition = "2021" | ||||
| license = "MIT OR Apache-2.0" | ||||
|  | ||||
|  | ||||
| [package.metadata.embassy_docs] | ||||
| src_base = "https://github.com/embassy-rs/embassy/blob/embassy-net-driver-v$VERSION/embassy-net-driver/src/" | ||||
| src_base_git = "https://github.com/embassy-rs/embassy/blob/$COMMIT/embassy-net-driver/src/" | ||||
| features = ["defmt"] | ||||
| target = "thumbv7em-none-eabi" | ||||
|  | ||||
| [dependencies] | ||||
| defmt = { version = "0.3", optional = true } | ||||
							
								
								
									
										175
									
								
								embassy-net-driver/src/lib.rs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										175
									
								
								embassy-net-driver/src/lib.rs
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,175 @@ | ||||
| #![no_std] | ||||
|  | ||||
| use core::task::Context; | ||||
|  | ||||
| pub trait Driver { | ||||
|     type RxToken<'a>: RxToken | ||||
|     where | ||||
|         Self: 'a; | ||||
|     type TxToken<'a>: TxToken | ||||
|     where | ||||
|         Self: 'a; | ||||
|  | ||||
|     fn receive(&mut self, cx: &mut Context) -> Option<(Self::RxToken<'_>, Self::TxToken<'_>)>; | ||||
|     fn transmit(&mut self, cx: &mut Context) -> Option<Self::TxToken<'_>>; | ||||
|     fn link_state(&mut self, cx: &mut Context) -> LinkState; | ||||
|  | ||||
|     fn capabilities(&self) -> Capabilities; | ||||
|     fn ethernet_address(&self) -> [u8; 6]; | ||||
| } | ||||
|  | ||||
| impl<T: ?Sized + Driver> Driver for &mut T { | ||||
|     type RxToken<'a> = T::RxToken<'a> | ||||
|     where | ||||
|         Self: 'a; | ||||
|     type TxToken<'a> = T::TxToken<'a> | ||||
|     where | ||||
|         Self: 'a; | ||||
|  | ||||
|     fn transmit(&mut self, cx: &mut Context) -> Option<Self::TxToken<'_>> { | ||||
|         T::transmit(self, cx) | ||||
|     } | ||||
|     fn receive(&mut self, cx: &mut Context) -> Option<(Self::RxToken<'_>, Self::TxToken<'_>)> { | ||||
|         T::receive(self, cx) | ||||
|     } | ||||
|     fn capabilities(&self) -> Capabilities { | ||||
|         T::capabilities(self) | ||||
|     } | ||||
|     fn link_state(&mut self, cx: &mut Context) -> LinkState { | ||||
|         T::link_state(self, cx) | ||||
|     } | ||||
|     fn ethernet_address(&self) -> [u8; 6] { | ||||
|         T::ethernet_address(self) | ||||
|     } | ||||
| } | ||||
|  | ||||
| /// A token to receive a single network packet. | ||||
| pub trait RxToken { | ||||
|     /// Consumes the token to receive a single network packet. | ||||
|     /// | ||||
|     /// This method receives a packet and then calls the given closure `f` with the raw | ||||
|     /// packet bytes as argument. | ||||
|     fn consume<R, F>(self, f: F) -> R | ||||
|     where | ||||
|         F: FnOnce(&mut [u8]) -> R; | ||||
| } | ||||
|  | ||||
| /// A token to transmit a single network packet. | ||||
| pub trait TxToken { | ||||
|     /// Consumes the token to send a single network packet. | ||||
|     /// | ||||
|     /// This method constructs a transmit buffer of size `len` and calls the passed | ||||
|     /// closure `f` with a mutable reference to that buffer. The closure should construct | ||||
|     /// a valid network packet (e.g. an ethernet packet) in the buffer. When the closure | ||||
|     /// returns, the transmit buffer is sent out. | ||||
|     fn consume<R, F>(self, len: usize, f: F) -> R | ||||
|     where | ||||
|         F: FnOnce(&mut [u8]) -> R; | ||||
| } | ||||
|  | ||||
| /// A description of device capabilities. | ||||
| /// | ||||
| /// Higher-level protocols may achieve higher throughput or lower latency if they consider | ||||
| /// the bandwidth or packet size limitations. | ||||
| #[derive(Debug, Clone, Default)] | ||||
| #[cfg_attr(feature = "defmt", derive(defmt::Format))] | ||||
| #[non_exhaustive] | ||||
| pub struct Capabilities { | ||||
|     /// Medium of the device. | ||||
|     /// | ||||
|     /// This indicates what kind of packet the sent/received bytes are, and determines | ||||
|     /// some behaviors of Interface. For example, ARP/NDISC address resolution is only done | ||||
|     /// for Ethernet mediums. | ||||
|     pub medium: Medium, | ||||
|  | ||||
|     /// Maximum transmission unit. | ||||
|     /// | ||||
|     /// The network device is unable to send or receive frames larger than the value returned | ||||
|     /// by this function. | ||||
|     /// | ||||
|     /// For Ethernet devices, this is the maximum Ethernet frame size, including the Ethernet header (14 octets), but | ||||
|     /// *not* including the Ethernet FCS (4 octets). Therefore, Ethernet MTU = IP MTU + 14. | ||||
|     /// | ||||
|     /// Note that in Linux and other OSes, "MTU" is the IP MTU, not the Ethernet MTU, even for Ethernet | ||||
|     /// devices. This is a common source of confusion. | ||||
|     /// | ||||
|     /// Most common IP MTU is 1500. Minimum is 576 (for IPv4) or 1280 (for IPv6). Maximum is 9216 octets. | ||||
|     pub max_transmission_unit: usize, | ||||
|  | ||||
|     /// Maximum burst size, in terms of MTU. | ||||
|     /// | ||||
|     /// The network device is unable to send or receive bursts large than the value returned | ||||
|     /// by this function. | ||||
|     /// | ||||
|     /// If `None`, there is no fixed limit on burst size, e.g. if network buffers are | ||||
|     /// dynamically allocated. | ||||
|     pub max_burst_size: Option<usize>, | ||||
|  | ||||
|     /// Checksum behavior. | ||||
|     /// | ||||
|     /// If the network device is capable of verifying or computing checksums for some protocols, | ||||
|     /// it can request that the stack not do so in software to improve performance. | ||||
|     pub checksum: ChecksumCapabilities, | ||||
| } | ||||
|  | ||||
| /// Type of medium of a device. | ||||
| #[derive(Debug, Eq, PartialEq, Copy, Clone)] | ||||
| #[cfg_attr(feature = "defmt", derive(defmt::Format))] | ||||
| pub enum Medium { | ||||
|     /// Ethernet medium. Devices of this type send and receive Ethernet frames, | ||||
|     /// and interfaces using it must do neighbor discovery via ARP or NDISC. | ||||
|     /// | ||||
|     /// Examples of devices of this type are Ethernet, WiFi (802.11), Linux `tap`, and VPNs in tap (layer 2) mode. | ||||
|     Ethernet, | ||||
|  | ||||
|     /// IP medium. Devices of this type send and receive IP frames, without an | ||||
|     /// Ethernet header. MAC addresses are not used, and no neighbor discovery (ARP, NDISC) is done. | ||||
|     /// | ||||
|     /// Examples of devices of this type are the Linux `tun`, PPP interfaces, VPNs in tun (layer 3) mode. | ||||
|     Ip, | ||||
| } | ||||
|  | ||||
| impl Default for Medium { | ||||
|     fn default() -> Medium { | ||||
|         Medium::Ethernet | ||||
|     } | ||||
| } | ||||
|  | ||||
| /// A description of checksum behavior for every supported protocol. | ||||
| #[derive(Debug, Clone, Default)] | ||||
| #[cfg_attr(feature = "defmt", derive(defmt::Format))] | ||||
| #[non_exhaustive] | ||||
| pub struct ChecksumCapabilities { | ||||
|     pub ipv4: Checksum, | ||||
|     pub udp: Checksum, | ||||
|     pub tcp: Checksum, | ||||
|     pub icmpv4: Checksum, | ||||
|     pub icmpv6: Checksum, | ||||
| } | ||||
|  | ||||
| /// A description of checksum behavior for a particular protocol. | ||||
| #[derive(Debug, Clone, Copy)] | ||||
| #[cfg_attr(feature = "defmt", derive(defmt::Format))] | ||||
| pub enum Checksum { | ||||
|     /// Verify checksum when receiving and compute checksum when sending. | ||||
|     Both, | ||||
|     /// Verify checksum when receiving. | ||||
|     Rx, | ||||
|     /// Compute checksum before sending. | ||||
|     Tx, | ||||
|     /// Ignore checksum completely. | ||||
|     None, | ||||
| } | ||||
|  | ||||
| impl Default for Checksum { | ||||
|     fn default() -> Checksum { | ||||
|         Checksum::Both | ||||
|     } | ||||
| } | ||||
|  | ||||
| #[derive(PartialEq, Eq, Clone, Copy)] | ||||
| #[cfg_attr(feature = "defmt", derive(defmt::Format))] | ||||
| pub enum LinkState { | ||||
|     Down, | ||||
|     Up, | ||||
| } | ||||
| @@ -2,44 +2,49 @@ | ||||
| name = "embassy-net" | ||||
| version = "0.1.0" | ||||
| edition = "2021" | ||||
| license = "MIT OR Apache-2.0" | ||||
|  | ||||
|  | ||||
| [package.metadata.embassy_docs] | ||||
| src_base = "https://github.com/embassy-rs/embassy/blob/embassy-net-v$VERSION/embassy-net/src/" | ||||
| src_base_git = "https://github.com/embassy-rs/embassy/blob/$COMMIT/embassy-net/src/" | ||||
| features = [ "pool-4", "defmt", "tcp", "dns", "dhcpv4", "proto-ipv6", "medium-ethernet", "medium-ip", "embassy-time/tick-1mhz"] | ||||
| features = ["nightly", "unstable-traits", "defmt", "tcp", "udp", "dns", "dhcpv4", "proto-ipv6", "medium-ethernet", "medium-ip", "igmp"] | ||||
| target = "thumbv7em-none-eabi" | ||||
|  | ||||
| [features] | ||||
| default = [] | ||||
| std = [] | ||||
|  | ||||
| defmt = ["dep:defmt", "smoltcp/defmt"] | ||||
| defmt = ["dep:defmt", "smoltcp/defmt", "embassy-net-driver/defmt"] | ||||
|  | ||||
| nightly = ["dep:embedded-io", "embedded-io?/async", "dep:embedded-nal-async"] | ||||
| unstable-traits = [] | ||||
|  | ||||
| udp = ["smoltcp/socket-udp"] | ||||
| tcp = ["smoltcp/socket-tcp"] | ||||
| dns = ["smoltcp/socket-dns"] | ||||
| dns = ["smoltcp/socket-dns", "smoltcp/proto-dns"] | ||||
| dhcpv4 = ["medium-ethernet", "smoltcp/socket-dhcpv4"] | ||||
| proto-ipv6 = ["smoltcp/proto-ipv6"] | ||||
| medium-ethernet = ["smoltcp/medium-ethernet"] | ||||
| medium-ip = ["smoltcp/medium-ip"] | ||||
|  | ||||
| pool-4 = [] | ||||
| pool-8 = [] | ||||
| pool-16 = [] | ||||
| pool-32 = [] | ||||
| pool-64 = [] | ||||
| pool-128 = [] | ||||
| unstable-traits = [] | ||||
| igmp = ["smoltcp/proto-igmp"] | ||||
|  | ||||
| [dependencies] | ||||
|  | ||||
| defmt = { version = "0.3", optional = true } | ||||
| log = { version = "0.4.14", optional = true } | ||||
|  | ||||
| smoltcp = { version = "0.9.0", default-features = false, features = [ | ||||
|   "proto-ipv4", | ||||
|   "socket", | ||||
|   "async", | ||||
| ]} | ||||
|  | ||||
| embassy-net-driver = { version = "0.1.0", path = "../embassy-net-driver" } | ||||
| embassy-hal-common = { version = "0.1.0", path = "../embassy-hal-common" } | ||||
| embassy-time = { version = "0.1.0", path = "../embassy-time" } | ||||
| embassy-sync = { version = "0.1.0", path = "../embassy-sync" } | ||||
| embedded-io = { version = "0.3.0", features = [ "async" ] } | ||||
| embassy-sync = { version = "0.2.0", path = "../embassy-sync" } | ||||
| embedded-io = { version = "0.4.0", optional = true } | ||||
|  | ||||
| managed = { version = "0.8.0", default-features = false, features = [ "map" ] } | ||||
| heapless = { version = "0.7.5", default-features = false } | ||||
| @@ -48,16 +53,5 @@ generic-array = { version = "0.14.4", default-features = false } | ||||
| stable_deref_trait = { version = "1.2.0", default-features = false } | ||||
| futures = { version = "0.3.17", default-features = false, features = [ "async-await" ] } | ||||
| atomic-pool = "1.0" | ||||
| atomic-polyfill = "1.0.1" | ||||
| embedded-nal-async = "0.2.0" | ||||
|  | ||||
| [dependencies.smoltcp] | ||||
| version = "0.8.0" | ||||
| git = "https://github.com/smoltcp-rs/smoltcp" | ||||
| rev = "ed0cf16750a42f30e31fcaf5347915592924b1e3" | ||||
| default-features = false | ||||
| features = [ | ||||
|   "proto-ipv4", | ||||
|   "socket", | ||||
|   "async", | ||||
| ] | ||||
| embedded-nal-async = { version = "0.4.0", optional = true } | ||||
| atomic-polyfill = { version = "1.0" } | ||||
|   | ||||
| @@ -17,11 +17,13 @@ sudo ip -6 route add fe80::/64 dev tap0 | ||||
| sudo ip -6 route add fdaa::/64 dev tap0 | ||||
| ``` | ||||
|  | ||||
| Second, have something listening there. For example `nc -l 8000` | ||||
|  | ||||
| Then run the example located in the `examples` folder: | ||||
|  | ||||
| ```sh | ||||
| cd $EMBASSY_ROOT/examples/std/ | ||||
| cargo run --bin net | ||||
| cargo run --bin net -- --static-ip | ||||
| ``` | ||||
|  | ||||
| ## License | ||||
|   | ||||
| @@ -1,129 +1,102 @@ | ||||
| use core::task::Waker; | ||||
| use core::task::Context; | ||||
|  | ||||
| use smoltcp::phy::{Device as SmolDevice, DeviceCapabilities}; | ||||
| use smoltcp::time::Instant as SmolInstant; | ||||
| use embassy_net_driver::{Capabilities, Checksum, Driver, Medium, RxToken, TxToken}; | ||||
| use smoltcp::phy; | ||||
| use smoltcp::time::Instant; | ||||
|  | ||||
| use crate::packet_pool::PacketBoxExt; | ||||
| use crate::{Packet, PacketBox, PacketBuf}; | ||||
|  | ||||
| #[derive(PartialEq, Eq, Clone, Copy)] | ||||
| pub enum LinkState { | ||||
|     Down, | ||||
|     Up, | ||||
| pub(crate) struct DriverAdapter<'d, 'c, T> | ||||
| where | ||||
|     T: Driver, | ||||
| { | ||||
|     // must be Some when actually using this to rx/tx | ||||
|     pub cx: Option<&'d mut Context<'c>>, | ||||
|     pub inner: &'d mut T, | ||||
| } | ||||
|  | ||||
| // 'static required due to the "fake GAT" in smoltcp::phy::Device. | ||||
| // https://github.com/smoltcp-rs/smoltcp/pull/572 | ||||
| pub trait Device { | ||||
|     fn is_transmit_ready(&mut self) -> bool; | ||||
|     fn transmit(&mut self, pkt: PacketBuf); | ||||
|     fn receive(&mut self) -> Option<PacketBuf>; | ||||
| impl<'d, 'c, T> phy::Device for DriverAdapter<'d, 'c, T> | ||||
| where | ||||
|     T: Driver, | ||||
| { | ||||
|     type RxToken<'a> = RxTokenAdapter<T::RxToken<'a>> where Self: 'a; | ||||
|     type TxToken<'a> = TxTokenAdapter<T::TxToken<'a>> where Self: 'a; | ||||
|  | ||||
|     fn register_waker(&mut self, waker: &Waker); | ||||
|     fn capabilities(&self) -> DeviceCapabilities; | ||||
|     fn link_state(&mut self) -> LinkState; | ||||
|     fn ethernet_address(&self) -> [u8; 6]; | ||||
| } | ||||
|  | ||||
| impl<T: ?Sized + Device> Device for &'static mut T { | ||||
|     fn is_transmit_ready(&mut self) -> bool { | ||||
|         T::is_transmit_ready(self) | ||||
|     } | ||||
|     fn transmit(&mut self, pkt: PacketBuf) { | ||||
|         T::transmit(self, pkt) | ||||
|     } | ||||
|     fn receive(&mut self) -> Option<PacketBuf> { | ||||
|         T::receive(self) | ||||
|     } | ||||
|     fn register_waker(&mut self, waker: &Waker) { | ||||
|         T::register_waker(self, waker) | ||||
|     } | ||||
|     fn capabilities(&self) -> DeviceCapabilities { | ||||
|         T::capabilities(self) | ||||
|     } | ||||
|     fn link_state(&mut self) -> LinkState { | ||||
|         T::link_state(self) | ||||
|     } | ||||
|     fn ethernet_address(&self) -> [u8; 6] { | ||||
|         T::ethernet_address(self) | ||||
|     } | ||||
| } | ||||
|  | ||||
| pub struct DeviceAdapter<D: Device> { | ||||
|     pub device: D, | ||||
|     caps: DeviceCapabilities, | ||||
| } | ||||
|  | ||||
| impl<D: Device> DeviceAdapter<D> { | ||||
|     pub(crate) fn new(device: D) -> Self { | ||||
|         Self { | ||||
|             caps: device.capabilities(), | ||||
|             device, | ||||
|         } | ||||
|     } | ||||
| } | ||||
|  | ||||
| impl<'a, D: Device + 'static> SmolDevice<'a> for DeviceAdapter<D> { | ||||
|     type RxToken = RxToken; | ||||
|     type TxToken = TxToken<'a, D>; | ||||
|  | ||||
|     fn receive(&'a mut self) -> Option<(Self::RxToken, Self::TxToken)> { | ||||
|         let tx_pkt = PacketBox::new(Packet::new())?; | ||||
|         let rx_pkt = self.device.receive()?; | ||||
|         let rx_token = RxToken { pkt: rx_pkt }; | ||||
|         let tx_token = TxToken { | ||||
|             device: &mut self.device, | ||||
|             pkt: tx_pkt, | ||||
|         }; | ||||
|  | ||||
|         Some((rx_token, tx_token)) | ||||
|     fn receive(&mut self, _timestamp: Instant) -> Option<(Self::RxToken<'_>, Self::TxToken<'_>)> { | ||||
|         self.inner | ||||
|             .receive(self.cx.as_deref_mut().unwrap()) | ||||
|             .map(|(rx, tx)| (RxTokenAdapter(rx), TxTokenAdapter(tx))) | ||||
|     } | ||||
|  | ||||
|     /// Construct a transmit token. | ||||
|     fn transmit(&'a mut self) -> Option<Self::TxToken> { | ||||
|         if !self.device.is_transmit_ready() { | ||||
|             return None; | ||||
|         } | ||||
|  | ||||
|         let tx_pkt = PacketBox::new(Packet::new())?; | ||||
|         Some(TxToken { | ||||
|             device: &mut self.device, | ||||
|             pkt: tx_pkt, | ||||
|         }) | ||||
|     fn transmit(&mut self, _timestamp: Instant) -> Option<Self::TxToken<'_>> { | ||||
|         self.inner.transmit(self.cx.as_deref_mut().unwrap()).map(TxTokenAdapter) | ||||
|     } | ||||
|  | ||||
|     /// Get a description of device capabilities. | ||||
|     fn capabilities(&self) -> DeviceCapabilities { | ||||
|         self.caps.clone() | ||||
|     fn capabilities(&self) -> phy::DeviceCapabilities { | ||||
|         fn convert(c: Checksum) -> phy::Checksum { | ||||
|             match c { | ||||
|                 Checksum::Both => phy::Checksum::Both, | ||||
|                 Checksum::Tx => phy::Checksum::Tx, | ||||
|                 Checksum::Rx => phy::Checksum::Rx, | ||||
|                 Checksum::None => phy::Checksum::None, | ||||
|             } | ||||
|         } | ||||
|         let caps: Capabilities = self.inner.capabilities(); | ||||
|         let mut smolcaps = phy::DeviceCapabilities::default(); | ||||
|  | ||||
|         smolcaps.max_transmission_unit = caps.max_transmission_unit; | ||||
|         smolcaps.max_burst_size = caps.max_burst_size; | ||||
|         smolcaps.medium = match caps.medium { | ||||
|             #[cfg(feature = "medium-ethernet")] | ||||
|             Medium::Ethernet => phy::Medium::Ethernet, | ||||
|             #[cfg(feature = "medium-ip")] | ||||
|             Medium::Ip => phy::Medium::Ip, | ||||
|             _ => panic!( | ||||
|                 "Unsupported medium {:?}. MAke sure to enable it in embassy-net's Cargo features.", | ||||
|                 caps.medium | ||||
|             ), | ||||
|         }; | ||||
|         smolcaps.checksum.ipv4 = convert(caps.checksum.ipv4); | ||||
|         smolcaps.checksum.tcp = convert(caps.checksum.tcp); | ||||
|         smolcaps.checksum.udp = convert(caps.checksum.udp); | ||||
|         smolcaps.checksum.icmpv4 = convert(caps.checksum.icmpv4); | ||||
|         #[cfg(feature = "proto-ipv6")] | ||||
|         { | ||||
|             smolcaps.checksum.icmpv6 = convert(caps.checksum.icmpv6); | ||||
|         } | ||||
|  | ||||
|         smolcaps | ||||
|     } | ||||
| } | ||||
|  | ||||
| pub struct RxToken { | ||||
|     pkt: PacketBuf, | ||||
| } | ||||
| pub(crate) struct RxTokenAdapter<T>(T) | ||||
| where | ||||
|     T: RxToken; | ||||
|  | ||||
| impl smoltcp::phy::RxToken for RxToken { | ||||
|     fn consume<R, F>(mut self, _timestamp: SmolInstant, f: F) -> smoltcp::Result<R> | ||||
| impl<T> phy::RxToken for RxTokenAdapter<T> | ||||
| where | ||||
|     T: RxToken, | ||||
| { | ||||
|     fn consume<R, F>(self, f: F) -> R | ||||
|     where | ||||
|         F: FnOnce(&mut [u8]) -> smoltcp::Result<R>, | ||||
|         F: FnOnce(&mut [u8]) -> R, | ||||
|     { | ||||
|         f(&mut self.pkt) | ||||
|         self.0.consume(|buf| f(buf)) | ||||
|     } | ||||
| } | ||||
|  | ||||
| pub struct TxToken<'a, D: Device> { | ||||
|     device: &'a mut D, | ||||
|     pkt: PacketBox, | ||||
| } | ||||
| pub(crate) struct TxTokenAdapter<T>(T) | ||||
| where | ||||
|     T: TxToken; | ||||
|  | ||||
| impl<'a, D: Device> smoltcp::phy::TxToken for TxToken<'a, D> { | ||||
|     fn consume<R, F>(self, _timestamp: SmolInstant, len: usize, f: F) -> smoltcp::Result<R> | ||||
| impl<T> phy::TxToken for TxTokenAdapter<T> | ||||
| where | ||||
|     T: TxToken, | ||||
| { | ||||
|     fn consume<R, F>(self, len: usize, f: F) -> R | ||||
|     where | ||||
|         F: FnOnce(&mut [u8]) -> smoltcp::Result<R>, | ||||
|         F: FnOnce(&mut [u8]) -> R, | ||||
|     { | ||||
|         let mut buf = self.pkt.slice(0..len); | ||||
|         let r = f(&mut buf)?; | ||||
|         self.device.transmit(buf); | ||||
|         Ok(r) | ||||
|         self.0.consume(len, |buf| f(buf)) | ||||
|     } | ||||
| } | ||||
|   | ||||
							
								
								
									
										97
									
								
								embassy-net/src/dns.rs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										97
									
								
								embassy-net/src/dns.rs
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,97 @@ | ||||
| //! DNS socket with async support. | ||||
| use heapless::Vec; | ||||
| pub use smoltcp::socket::dns::{DnsQuery, Socket}; | ||||
| pub(crate) use smoltcp::socket::dns::{GetQueryResultError, StartQueryError}; | ||||
| pub use smoltcp::wire::{DnsQueryType, IpAddress}; | ||||
|  | ||||
| use crate::{Driver, Stack}; | ||||
|  | ||||
| /// Errors returned by DnsSocket. | ||||
| #[derive(Debug, PartialEq, Eq, Clone, Copy)] | ||||
| #[cfg_attr(feature = "defmt", derive(defmt::Format))] | ||||
| pub enum Error { | ||||
|     /// Invalid name | ||||
|     InvalidName, | ||||
|     /// Name too long | ||||
|     NameTooLong, | ||||
|     /// Name lookup failed | ||||
|     Failed, | ||||
| } | ||||
|  | ||||
| impl From<GetQueryResultError> for Error { | ||||
|     fn from(_: GetQueryResultError) -> Self { | ||||
|         Self::Failed | ||||
|     } | ||||
| } | ||||
|  | ||||
| impl From<StartQueryError> for Error { | ||||
|     fn from(e: StartQueryError) -> Self { | ||||
|         match e { | ||||
|             StartQueryError::NoFreeSlot => Self::Failed, | ||||
|             StartQueryError::InvalidName => Self::InvalidName, | ||||
|             StartQueryError::NameTooLong => Self::NameTooLong, | ||||
|         } | ||||
|     } | ||||
| } | ||||
|  | ||||
| /// Async socket for making DNS queries. | ||||
| pub struct DnsSocket<'a, D> | ||||
| where | ||||
|     D: Driver + 'static, | ||||
| { | ||||
|     stack: &'a Stack<D>, | ||||
| } | ||||
|  | ||||
| impl<'a, D> DnsSocket<'a, D> | ||||
| where | ||||
|     D: Driver + 'static, | ||||
| { | ||||
|     /// Create a new DNS socket using the provided stack. | ||||
|     /// | ||||
|     /// NOTE: If using DHCP, make sure it has reconfigured the stack to ensure the DNS servers are updated. | ||||
|     pub fn new(stack: &'a Stack<D>) -> Self { | ||||
|         Self { stack } | ||||
|     } | ||||
|  | ||||
|     /// Make a query for a given name and return the corresponding IP addresses. | ||||
|     pub async fn query(&self, name: &str, qtype: DnsQueryType) -> Result<Vec<IpAddress, 1>, Error> { | ||||
|         self.stack.dns_query(name, qtype).await | ||||
|     } | ||||
| } | ||||
|  | ||||
| #[cfg(all(feature = "unstable-traits", feature = "nightly"))] | ||||
| impl<'a, D> embedded_nal_async::Dns for DnsSocket<'a, D> | ||||
| where | ||||
|     D: Driver + 'static, | ||||
| { | ||||
|     type Error = Error; | ||||
|  | ||||
|     async fn get_host_by_name( | ||||
|         &self, | ||||
|         host: &str, | ||||
|         addr_type: embedded_nal_async::AddrType, | ||||
|     ) -> Result<embedded_nal_async::IpAddr, Self::Error> { | ||||
|         use embedded_nal_async::{AddrType, IpAddr}; | ||||
|         let qtype = match addr_type { | ||||
|             AddrType::IPv6 => DnsQueryType::Aaaa, | ||||
|             _ => DnsQueryType::A, | ||||
|         }; | ||||
|         let addrs = self.query(host, qtype).await?; | ||||
|         if let Some(first) = addrs.get(0) { | ||||
|             Ok(match first { | ||||
|                 IpAddress::Ipv4(addr) => IpAddr::V4(addr.0.into()), | ||||
|                 #[cfg(feature = "proto-ipv6")] | ||||
|                 IpAddress::Ipv6(addr) => IpAddr::V6(addr.0.into()), | ||||
|             }) | ||||
|         } else { | ||||
|             Err(Error::Failed) | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     async fn get_host_by_address( | ||||
|         &self, | ||||
|         _addr: embedded_nal_async::IpAddr, | ||||
|     ) -> Result<heapless::String<256>, Self::Error> { | ||||
|         todo!() | ||||
|     } | ||||
| } | ||||
Some files were not shown because too many files have changed in this diff Show More
		Reference in New Issue
	
	Block a user