Start using pdata area for less-frequently used data
[fw/altos] / ao_usb.c
index 56036af9489795ddbea098f9ba682afb03cc4612..83fd798f46ac8c612ad814bf366e0881bcfb800f 100644 (file)
--- a/ao_usb.c
+++ b/ao_usb.c
@@ -32,8 +32,10 @@ struct ao_task __xdata ao_usb_task;
 #define AO_USB_IN_SIZE         256
 #define AO_USB_OUT_SIZE                128
 
-static uint16_t        ao_usb_in_bytes;
-static uint16_t        ao_usb_out_bytes;
+static __xdata uint16_t        ao_usb_in_bytes;
+static __xdata uint16_t        ao_usb_out_bytes;
+static __xdata uint8_t ao_usb_iif;
+static __xdata uint8_t ao_usb_oif;
 
 /* This interrupt is shared with port 2, 
  * so when we hook that up, fix this
@@ -41,16 +43,15 @@ static uint16_t     ao_usb_out_bytes;
 void
 ao_usb_isr(void) interrupt 6
 {
-       uint8_t usb_if;
-
-       usb_if = USBIIF;
-       if (usb_if & 1)
+       USBIF = 0;
+       ao_usb_iif |= USBIIF;
+       if (ao_usb_iif & 1)
                ao_wakeup(&ao_usb_task);
-       if (usb_if & (1 << AO_USB_IN_EP))
+       if (ao_usb_iif & (1 << AO_USB_IN_EP))
                ao_wakeup(&ao_usb_in_bytes);
 
-       usb_if = USBOIF;
-       if (usb_if & (1 << AO_USB_OUT_EP))
+       ao_usb_oif |= USBOIF;
+       if (ao_usb_oif & (1 << AO_USB_OUT_EP))
                ao_wakeup(&ao_usb_out_bytes);
 }
 
@@ -59,37 +60,37 @@ ao_usb_isr(void) interrupt 6
 #define AO_USB_EP0_DATA_OUT    2
 
 struct ao_usb_setup {
-       uint8_t         recip : 5;
-       uint8_t         type : 2;
-       uint8_t         dir : 1;
+       uint8_t         dir_type_recip;
        uint8_t         request;
        uint16_t        value;
        uint16_t        index;
        uint16_t        length;
-} ao_usb_setup;
+} __xdata ao_usb_setup;
 
-__data uint8_t ao_usb_ep0_state;
-uint8_t * __data ao_usb_ep0_in_data;
-__data uint8_t ao_usb_ep0_in_len;
+__xdata uint8_t ao_usb_ep0_state;
+uint8_t * __xdata ao_usb_ep0_in_data;
+__xdata uint8_t ao_usb_ep0_in_len;
 __xdata uint8_t        ao_usb_ep0_in_buf[2];
-__data uint8_t ao_usb_ep0_out_len;
+__xdata uint8_t ao_usb_ep0_out_len;
 __xdata uint8_t *__data ao_usb_ep0_out_data;
-__data uint8_t ao_usb_configuration;
+__xdata uint8_t ao_usb_configuration;
 
 /* Send an IN data packet */
 static void
 ao_usb_ep0_flush(void)
 {
-       uint8_t this_len;
-       uint8_t cs0;
+       __xdata uint8_t this_len;
+       __xdata uint8_t cs0;
        
-       if (ao_usb_ep0_state != AO_USB_EP0_DATA_IN)
-               return;
+       USBINDEX = 0;
+       cs0 = USBCS0;
+       if (cs0 & USBCS0_INPKT_RDY)
+               ao_panic(0);
 
-       cs0 = USBCS0_INPKT_RDY;
        this_len = ao_usb_ep0_in_len;
        if (this_len > AO_USB_CONTROL_SIZE)
                this_len = AO_USB_CONTROL_SIZE;
+       cs0 = USBCS0_INPKT_RDY;
        if (this_len != AO_USB_CONTROL_SIZE) {
                cs0 = USBCS0_INPKT_RDY | USBCS0_DATA_END;
                ao_usb_ep0_state = AO_USB_EP0_IDLE;
@@ -119,7 +120,7 @@ struct ao_usb_line_coding {
        uint8_t         data_bits;
 } ;
 
-static struct ao_usb_line_coding ao_usb_line_coding = {115200, 0, 0, 8};
+__xdata static struct ao_usb_line_coding ao_usb_line_coding = {115200, 0, 0, 8};
 
 /* USB descriptors in one giant block of bytes */
 static const uint8_t ao_usb_descriptors [] = 
@@ -240,22 +241,24 @@ static const uint8_t ao_usb_descriptors [] =
        /* iSerial */
        0x0e,
        AO_USB_DESC_STRING,
-       't', 0, 'e', 0, 'l', 0, 'e', 0, '-', 0, '0', 0, 
+       '0', 0, '0', 0, '0', 0, '0', 0, '0', 0, '0', 0, 
 
        /* Terminating zero */
        0
 };
 
+/* Walk through the list of descriptors and find a match
+ */
 static void
-ao_usb_get_descriptor(uint8_t type, uint8_t id)
+ao_usb_get_descriptor(uint16_t value)
 {
-       const uint8_t   *descriptor;
+       const uint8_t           *__xdata descriptor;
+       __xdata uint8_t         type = value >> 8;
+       __xdata uint8_t         index = value;
 
        descriptor = ao_usb_descriptors;
-       ao_usb_ep0_in_len = 0;
-       ao_usb_ep0_in_data = NULL;
        while (descriptor[0] != 0) {
-               if (descriptor[1] == type && id-- == 0) {
+               if (descriptor[1] == type && index-- == 0) {
                        if (type == AO_USB_DESC_CONFIGURATION)
                                ao_usb_ep0_in_len = descriptor[2];
                        else
@@ -265,13 +268,14 @@ ao_usb_get_descriptor(uint8_t type, uint8_t id)
                }
                descriptor += descriptor[0];
        }
-       ao_usb_ep0_state = AO_USB_EP0_DATA_IN;
 }
 
+/* Read data from the ep0 OUT fifo
+ */
 static void
 ao_usb_ep0_fill(void)
 {
-       uint8_t len;
+       __xdata uint8_t len;
        
        USBINDEX = 0;
        len = USBCNT0;
@@ -283,29 +287,9 @@ ao_usb_ep0_fill(void)
 }
 
 void
-ao_usb_send_two_bytes(uint8_t a, uint8_t b)
-{
-       ao_usb_ep0_in_buf[0] = a;
-       ao_usb_ep0_in_buf[1] = b;
-       ao_usb_ep0_in_data = ao_usb_ep0_in_buf;
-       ao_usb_ep0_in_len = 2;
-       ao_usb_ep0_state = AO_USB_EP0_DATA_IN;
-}
-
-void
-ao_usb_send_one_byte(uint8_t a)
-{
-       ao_usb_ep0_in_buf[0] = a;
-       ao_usb_ep0_in_data = ao_usb_ep0_in_buf;
-       ao_usb_ep0_in_len = 1;
-       ao_usb_ep0_state = AO_USB_EP0_DATA_IN;
-}
-
-void
-ao_usb_send_zero_bytes(void)
+ao_usb_ep0_queue_byte(uint8_t a)
 {
-       ao_usb_ep0_in_len = 0;
-       ao_usb_ep0_state = AO_USB_EP0_DATA_IN;
+       ao_usb_ep0_in_buf[ao_usb_ep0_in_len++] = a;
 }
 
 void
@@ -316,54 +300,92 @@ ao_usb_set_address(uint8_t address)
                ;
 }
 
+static void
+ao_usb_set_configuration(void)
+{
+       /* Set the IN max packet size, double buffered */
+       USBINDEX = AO_USB_IN_EP;
+       USBMAXI = AO_USB_IN_SIZE >> 3;
+       USBCSIH |= USBCSIH_IN_DBL_BUF;
+
+       /* Set the OUT max packet size, double buffered */
+       USBINDEX = AO_USB_OUT_EP;
+       USBMAXO = AO_USB_OUT_SIZE >> 3;
+       USBCSOH = USBCSOH_OUT_DBL_BUF;
+}
+
 static void
 ao_usb_ep0_setup(void)
 {
+       /* Pull the setup packet out of the fifo */
        ao_usb_ep0_out_data = (__xdata uint8_t *) &ao_usb_setup;
        ao_usb_ep0_out_len = 8;
        ao_usb_ep0_fill();
-       if (ao_usb_ep0_out_len > 0)
+       if (ao_usb_ep0_out_len != 0)
                return;
-       switch(ao_usb_setup.type) {
+
+       /* Figure out how to ACK the setup packet */
+       if (ao_usb_setup.dir_type_recip & AO_USB_DIR_IN) {
+               if (ao_usb_setup.length)
+                       ao_usb_ep0_state = AO_USB_EP0_DATA_IN;
+               else
+                       ao_usb_ep0_state = AO_USB_EP0_IDLE;
+       } else {
+               if (ao_usb_setup.length)
+                       ao_usb_ep0_state = AO_USB_EP0_DATA_OUT;
+               else
+                       ao_usb_ep0_state = AO_USB_EP0_IDLE;
+       }
+       USBINDEX = 0;
+       if (ao_usb_ep0_state == AO_USB_EP0_IDLE)
+               USBCS0 = USBCS0_CLR_OUTPKT_RDY | USBCS0_DATA_END;
+       else
+               USBCS0 = USBCS0_CLR_OUTPKT_RDY;
+       
+       ao_usb_ep0_in_data = ao_usb_ep0_in_buf;
+       ao_usb_ep0_in_len = 0;
+       switch(ao_usb_setup.dir_type_recip & AO_USB_SETUP_TYPE_MASK) {
        case AO_USB_TYPE_STANDARD:
-               switch(ao_usb_setup.recip) {
+               switch(ao_usb_setup.dir_type_recip & AO_USB_SETUP_RECIP_MASK) {
                case AO_USB_RECIP_DEVICE:
                        switch(ao_usb_setup.request) {
                        case AO_USB_REQ_GET_STATUS:
-                               ao_usb_send_two_bytes(0,0);
+                               ao_usb_ep0_queue_byte(0);
+                               ao_usb_ep0_queue_byte(0);
                                break;
                        case AO_USB_REQ_SET_ADDRESS:
                                ao_usb_set_address(ao_usb_setup.value);
                                break;
                        case AO_USB_REQ_GET_DESCRIPTOR:
-                               ao_usb_get_descriptor(ao_usb_setup.value,
-                                                     ao_usb_setup.index);
+                               ao_usb_get_descriptor(ao_usb_setup.value);
                                break;
                        case AO_USB_REQ_GET_CONFIGURATION:
-                               ao_usb_send_one_byte(ao_usb_configuration);
+                               ao_usb_ep0_queue_byte(ao_usb_configuration);
                                break;
                        case AO_USB_REQ_SET_CONFIGURATION:
                                ao_usb_configuration = ao_usb_setup.value;
+                               ao_usb_set_configuration();
                                break;
                        }
                        break;
                case AO_USB_RECIP_INTERFACE:
                        switch(ao_usb_setup.request) {
                        case AO_USB_REQ_GET_STATUS:
-                               ao_usb_send_two_bytes(0,0);
+                               ao_usb_ep0_queue_byte(0);
+                               ao_usb_ep0_queue_byte(0);
                                break;
                        case AO_USB_REQ_GET_INTERFACE:
-                               ao_usb_send_one_byte(0);
+                               ao_usb_ep0_queue_byte(0);
                                break;
                        case AO_USB_REQ_SET_INTERFACE:
-                               ao_usb_send_zero_bytes();
                                break;
                        }
                        break;
                case AO_USB_RECIP_ENDPOINT:
                        switch(ao_usb_setup.request) {
                        case AO_USB_REQ_GET_STATUS:
-                               ao_usb_send_two_bytes(0, 0);
+                               ao_usb_ep0_queue_byte(0);
+                               ao_usb_ep0_queue_byte(0);
                                break;
                        }
                        break;
@@ -374,30 +396,38 @@ ao_usb_ep0_setup(void)
                case SET_LINE_CODING:
                        ao_usb_ep0_out_len = 7;
                        ao_usb_ep0_out_data = (__xdata uint8_t *) &ao_usb_line_coding;
-                       ao_usb_ep0_state = AO_USB_EP0_DATA_OUT;
                        break;
                case GET_LINE_CODING:
                        ao_usb_ep0_in_len = 7;
                        ao_usb_ep0_in_data = (uint8_t *) &ao_usb_line_coding;
-                       ao_usb_ep0_state = AO_USB_EP0_DATA_IN;
                        break;
                case SET_CONTROL_LINE_STATE:
                        break;
                }
                break;
        }
-       ao_usb_ep0_flush();
+       if (ao_usb_ep0_state != AO_USB_EP0_DATA_OUT) {
+               if (ao_usb_setup.length < ao_usb_ep0_in_len)
+                       ao_usb_ep0_in_len = ao_usb_setup.length;
+               ao_usb_ep0_flush();
+       }
 }
 
 /* End point 0 receives all of the control messages. */
 static void
 ao_usb_ep0(void)
 {
-       uint8_t cs0;
+       __xdata uint8_t cs0;
 
        ao_usb_ep0_state = AO_USB_EP0_IDLE;
        for (;;) {
-               ao_sleep(&ao_usb_task);
+               __critical for (;;) {
+                       if (ao_usb_iif & 1) {
+                               ao_usb_iif &= ~1;
+                               break;
+                       }
+                       ao_sleep(&ao_usb_task);
+               }
                USBINDEX = 0;
                cs0 = USBCS0;
                if (cs0 & USBCS0_SETUP_END) {
@@ -408,7 +438,9 @@ ao_usb_ep0(void)
                        ao_usb_ep0_state = AO_USB_EP0_IDLE;
                        USBCS0 &= ~USBCS0_SENT_STALL;
                }
-               if (cs0 & USBCS0_INPKT_RDY) {
+               if (ao_usb_ep0_state == AO_USB_EP0_DATA_IN &&
+                   (cs0 & USBCS0_INPKT_RDY) == 0)
+               {
                        ao_usb_ep0_flush();
                }
                if (cs0 & USBCS0_OUTPKT_RDY) {
@@ -420,33 +452,30 @@ ao_usb_ep0(void)
                                ao_usb_ep0_fill();
                                if (ao_usb_ep0_out_len == 0)
                                        ao_usb_ep0_state = AO_USB_EP0_IDLE;
+                               USBINDEX = 0;
+                               if (ao_usb_ep0_state == AO_USB_EP0_IDLE)
+                                       USBCS0 = USBCS0_CLR_OUTPKT_RDY | USBCS0_DATA_END;
+                               else
+                                       USBCS0 = USBCS0_CLR_OUTPKT_RDY;
                                break;
                        }
-                       USBINDEX = 0;
-                       if (ao_usb_ep0_state == AO_USB_EP0_IDLE)
-                               USBCS0 = USBCS0_CLR_OUTPKT_RDY | USBCS0_DATA_END;
-                       else
-                               USBCS0 = USBCS0_CLR_OUTPKT_RDY;
                }
        }
 }
 
 void
-ao_usb_flush(void)
+ao_usb_flush(void) __critical
 {
-       ao_interrupt_disable();
        if (ao_usb_in_bytes) {
                USBINDEX = AO_USB_IN_EP;
                USBCSIL |= USBCSIL_INPKT_RDY;
                ao_usb_in_bytes = 0;
        }
-       ao_interrupt_enable();
 }
 
 void
-ao_usb_putchar(uint8_t c)
+ao_usb_putchar(uint8_t c) __critical
 {
-       ao_interrupt_disable();
        for (;;) {
                USBINDEX = AO_USB_IN_EP;
                if ((USBCSIL & USBCSIL_INPKT_RDY) == 0)
@@ -454,16 +483,17 @@ ao_usb_putchar(uint8_t c)
                ao_sleep(&ao_usb_in_bytes);
        }
        USBFIFO[AO_USB_IN_EP << 1] = c;
-       if (++ao_usb_in_bytes == AO_USB_IN_SIZE)
-               ao_usb_flush();
-       ao_interrupt_enable();
+       if (++ao_usb_in_bytes == AO_USB_IN_SIZE) {
+               USBINDEX = AO_USB_IN_EP;
+               USBCSIL |= USBCSIL_INPKT_RDY;
+               ao_usb_in_bytes = 0;
+       }
 }
 
 uint8_t
-ao_usb_getchar(void)
+ao_usb_getchar(void) __critical
 {
-       uint8_t c;
-       ao_interrupt_disable();
+       __xdata uint8_t c;
        while (ao_usb_out_bytes == 0) {
                for (;;) {
                        USBINDEX = AO_USB_OUT_EP;
@@ -475,7 +505,10 @@ ao_usb_getchar(void)
        }
        --ao_usb_out_bytes;
        c = USBFIFO[AO_USB_OUT_EP << 1];
-       ao_interrupt_enable();
+       if (ao_usb_out_bytes == 0) {
+               USBINDEX = AO_USB_OUT_EP;
+               USBCSOL &= ~USBCSOL_OUTPKT_RDY;
+       }
        return c;
 }
 
@@ -485,15 +518,7 @@ ao_usb_init(void)
        /* Turn on the USB controller */
        SLEEP |= SLEEP_USB_EN;
 
-       /* Set the IN max packet size, double buffered */
-       USBINDEX = AO_USB_IN_EP;
-       USBMAXI = AO_USB_IN_SIZE >> 3;
-       USBCSIH |= USBCSIH_IN_DBL_BUF;
-
-       /* Set the OUT max packet size, double buffered */
-       USBINDEX = AO_USB_OUT_EP;
-       USBMAXO = AO_USB_OUT_SIZE >> 3;
-       USBCSOH = USBCSOH_OUT_DBL_BUF;
+       ao_usb_set_configuration();
        
        /* IN interrupts on the control an IN endpoints */
        USBIIE = (1 << AO_USB_CONTROL_EP) | (1 << AO_USB_IN_EP);
@@ -504,10 +529,13 @@ ao_usb_init(void)
        /* Ignore control interrupts */
        USBCIE = 0;
        
+       /* enable USB interrupts */
+       IEN2 |= IEN2_USBIE;
+
        /* Clear any pending interrupts */
        USBCIF = 0;
        USBOIF = 0;
        USBIIF = 0;
        
-       ao_add_task(&ao_usb_task, ao_usb_ep0);
+       ao_add_task(&ao_usb_task, ao_usb_ep0, "usb");
 }