Browse Source

usb audio isochronous input/output

teensy4-core
PaulStoffregen 8 years ago
parent
commit
425fe1288f
3 changed files with 120 additions and 14 deletions
  1. +60
    -0
      teensy3/usb_audio.cpp
  2. +53
    -12
      teensy3/usb_dev.c
  3. +7
    -2
      teensy3/usb_dev.h

+ 60
- 0
teensy3/usb_audio.cpp View File

/* Teensyduino Core Library
* http://www.pjrc.com/teensy/
* Copyright (c) 2016 PJRC.COM, LLC.
*
* Permission is hereby granted, free of charge, to any person obtaining
* a copy of this software and associated documentation files (the
* "Software"), to deal in the Software without restriction, including
* without limitation the rights to use, copy, modify, merge, publish,
* distribute, sublicense, and/or sell copies of the Software, and to
* permit persons to whom the Software is furnished to do so, subject to
* the following conditions:
*
* 1. The above copyright notice and this permission notice shall be
* included in all copies or substantial portions of the Software.
*
* 2. If the Software is incorporated into a build system that allows
* selection among a list of target devices, then similar target
* devices manufactured by PJRC.COM must be included in the list of
* target devices and selectable in the same manner.
*
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
* EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
* MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
* NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS
* BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN
* ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN
* CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
* SOFTWARE.
*/

#include "usb_dev.h"
#include "HardwareSerial.h"
#include <string.h> // for memcpy()

#ifdef AUDIO_INTERFACE // defined by usb_dev.h -> usb_desc.h
#if F_CPU >= 20000000

uint16_t usb_audio_receive_buffer[AUDIO_RX_SIZE/2] __attribute__ ((section(".dmabuffers"), aligned (4)));
uint16_t usb_audio_transmit_buffer[AUDIO_TX_SIZE/2] __attribute__ ((section(".dmabuffers"), aligned (4)));

// Called from the USB interrupt when an isochronous packet arrives
// we must completely remove it from the receive buffer before returning
//
void usb_audio_receive_callback(unsigned int len)
{
//serial_print(".");
}

// Called from the USB interrupt when ready to transmit another
// isochronous packet. If we place data into the transmit buffer,
// the return is the number of bytes. Otherwise, return 0 means
// no data to transmit
unsigned int usb_audio_transmit_callback(void)
{
return 0;
}


#endif // F_CPU
#endif // AUDIO_INTERFACE

+ 53
- 12
teensy3/usb_dev.c View File

/* Teensyduino Core Library /* Teensyduino Core Library
* http://www.pjrc.com/teensy/ * http://www.pjrc.com/teensy/
* Copyright (c) 2013 PJRC.COM, LLC.
* Copyright (c) 2016 PJRC.COM, LLC.
* *
* Permission is hereby granted, free of charge, to any person obtaining * Permission is hereby granted, free of charge, to any person obtaining
* a copy of this software and associated documentation files (the * a copy of this software and associated documentation files (the
epconf = *cfg++; epconf = *cfg++;
*reg = epconf; *reg = epconf;
reg += 4; reg += 4;
#ifdef AUDIO_INTERFACE
if (i == AUDIO_RX_ENDPOINT) {
table[index(i, RX, EVEN)].addr = usb_audio_receive_buffer;
table[index(i, RX, EVEN)].desc = (AUDIO_RX_SIZE<<16) | BDT_OWN;
table[index(i, RX, ODD)].addr = usb_audio_receive_buffer;
table[index(i, RX, ODD)].desc = (AUDIO_RX_SIZE<<16) | BDT_OWN;
} else
#endif
if (epconf & USB_ENDPT_EPRXEN) { if (epconf & USB_ENDPT_EPRXEN) {
usb_packet_t *p; usb_packet_t *p;
p = usb_malloc(); p = usb_malloc();
//serial_print("rx_mem:"); //serial_print("rx_mem:");
__disable_irq(); __disable_irq();
for (i=1; i <= NUM_ENDPOINTS; i++) { for (i=1; i <= NUM_ENDPOINTS; i++) {
#ifdef AUDIO_INTERFACE
if (i == AUDIO_RX_ENDPOINT) continue;
#endif
if (*cfg++ & USB_ENDPT_EPRXEN) { if (*cfg++ & USB_ENDPT_EPRXEN) {
if (table[index(i, RX, EVEN)].desc == 0) { if (table[index(i, RX, EVEN)].desc == 0) {
table[index(i, RX, EVEN)].addr = packet->buf; table[index(i, RX, EVEN)].addr = packet->buf;
__enable_irq(); __enable_irq();
} }


void usb_tx_isochronous(uint32_t endpoint, void *data, uint32_t len)
{
bdt_t *b = &table[index(endpoint, TX, EVEN)];
uint8_t next, state;


endpoint--;
if (endpoint >= NUM_ENDPOINTS) return;
__disable_irq();
state = tx_state[endpoint];
if (state == 0) {
next = 1;
} else {
b++;
next = 0;
}
tx_state[endpoint] = next;
b->addr = data;
b->desc = (len << 16) | BDT_OWN;
__enable_irq();
}






#endif #endif
endpoint--; // endpoint is index to zero-based arrays endpoint--; // endpoint is index to zero-based arrays


#ifdef AUDIO_INTERFACE
if ((endpoint == AUDIO_TX_ENDPOINT-1) && (stat & 0x08)) {
unsigned int len;
len = usb_audio_transmit_callback();
if (len > 0) {
b = (bdt_t *)((uint32_t)b ^ 8);
b->addr = usb_audio_transmit_buffer;
b->desc = (len << 16) | BDT_OWN;
tx_state[endpoint] ^= 1;
}
} else if ((endpoint == AUDIO_RX_ENDPOINT-1) && !(stat & 0x08)) {
usb_audio_receive_callback(b->desc >> 16);
b->addr = usb_audio_receive_buffer;
b->desc = (AUDIO_RX_SIZE << 16) | BDT_OWN;
} else
#endif
if (stat & 0x08) { // transmit if (stat & 0x08) { // transmit
usb_free(packet); usb_free(packet);
packet = tx_first[endpoint]; packet = tx_first[endpoint];
default: default:
break; break;
} }
#ifdef AUDIO_INTERFACE
// isochronous does not use data toggle bit
if (endpoint == AUDIO_TX_ENDPOINT) {
b->desc = BDT_DESC(packet->len, DATA0);
} else
#endif
b->desc = BDT_DESC(packet->len, b->desc = BDT_DESC(packet->len,
((uint32_t)b & 8) ? DATA1 : DATA0); ((uint32_t)b & 8) ? DATA1 : DATA0);
} else { } else {
} }
rx_last[endpoint] = packet; rx_last[endpoint] = packet;
usb_rx_byte_count_data[endpoint] += packet->len; usb_rx_byte_count_data[endpoint] += packet->len;
// TODO: implement a per-endpoint maximum # of allocated packets
// so a flood of incoming data on 1 endpoint doesn't starve
// the others if the user isn't reading it regularly
// TODO: implement a per-endpoint maximum # of allocated
// packets, so a flood of incoming data on 1 endpoint
// doesn't starve the others if the user isn't reading
// it regularly
packet = usb_malloc(); packet = usb_malloc();
if (packet) { if (packet) {
b->addr = packet->buf; b->addr = packet->buf;
b->desc = BDT_DESC(64, ((uint32_t)b & 8) ? DATA1 : DATA0);
b->desc = BDT_DESC(64,
((uint32_t)b & 8) ? DATA1 : DATA0);
} else { } else {
//serial_print("starving "); //serial_print("starving ");
//serial_phex(endpoint + 1); //serial_phex(endpoint + 1);
//serial_print(((uint32_t)b & 8) ? ",odd\n" : ",even\n");
b->desc = 0; b->desc = 0;
usb_rx_memory_needed++; usb_rx_memory_needed++;
} }

+ 7
- 2
teensy3/usb_dev.h View File

uint32_t usb_tx_byte_count(uint32_t endpoint); uint32_t usb_tx_byte_count(uint32_t endpoint);
uint32_t usb_tx_packet_count(uint32_t endpoint); uint32_t usb_tx_packet_count(uint32_t endpoint);
void usb_tx(uint32_t endpoint, usb_packet_t *packet); void usb_tx(uint32_t endpoint, usb_packet_t *packet);
void usb_tx_isr(uint32_t endpoint, usb_packet_t *packet);
void usb_tx_isochronous(uint32_t endpoint, void *data, uint32_t len);


extern volatile uint8_t usb_configuration; extern volatile uint8_t usb_configuration;


extern void usb_flightsim_flush_callback(void); extern void usb_flightsim_flush_callback(void);
#endif #endif



#ifdef AUDIO_INTERFACE
extern uint16_t usb_audio_receive_buffer[];
extern uint16_t usb_audio_transmit_buffer[];
extern void usb_audio_receive_callback(unsigned int len);
extern unsigned int usb_audio_transmit_callback(void);
#endif







Loading…
Cancel
Save