Switch from GPLv2 to GPLv2+
[fw/altos] / src / kernel / ao_radio_cmac_cmd.c
1 /*
2  * Copyright © 2012 Keith Packard <keithp@keithp.com>
3  *
4  * This program is free software; you can redistribute it and/or modify
5  * it under the terms of the GNU General Public License as published by
6  * the Free Software Foundation; either version 2 of the License, or
7  * (at your option) any later version.
8  *
9  * This program is distributed in the hope that it will be useful, but
10  * WITHOUT ANY WARRANTY; without even the implied warranty of
11  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
12  * General Public License for more details.
13  *
14  * You should have received a copy of the GNU General Public License along
15  * with this program; if not, write to the Free Software Foundation, Inc.,
16  * 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA.
17  */
18
19 #include <ao.h>
20 #include <ao_radio_cmac_cmd.h>
21 #include <ao_radio_cmac.h>
22
23 static __xdata uint8_t cmac_data[AO_CMAC_MAX_LEN];
24
25 static uint8_t
26 getnibble(void)
27 {
28         int8_t  b;
29
30         b = ao_cmd_hexchar(getchar());
31         if (b < 0) {
32                 ao_cmd_status = ao_cmd_lex_error;
33                 return 0;
34         }
35         return (uint8_t) b;
36 }
37
38 static uint8_t
39 getbyte(void)
40 {
41         uint8_t b;
42         b = getnibble() << 4;
43         b |= getnibble();
44         return b;
45 }
46         
47 static void
48 radio_cmac_send_cmd(void) __reentrant
49 {
50         uint8_t i;
51         uint8_t len;
52
53         ao_cmd_decimal();
54         if (ao_cmd_status != ao_cmd_success)
55                 return;
56         len = ao_cmd_lex_i;
57         if (len > AO_CMAC_MAX_LEN) {
58                 ao_cmd_status = ao_cmd_syntax_error;
59                 return;
60         }
61         flush();
62         len = ao_cmd_lex_i;
63         for (i = 0; i < len; i++) {
64                 cmac_data[i] = getbyte();
65                 if (ao_cmd_status != ao_cmd_success)
66                         return;
67         }
68         ao_radio_cmac_send(cmac_data, len);
69 }
70
71 static void
72 radio_cmac_recv_cmd(void) __reentrant
73 {
74         uint8_t         len, i;
75         uint16_t        timeout;
76
77         ao_cmd_decimal();
78         if (ao_cmd_status != ao_cmd_success)
79                 return;
80         len = ao_cmd_lex_i;
81         ao_cmd_decimal();
82         if (ao_cmd_status != ao_cmd_success)
83                 return;
84         timeout = AO_MS_TO_TICKS(ao_cmd_lex_i);
85         i = ao_radio_cmac_recv(cmac_data, len, timeout);
86         if (i == AO_RADIO_CMAC_OK) {
87                 printf ("PACKET ");
88                 for (i = 0; i < len; i++)
89                         printf("%02x", cmac_data[i]);
90                 printf (" %d\n", ao_radio_cmac_rssi);
91         } else
92                 printf ("ERROR %d %d\n", i, ao_radio_cmac_rssi);
93 }
94
95 static __code struct ao_cmds ao_radio_cmac_cmds[] = {
96         { radio_cmac_send_cmd,  "s <length>\0Send AES-CMAC packet. Bytes to send follow on next line" },
97         { radio_cmac_recv_cmd,  "S <length> <timeout>\0Receive AES-CMAC packet. Timeout in ms" },
98         { 0, NULL },
99 };
100
101 void
102 ao_radio_cmac_cmd_init(void)
103 {
104         ao_cmd_register(&ao_radio_cmac_cmds[0]);
105 }