Keep the newer GCC happy.
2 * Test various AVR bits and pieces
5 * Daniel O'Connor <darius@dons.net.au>. All rights reserved.
7 * Redistribution and use in source and binary forms, with or without
8 * modification, are permitted provided that the following conditions
10 * 1. Redistributions of source code must retain the above copyright
11 * notice, this list of conditions and the following disclaimer.
12 * 2. Redistributions in binary form must reproduce the above copyright
13 * notice, this list of conditions and the following disclaimer in the
14 * documentation and/or other materials provided with the distribution.
16 * THIS SOFTWARE IS PROVIDED BY AUTHOR AND CONTRIBUTORS ``AS IS'' AND
17 * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
18 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
19 * ARE DISCLAIMED. IN NO EVENT SHALL AUTHOR OR CONTRIBUTORS BE LIABLE
20 * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
21 * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
22 * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
23 * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
24 * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
25 * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
30 #include <avr/interrupt.h>
31 #include <avr/pgmspace.h>
36 #include <util/delay.h>
48 * Mirror of the MCUCSR register, taken early during startup.
50 uint8_t mcucsr __attribute__((section(".noinit")));
52 void process_cmd(void);
55 * Read out and reset MCUCSR early during startup.
57 void handle_mcucsr(void)
58 __attribute__((section(".init3")))
59 __attribute__((naked));
60 void handle_mcucsr(void) {
68 /* Disable interrupts while we frob stuff */
72 /* Disable JTAG (yes twice) */
78 /* USB data bus (7:0) */
82 /* USB control (4:0) */
97 /* USART (0:1), IDBus (2:5), 485 (6:6), GPIO (7:7) */
104 * Fpwm = Fclk / (N * 510)
108 * Behind ifndef because PB3 is A0 on PDI chip
110 * Phase correct PWM, non-inverted output, divide by 8 */
111 TCCR0 = _BV(WGM00) | _BV(WGM11) | _BV(COM00) | _BV(COM01) | _BV(CS01);
115 /* Set up the one wire stuff */
124 /* Init temperature control stuff */
127 printf_P(PSTR("\r\n\r\n===============\r\n"
130 if ((mcucsr & _BV(PORF)) == _BV(PORF))
131 printf_P(PSTR("Power on reset\r\n"));
133 if ((mcucsr & _BV(EXTRF)) == _BV(EXTRF))
134 printf_P(PSTR("External reset\r\n"));
136 if ((mcucsr & _BV(BORF)) == _BV(BORF))
137 printf_P(PSTR("Brown-out reset\r\n"));
139 if ((mcucsr & _BV(WDRF)) == _BV(WDRF))
140 printf_P(PSTR("Watchdog reset\r\n"));
142 if ((mcucsr & _BV(JTRF)) == _BV(JTRF))
143 printf_P(PSTR("JTAG reset\r\n"));
149 * Enable the watchdog with the largest prescaler. Will cause a
150 * watchdog reset after approximately 2 s @ Vcc = 5 V
152 * Gets reset in the loop below and in the tempctrl.c timer IRQ
157 printf_P(PSTR("Calling usb_init\r\n"));
159 printf_P(PSTR("done\r\n"));
162 printf_P(PSTR("> "));
165 /* Wait for user input or an "interrupt" */
171 if (cmd.state == 255) {
173 printf_P(PSTR("> "));
174 /* Allow new characters to be processed */
179 if (!(PDICTL & _BV(PDIINT)))
187 uint8_t ROM[8], crc, buf[9], temp;
191 /* User just pressed enter */
195 if (!strcasecmp_P((char *)cmd.buf, PSTR("?")) ||
196 !strcasecmp_P((char *)cmd.buf, PSTR("help"))) {
197 printf_P(PSTR("rs Reset and check for presence\r\n"
198 "sr Search the bus for ROMs\r\n"
201 "wr bit Write a bit\r\n"
202 "wb byte Write a byte (hex)\r\n"
203 "wc cmd [ROMID] Write command\r\n"
204 "te ROMID Read the temperature from a DS1820\r\n"
205 "in port Read from a port\r\n"
206 "ou port val Write to a port (val in hex)\r\n"
207 "dd port [val] Read/write DDR for a port (val in hex)\r\n"
208 "rt ROMID Read DS2502 status page\r\n"
209 "we ROMID adr val Write data into a DS2502 PROM (adr & val in hex)\r\n"
210 "rr ROMID Read DS2502 PROM\r\n"
212 "gc Get time of day\r\n"
213 "sc time Set time of day (time is YYYY/MM/DD HH:MM:SS)\r\n"
215 "us Generate USB data\r\n"
217 "tc ... Temperature control related (tc help for more)\r\n"));
220 } else if (!strncasecmp_P((char *)cmd.buf, PSTR("zz"), 2)) {
222 wdt_enable(WDTO_15MS);
225 } else if (!strncasecmp_P((char *)cmd.buf, PSTR("rs"), 2)) {
226 printf_P(PSTR("Resetting... "));
228 if (OWTouchReset() == 1)
229 printf_P(PSTR("No presence pulse found\r\n"));
231 printf_P(PSTR("Presence pulse found\r\n"));
232 } else if (!strncasecmp_P((char *)cmd.buf, PSTR("re"), 2)) {
234 printf_P(PSTR("Read a 1\r\n"));
236 printf_P(PSTR("Read a 0\r\n"));
237 } else if (!strncasecmp_P((char *)cmd.buf, PSTR("rb"), 2)) {
238 printf_P(PSTR("Read a 0x%02x\r\n"), OWReadByte());
239 } else if (!strncasecmp_P((char *)cmd.buf, PSTR("wr"), 2)) {
240 arg = strtol((char *)cmd.buf + 3, (char **)NULL, 10);
242 printf_P(PSTR("Wrote a %c\r\n"), arg ? '1' : '0');
243 } else if (!strncasecmp_P((char *)cmd.buf, PSTR("wb"), 2)) {
244 arg = (int)strtol((char *)cmd.buf + 3, (char **)NULL, 16);
246 } else if (!strncasecmp_P((char *)cmd.buf, PSTR("rt"), 2)) {
247 if (sscanf_P((char *)cmd.buf, PSTR("rt %hhx:%hhx:%hhx:%hhx:%hhx:%hhx:%hhx:%hhx"),
248 &ROM[0], &ROM[1], &ROM[2], &ROM[3],
249 &ROM[4], &ROM[5], &ROM[6], &ROM[7]) != 8) {
250 printf_P(PSTR("Unable to parse ROM ID\r\n"));
254 if (ROM[0] != OW_FAMILY_ROM) {
255 printf_P(PSTR("ROM specified isn't a DS2502\r\n"));
259 if (OWTouchReset() != 0) {
260 printf_P(PSTR("No presence\r\n"));
266 OWCRC(OW_READ_STATUS, &crc);
267 OWSendCmd(ROM, OW_READ_STATUS);
275 if (crc != OWReadByte()) {
276 printf_P(PSTR("CRC mismatch on command & address\r\n"));
281 for (i = 0; i < 8; i++) {
283 printf_P(PSTR("%02x "), temp);
286 printf_P(PSTR("\r\n"));
287 if (crc != OWReadByte()) {
288 printf_P(PSTR("CRC mismatch on data\r\n"));
291 } else if (!strncasecmp_P((char *)cmd.buf, PSTR("we"), 2)) {
294 if (sscanf_P((char *)cmd.buf, PSTR("we %hhx:%hhx:%hhx:%hhx:%hhx:%hhx:%hhx:%hhx %hhx %hhx"),
295 &ROM[0], &ROM[1], &ROM[2], &ROM[3],
296 &ROM[4], &ROM[5], &ROM[6], &ROM[7],
297 &adr, &data) != 10) {
298 printf_P(PSTR("Unable to parse ROM ID\r\n"));
302 if (ROM[0] != OW_FAMILY_ROM) {
303 printf_P(PSTR("ID specified isn't a ROM\r\n"));
307 if (OWTouchReset() != 0) {
308 printf_P(PSTR("No presence\r\n"));
312 printf_P(PSTR("OWProgROM returned %S\r\n"), OWProgROM_Status[OWProgROM(ROM, buf[0], 2, &buf[1], 0, 0)]);
313 } else if (!strncasecmp_P((char *)cmd.buf, PSTR("rr"), 2)) {
314 if (sscanf_P((char *)cmd.buf, PSTR("rr %hhx:%hhx:%hhx:%hhx:%hhx:%hhx:%hhx:%hhx"),
315 &ROM[0], &ROM[1], &ROM[2], &ROM[3],
316 &ROM[4], &ROM[5], &ROM[6], &ROM[7]) != 8) {
317 printf_P(PSTR("Unable to parse ROM ID\r\n"));
321 if (ROM[0] != OW_FAMILY_ROM) {
322 printf_P(PSTR("ROM specified isn't a ROM\r\n"));
327 OWSendCmd(ROM, OW_READ_MEMORY);
328 OWCRC(OW_READ_MEMORY, &crc);
336 if (crc != OWReadByte()) {
337 printf_P(PSTR("CRC mismatch on command & address\r\n"));
342 for (buf[0] = 0; buf[0] < 128; buf[0]++) {
343 buf[1] = OWReadByte();
345 if (buf[0] % 16 != 0)
348 printf_P(PSTR("\r\n"));
351 printf_P(PSTR("%02x"), buf[1]);
354 printf_P(PSTR("\r\n"));
355 if (crc != OWReadByte()) {
356 printf_P(PSTR("CRC mismatch on data\r\n"));
360 } else if (!strncasecmp_P((char *)cmd.buf, PSTR("wc"), 2)) {
363 i = sscanf_P((char *)cmd.buf, PSTR("wc %hhx %hhx:%hhx:%hhx:%hhx:%hhx:%hhx:%hhx:%hhx"),
364 &ROM[0], &ROM[1], &ROM[2], &ROM[3],
365 &ROM[4], &ROM[5], &ROM[6], &ROM[7],
368 if (i != 1 && i != 9) {
369 printf_P(PSTR("Incorrect usage\r\n"));
374 OWSendCmd(i == 1 ? NULL : ROM, c);
377 } else if (!strncasecmp_P((char *)cmd.buf, PSTR("te"), 2)) {
378 if (sscanf_P((char *)cmd.buf, PSTR("te %hhx:%hhx:%hhx:%hhx:%hhx:%hhx:%hhx:%hhx"),
379 &ROM[0], &ROM[1], &ROM[2], &ROM[3],
380 &ROM[4], &ROM[5], &ROM[6], &ROM[7]) != 8) {
381 printf_P(PSTR("Unable to parse ROM ID\r\n"));
387 case OW_TEMP_WRONG_FAM:
388 printf_P(PSTR("ROM specified isn't a temperature sensor\r\n"));
391 case OW_TEMP_CRC_ERR:
392 printf_P(PSTR("CRC mismatch\r\n"));
396 printf_P(PSTR("No ROM found\r\n"));
400 printf_P(PSTR("%d.%02d\r\n"), GETWHOLE(t), GETFRAC(t));
403 } else if (!strncasecmp_P((char *)cmd.buf, PSTR("sr"), 2)) {
406 i = OWFirst(ROM, 1, 0);
410 printf_P(PSTR("Presence pulse, but no module found, bad module/cabling?\r\n"));
414 printf_P(PSTR("No presence pulse found\r\n"));
418 printf_P(PSTR("Bad CRC\r\n"));
426 printf_P(PSTR("Unknown error from 1 wire library\r\n"));
433 for (i = 0; i < 8; i++)
434 printf_P(PSTR("%02x%S"), ROM[i], i == 7 ? PSTR("\r\n") : PSTR(":"));
436 i = OWNext(ROM, 1, 0);
438 } else if (!strncasecmp_P((char *)cmd.buf, PSTR("in"), 2)) {
441 switch (tolower(cmd.buf[3])) {
459 printf_P(PSTR("Unknown port\r\n"));
462 printf_P(PSTR("0x%02x\r\n"), inp);
463 } else if (!strncasecmp_P((char *)cmd.buf, PSTR("ou"), 2)) {
467 if (sscanf_P((char *)cmd.buf, PSTR("ou %c %x"), &port, &val) != 2) {
468 printf_P(PSTR("Unable to parse ou arguments\r\n"));
490 printf_P(PSTR("Unknown port\r\n"));
493 printf_P(PSTR("PORT%c <= 0x%02x\r\n"), toupper(port), val);
494 } else if (!strncasecmp_P((char *)cmd.buf, PSTR("dd"), 2)) {
499 num = sscanf_P((char *)cmd.buf, PSTR("dd %c %x"), &port, &val);
501 if (num != 2 && num != 3) {
502 printf_P(PSTR("Unable to parse dd arguments\r\n"));
525 printf_P(PSTR("Unknown port\r\n"));
528 printf_P(PSTR("DDR%c => 0x%02x\r\n"), toupper(port), val);
548 printf_P(PSTR("Unknown port\r\n"));
551 printf_P(PSTR("DDR%c <= 0x%02x\r\n"), toupper(port), val);
553 } else if (!strncasecmp_P((char *)cmd.buf, PSTR("tc"), 2)) {
554 tempctrl_cmd((char *)cmd.buf);
556 } else if (!strncasecmp_P((char *)cmd.buf, PSTR("us"), 2)) {
559 } else if (!strncasecmp_P((char *)cmd.buf, PSTR("gc"), 2)) {
560 ds1307_printtime(PSTR(""), PSTR("\r\n"));
561 } else if (!strncasecmp_P((char *)cmd.buf, PSTR("sc"), 2)) {
563 printf_P(PSTR("Command not long enough\r\n"));
565 if (ds1307_settod((char *)cmd.buf + 3) != 1)
566 printf_P(PSTR("Unable to set time of day\r\n"));
569 printf_P(PSTR("Unknown command, help for a list\r\n"));