81 lines
1.8 KiB
C
81 lines
1.8 KiB
C
#include <avr/interrupt.h>
|
|
#include "usb_debug.h"
|
|
|
|
|
|
// the time remaining before we transmit any partially full
|
|
// packet, or send a zero length packet.
|
|
volatile uint8_t debug_flush_timer=0;
|
|
|
|
|
|
// transmit a character. 0 returned on success, -1 on error
|
|
int8_t usb_debug_putchar(uint8_t c)
|
|
{
|
|
static uint8_t previous_timeout=0;
|
|
uint8_t timeout, intr_state;
|
|
|
|
// if we're not online (enumerated and configured), error
|
|
if (!usb_configured()) return -1;
|
|
// interrupts are disabled so these functions can be
|
|
// used from the main program or interrupt context,
|
|
// even both in the same program!
|
|
intr_state = SREG;
|
|
cli();
|
|
UENUM = DEBUG_TX_ENDPOINT;
|
|
// if we gave up due to timeout before, don't wait again
|
|
if (previous_timeout) {
|
|
if (!(UEINTX & (1<<RWAL))) {
|
|
SREG = intr_state;
|
|
return -1;
|
|
}
|
|
previous_timeout = 0;
|
|
}
|
|
// wait for the FIFO to be ready to accept data
|
|
timeout = UDFNUML + 4;
|
|
while (1) {
|
|
// are we ready to transmit?
|
|
if (UEINTX & (1<<RWAL)) break;
|
|
SREG = intr_state;
|
|
// have we waited too long?
|
|
if (UDFNUML == timeout) {
|
|
previous_timeout = 1;
|
|
return -1;
|
|
}
|
|
// has the USB gone offline?
|
|
if (!usb_configured()) return -1;
|
|
// get ready to try checking again
|
|
intr_state = SREG;
|
|
cli();
|
|
UENUM = DEBUG_TX_ENDPOINT;
|
|
}
|
|
// actually write the byte into the FIFO
|
|
UEDATX = c;
|
|
// if this completed a packet, transmit it now!
|
|
if (!(UEINTX & (1<<RWAL))) {
|
|
UEINTX = 0x3A;
|
|
debug_flush_timer = 0;
|
|
} else {
|
|
debug_flush_timer = 2;
|
|
}
|
|
SREG = intr_state;
|
|
return 0;
|
|
}
|
|
|
|
|
|
// immediately transmit any buffered output.
|
|
void usb_debug_flush_output(void)
|
|
{
|
|
uint8_t intr_state;
|
|
|
|
intr_state = SREG;
|
|
cli();
|
|
if (debug_flush_timer) {
|
|
UENUM = DEBUG_TX_ENDPOINT;
|
|
while ((UEINTX & (1<<RWAL))) {
|
|
UEDATX = 0;
|
|
}
|
|
UEINTX = 0x3A;
|
|
debug_flush_timer = 0;
|
|
}
|
|
SREG = intr_state;
|
|
}
|