ucmb: Update AVR sample implementation and update license headers.
authorMichael Büsch <mb@bu3sch.de>
Sat, 21 Feb 2009 19:37:52 +0000 (19:37 +0000)
committerMichael Büsch <mb@bu3sch.de>
Sat, 21 Feb 2009 19:37:52 +0000 (19:37 +0000)
SVN-Revision: 14604

utils/ucmb-tools/tools/ucmb.c
utils/ucmb/driver/ucmb.c
utils/ucmb/microcontroller_examples/atmel_avr8/ucmb.c

index 0bf82e42aa949d8e57a12bed9471989972008206..f432b0ec42da925f4f6a52d7640efed8f45ed8e2 100644 (file)
@@ -1,10 +1,18 @@
 /*
- * Microcontroller Message Bus
- * Userspace commandline utility
+ *   Microcontroller Message Bus
+ *   Userspace commandline utility
  *
- * Copyright (c) 2009 Michael Buesch <mb@bu3sch.de>
+ *   Copyright (c) 2009 Michael Buesch <mb@bu3sch.de>
  *
- * Licensed under the GNU/GPL. See COPYING for details.
+ *   This program is free software; you can redistribute it and/or
+ *   modify it under the terms of the GNU General Public License
+ *   as published by the Free Software Foundation; either version 2
+ *   of the License, or (at your option) any later version.
+ *
+ *   This program is distributed in the hope that it will be useful,
+ *   but WITHOUT ANY WARRANTY; without even the implied warranty of
+ *   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+ *   GNU General Public License for more details.
  */
 
 #include <stdlib.h>
index 37fb889a33e9209eeab8db9b35619a40b1dcbc9f..890a2abbd517889ace5548353b22908a5d4b6c93 100644 (file)
@@ -1,9 +1,18 @@
 /*
- * Microcontroller Message Bus
+ *   Microcontroller Message Bus
+ *   Linux kernel driver
  *
- * Copyright (c) 2009 Michael Buesch <mb@bu3sch.de>
+ *   Copyright (c) 2009 Michael Buesch <mb@bu3sch.de>
  *
- * Licensed under the GNU/GPL. See COPYING for details.
+ *   This program is free software; you can redistribute it and/or
+ *   modify it under the terms of the GNU General Public License
+ *   as published by the Free Software Foundation; either version 2
+ *   of the License, or (at your option) any later version.
+ *
+ *   This program is distributed in the hope that it will be useful,
+ *   but WITHOUT ANY WARRANTY; without even the implied warranty of
+ *   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+ *   GNU General Public License for more details.
  */
 
 #include "ucmb.h"
index 142424c4a6d56db1c34c2e74bbb46fd2bb668968..fb2bc944bf6406bbd9191f7805561b063e389ecb 100644 (file)
@@ -29,7 +29,7 @@
 
 struct ucmb_message_hdr {
        uint16_t magic;         /* UCMB_MAGIC */
-       uint16_t len;           /* Payload length (excluding header) */
+       uint16_t len;           /* Payload length (excluding header and footer) */
 } __attribute__((packed));
 
 struct ucmb_message_footer {
@@ -62,9 +62,10 @@ static uint16_t ucmb_send_message_len;
 /* Statemachine */
 static uint8_t ucmb_state;
 enum {
-       UCMB_ST_LISTEN,
-       UCMB_ST_SENDSTATUS,
-       UCMB_ST_SENDMESSAGE,
+       UCMB_ST_LISTEN,         /* Listen for incoming messages. */
+       UCMB_ST_SENDSTATUS,     /* Send the status report. */
+       UCMB_ST_SENDMESSAGE,    /* Send the message. */
+       UCMB_ST_RETRSTATUS,     /* Retrieve the status report. */
 };
 
 #define TRAILING       1
@@ -100,7 +101,7 @@ static void ucmb_send_next_byte(void)
                if (ucmb_buf_ptr == full_length + TRAILING) {
                        ucmb_send_message_len = 0;
                        ucmb_buf_ptr = 0;
-                       ucmb_state = UCMB_ST_LISTEN;//FIXME retr status
+                       ucmb_state = UCMB_ST_RETRSTATUS;
                }
                break;
        } }
@@ -138,13 +139,16 @@ ISR(SPI_STC_vect)
        uint8_t data;
 
        data = SPDR;
+       SPDR = 0;
 
        switch (ucmb_state) {
        case UCMB_ST_LISTEN: {
                struct ucmb_message_hdr *hdr;
                struct ucmb_message_footer *footer;
 
-               ucmb_buf[ucmb_buf_ptr++] = data;
+               if (ucmb_buf_ptr < sizeof(ucmb_buf))
+                       ucmb_buf[ucmb_buf_ptr] = data;
+               ucmb_buf_ptr++;
                if (ucmb_buf_ptr < sizeof(struct ucmb_message_hdr))
                        return; /* Header RX not complete. */
                hdr = (struct ucmb_message_hdr *)ucmb_buf;
@@ -154,10 +158,8 @@ ISR(SPI_STC_vect)
                                ucmb_buf_ptr = 0;
                                return;
                        }
-                       if (hdr->len > UCMB_MAX_MSG_LEN) {
-                               /* Invalid length. */
-                               //FIXME don't interrupt, but poll len bytes and
-                               // send an immediate failure report
+                       if (hdr->len > 0x8000) {
+                               /* Completely bogus length! Reset. */
                                ucmb_buf_ptr = 0;
                                return;
                        }
@@ -167,17 +169,25 @@ ISR(SPI_STC_vect)
                if (ucmb_buf_ptr == sizeof(struct ucmb_message_hdr) +
                                    sizeof(struct ucmb_message_footer) +
                                    hdr->len) {
-                       footer = (struct ucmb_message_footer *)(
-                                       ucmb_buf + sizeof(struct ucmb_message_hdr) +
-                                       hdr->len);
                        status_buf.magic = UCMB_MAGIC;
                        status_buf.code = UCMB_STAT_OK;
-                       if (ucmb_calc_msg_buffer_crc() != footer->crc)
-                               status_buf.code = UCMB_STAT_ECRC;
+                       if (ucmb_buf_ptr > sizeof(ucmb_buf)) {
+                               /* Message is way too big and was truncated. */
+                               status_buf.code = UCMB_STAT_E2BIG;
+                       } else {
+                               footer = (struct ucmb_message_footer *)(
+                                               ucmb_buf + sizeof(struct ucmb_message_hdr) +
+                                               hdr->len);
+                               if (ucmb_calc_msg_buffer_crc() != footer->crc)
+                                       status_buf.code = UCMB_STAT_ECRC;
+                       }
                        ucmb_state = UCMB_ST_SENDSTATUS;
                        ucmb_buf_ptr = 0;
                        ucmb_send_next_byte();
 
+                       if (status_buf.code != UCMB_STAT_OK)
+                               return; /* Corrupt message. Don't pass it to user code. */
+
                        ucmb_send_message_len = ucmb_rx_message(
                                        ucmb_buf + sizeof(struct ucmb_message_hdr),
                                        hdr->len);
@@ -197,7 +207,17 @@ ISR(SPI_STC_vect)
        case UCMB_ST_SENDMESSAGE:
                ucmb_send_next_byte();
                break;
-       }
+       case UCMB_ST_RETRSTATUS: {
+               uint8_t *st = (uint8_t *)&status_buf;
+
+               st[ucmb_buf_ptr++] = data;
+               if (ucmb_buf_ptr == sizeof(struct ucmb_status)) {
+                       /* We could possibly handle the status report here... */
+                       ucmb_buf_ptr = 0;
+                       ucmb_state = UCMB_ST_LISTEN;
+               }
+               break;
+       } }
 }
 
 void ucmb_init(void)