avr: Use generic usb_cdc code for usb serial support
Use the generic usb_cdc driver code instead of the "pjrc" usb driver code. Signed-off-by: Kevin O'Connor <kevin@koconnor.net>
This commit is contained in:
parent
c0311bee33
commit
d05aa81927
|
@ -66,7 +66,7 @@ max_temp: 130
|
||||||
pin: PC6
|
pin: PC6
|
||||||
|
|
||||||
[mcu]
|
[mcu]
|
||||||
serial: /dev/ttyACM0
|
serial: /dev/serial/by-id/usb-Klipper_Klipper_firmware_12345-if00
|
||||||
|
|
||||||
[printer]
|
[printer]
|
||||||
kinematics: cartesian
|
kinematics: cartesian
|
||||||
|
|
|
@ -3,14 +3,14 @@
|
||||||
# Use the avr toolchain
|
# Use the avr toolchain
|
||||||
CROSS_PREFIX=avr-
|
CROSS_PREFIX=avr-
|
||||||
|
|
||||||
dirs-y += src/avr src/generic lib/pjrc_usb_serial
|
dirs-y += src/avr src/generic
|
||||||
|
|
||||||
CFLAGS += -mmcu=$(CONFIG_MCU)
|
CFLAGS += -mmcu=$(CONFIG_MCU)
|
||||||
|
|
||||||
# Add avr source files
|
# Add avr source files
|
||||||
src-y += avr/main.c avr/timer.c avr/gpio.c
|
src-y += avr/main.c avr/timer.c avr/gpio.c
|
||||||
src-$(CONFIG_AVR_WATCHDOG) += avr/watchdog.c
|
src-$(CONFIG_AVR_WATCHDOG) += avr/watchdog.c
|
||||||
src-$(CONFIG_AVR_USBSERIAL) += avr/usbserial.c ../lib/pjrc_usb_serial/usb_serial.c
|
src-$(CONFIG_AVR_USBSERIAL) += avr/usbserial.c generic/usb_cdc.c
|
||||||
src-$(CONFIG_AVR_SERIAL) += avr/serial.c generic/serial_irq.c
|
src-$(CONFIG_AVR_SERIAL) += avr/serial.c generic/serial_irq.c
|
||||||
|
|
||||||
# Suppress broken "misspelled signal handler" warnings on gcc 4.8.1
|
# Suppress broken "misspelled signal handler" warnings on gcc 4.8.1
|
||||||
|
|
|
@ -1,71 +1,250 @@
|
||||||
// Wrappers for AVR usb serial.
|
// Hardware interface to USB on AVR at90usb
|
||||||
//
|
//
|
||||||
// Copyright (C) 2016,2017 Kevin O'Connor <kevin@koconnor.net>
|
// Copyright (C) 2018 Kevin O'Connor <kevin@koconnor.net>
|
||||||
//
|
//
|
||||||
// This file may be distributed under the terms of the GNU GPLv3 license.
|
// This file may be distributed under the terms of the GNU GPLv3 license.
|
||||||
|
|
||||||
#include <string.h> // memmove
|
#include <avr/interrupt.h> // USB_COM_vect
|
||||||
#include "../lib/pjrc_usb_serial/usb_serial.h"
|
#include <string.h> // NULL
|
||||||
#include "board/misc.h" // console_sendf
|
#include "autoconf.h" // CONFIG_MACH_at90usb1286
|
||||||
#include "command.h" // command_find_and_dispatch
|
#include "board/usb_cdc.h" // usb_notify_ep0
|
||||||
|
#include "board/usb_cdc_ep.h" // USB_CDC_EP_BULK_IN
|
||||||
|
#include "pgm.h" // READP
|
||||||
#include "sched.h" // DECL_INIT
|
#include "sched.h" // DECL_INIT
|
||||||
|
|
||||||
static uint8_t receive_buf[MESSAGE_MAX], receive_pos;
|
// EPCFG0X definitions
|
||||||
|
#define EP_TYPE_CONTROL 0x00
|
||||||
|
#define EP_TYPE_BULK_IN 0x81
|
||||||
|
#define EP_TYPE_BULK_OUT 0x80
|
||||||
|
#define EP_TYPE_INTERRUPT_IN 0xC1
|
||||||
|
|
||||||
|
// EPCFG1X definitions
|
||||||
|
#define EP_SINGLE_BUFFER 0x02
|
||||||
|
#define EP_DOUBLE_BUFFER 0x06
|
||||||
|
#define EP_SIZE(s) ((s)==64 ? 0x30 : ((s)==32 ? 0x20 : ((s)==16 ? 0x10 : 0x00)))
|
||||||
|
|
||||||
|
static void
|
||||||
|
usb_write_packet(const uint8_t *data, uint8_t len)
|
||||||
|
{
|
||||||
|
while (len--)
|
||||||
|
UEDATX = *data++;
|
||||||
|
}
|
||||||
|
|
||||||
|
static void
|
||||||
|
usb_write_packet_progmem(const uint8_t *data, uint8_t len)
|
||||||
|
{
|
||||||
|
while (len--)
|
||||||
|
UEDATX = READP(*data++);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void
|
||||||
|
usb_read_packet(uint8_t *data, uint8_t len)
|
||||||
|
{
|
||||||
|
while (len--)
|
||||||
|
*data++ = UEDATX;
|
||||||
|
}
|
||||||
|
|
||||||
|
int_fast8_t
|
||||||
|
usb_read_bulk_out(void *data, uint_fast8_t max_len)
|
||||||
|
{
|
||||||
|
UENUM = USB_CDC_EP_BULK_OUT;
|
||||||
|
if (!(UEINTX & (1<<RWAL)))
|
||||||
|
// No data ready
|
||||||
|
return -1;
|
||||||
|
uint8_t len = UEBCLX;
|
||||||
|
usb_read_packet(data, len);
|
||||||
|
UEINTX = (uint8_t)~(1<<FIFOCON);
|
||||||
|
return len;
|
||||||
|
}
|
||||||
|
|
||||||
|
int_fast8_t
|
||||||
|
usb_send_bulk_in(void *data, uint_fast8_t len)
|
||||||
|
{
|
||||||
|
UENUM = USB_CDC_EP_BULK_IN;
|
||||||
|
if (!(UEINTX & (1<<RWAL)))
|
||||||
|
// Buffer full
|
||||||
|
return -1;
|
||||||
|
usb_write_packet(data, len);
|
||||||
|
UEINTX = (uint8_t)~((1<<FIFOCON) | (1<<RXOUTI));
|
||||||
|
return len;
|
||||||
|
}
|
||||||
|
|
||||||
|
int_fast8_t
|
||||||
|
usb_read_ep0(void *data, uint_fast8_t max_len)
|
||||||
|
{
|
||||||
|
UENUM = 0;
|
||||||
|
uint8_t ueintx = UEINTX;
|
||||||
|
if (ueintx & (1<<RXSTPI))
|
||||||
|
return -2;
|
||||||
|
if (!(ueintx & (1<<RXOUTI))) {
|
||||||
|
// Not ready to receive data
|
||||||
|
UEIENX = (1<<RXSTPE) | (1<<RXOUTE);
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
usb_read_packet(data, max_len);
|
||||||
|
if (UEINTX & (1<<RXSTPI))
|
||||||
|
return -2;
|
||||||
|
UEINTX = ~(1<<RXOUTI);
|
||||||
|
return max_len;
|
||||||
|
}
|
||||||
|
|
||||||
|
int_fast8_t
|
||||||
|
usb_read_ep0_setup(void *data, uint_fast8_t max_len)
|
||||||
|
{
|
||||||
|
UENUM = 0;
|
||||||
|
uint8_t ueintx = UEINTX;
|
||||||
|
if (!(ueintx & ((1<<RXSTPI)))) {
|
||||||
|
// No data ready to read
|
||||||
|
UEIENX = 1<<RXSTPE;
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
usb_read_packet(data, max_len);
|
||||||
|
UEINTX = ~((1<<RXSTPI) | (1<<RXOUTI));
|
||||||
|
return max_len;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int8_t
|
||||||
|
_usb_send_ep0(const void *data, uint8_t len, uint8_t progmem)
|
||||||
|
{
|
||||||
|
UENUM = 0;
|
||||||
|
uint8_t ueintx = UEINTX;
|
||||||
|
if (ueintx & ((1<<RXSTPI) | (1<<RXOUTI)))
|
||||||
|
return -2;
|
||||||
|
if (!(ueintx & (1<<TXINI))) {
|
||||||
|
// Not ready to send
|
||||||
|
UEIENX = (1<<RXSTPE) | (1<<RXOUTE) | (1<<TXINE);
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
if (progmem)
|
||||||
|
usb_write_packet_progmem(data, len);
|
||||||
|
else
|
||||||
|
usb_write_packet(data, len);
|
||||||
|
UEINTX = ~(1<<TXINI);
|
||||||
|
return len;
|
||||||
|
}
|
||||||
|
|
||||||
|
int_fast8_t
|
||||||
|
usb_send_ep0(const void *data, uint_fast8_t len)
|
||||||
|
{
|
||||||
|
return _usb_send_ep0(data, len, 0);
|
||||||
|
}
|
||||||
|
|
||||||
|
int_fast8_t
|
||||||
|
usb_send_ep0_progmem(const void *data, uint_fast8_t len)
|
||||||
|
{
|
||||||
|
return _usb_send_ep0(data, len, 1);
|
||||||
|
}
|
||||||
|
|
||||||
|
void
|
||||||
|
usb_stall_ep0(void)
|
||||||
|
{
|
||||||
|
UENUM = 0;
|
||||||
|
UECONX = (1<<STALLRQ) | (1<<EPEN);
|
||||||
|
UEIENX = 1<<RXSTPE;
|
||||||
|
}
|
||||||
|
|
||||||
|
static uint8_t set_address;
|
||||||
|
|
||||||
|
void
|
||||||
|
usb_set_address(uint_fast8_t addr)
|
||||||
|
{
|
||||||
|
set_address = addr | (1<<ADDEN);
|
||||||
|
_usb_send_ep0(NULL, 0, 0);
|
||||||
|
UEIENX = (1<<RXSTPE) | (1<<TXINE);
|
||||||
|
}
|
||||||
|
|
||||||
|
void
|
||||||
|
usb_set_configure(void)
|
||||||
|
{
|
||||||
|
UENUM = USB_CDC_EP_ACM;
|
||||||
|
UECONX = 1<<EPEN;
|
||||||
|
UECFG0X = EP_TYPE_INTERRUPT_IN;
|
||||||
|
UECFG1X = EP_SIZE(USB_CDC_EP_ACM_SIZE) | EP_SINGLE_BUFFER;
|
||||||
|
|
||||||
|
UENUM = USB_CDC_EP_BULK_OUT;
|
||||||
|
UECONX = 1<<EPEN;
|
||||||
|
UECFG0X = EP_TYPE_BULK_OUT;
|
||||||
|
UECFG1X = EP_SIZE(USB_CDC_EP_BULK_OUT_SIZE) | EP_DOUBLE_BUFFER;
|
||||||
|
UEIENX = 1<<RXOUTE;
|
||||||
|
|
||||||
|
UENUM = USB_CDC_EP_BULK_IN;
|
||||||
|
UECONX = 1<<EPEN;
|
||||||
|
UECFG0X = EP_TYPE_BULK_IN;
|
||||||
|
UECFG1X = EP_SIZE(USB_CDC_EP_BULK_IN_SIZE) | EP_DOUBLE_BUFFER;
|
||||||
|
UEIENX = 1<<TXINE;
|
||||||
|
}
|
||||||
|
|
||||||
void
|
void
|
||||||
usbserial_init(void)
|
usbserial_init(void)
|
||||||
{
|
{
|
||||||
usb_init();
|
// Set USB controller to device mode
|
||||||
|
UHWCON = (1<<UIMOD) | (1<<UVREGE);
|
||||||
|
|
||||||
|
// Enable USB clock
|
||||||
|
USBCON = (1<<USBE) | (1<<FRZCLK);
|
||||||
|
if (CONFIG_MACH_at90usb1286)
|
||||||
|
PLLCSR = (1<<PLLP2) | (1<<PLLP0) | (1<<PLLE);
|
||||||
|
else
|
||||||
|
PLLCSR = (1<<PLLP2) | (1<<PLLP1) | (1<<PLLE);
|
||||||
|
while (!(PLLCSR & (1<<PLOCK)))
|
||||||
|
;
|
||||||
|
USBCON = (1<<USBE) | (1<<OTGPADE);
|
||||||
|
|
||||||
|
// Enable USB pullup
|
||||||
|
UDCON = 0;
|
||||||
|
|
||||||
|
// Enable interrupts
|
||||||
|
UDIEN = 1<<EORSTE;
|
||||||
}
|
}
|
||||||
DECL_INIT(usbserial_init);
|
DECL_INIT(usbserial_init);
|
||||||
|
|
||||||
// Check for new incoming data
|
ISR(USB_GEN_vect)
|
||||||
static void
|
|
||||||
console_check_input(void)
|
|
||||||
{
|
{
|
||||||
for (;;) {
|
uint8_t udint = UDINT;
|
||||||
if (receive_pos >= sizeof(receive_buf))
|
UDINT = 0;
|
||||||
break;
|
if (udint & (1<<EORSTI)) {
|
||||||
int16_t ret = usb_serial_getchar();
|
// Configure endpoint 0 after usb reset completes
|
||||||
if (ret == -1)
|
uint8_t old_uenum = UENUM;
|
||||||
break;
|
UENUM = 0;
|
||||||
receive_buf[receive_pos++] = ret;
|
UECONX = 1<<EPEN;
|
||||||
|
UECFG0X = EP_TYPE_CONTROL;
|
||||||
|
UECFG1X = EP_SIZE(USB_CDC_EP0_SIZE) | EP_SINGLE_BUFFER;
|
||||||
|
UEIENX = 1<<RXSTPE;
|
||||||
|
UENUM = old_uenum;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Remove from the receive buffer the given number of bytes
|
ISR(USB_COM_vect)
|
||||||
static void
|
|
||||||
console_pop_input(uint8_t len)
|
|
||||||
{
|
{
|
||||||
uint8_t needcopy = receive_pos - len;
|
uint8_t ueint = UEINT, old_uenum = UENUM;
|
||||||
if (needcopy) {
|
if (ueint & (1<<0)) {
|
||||||
memmove(receive_buf, &receive_buf[len], needcopy);
|
UENUM = 0;
|
||||||
sched_wake_tasks();
|
UEIENX = 0;
|
||||||
|
usb_notify_ep0();
|
||||||
|
|
||||||
|
uint8_t ueintx = UEINTX;
|
||||||
|
if (!(ueintx & (1<<RXSTPI)) && (ueintx & (1<<TXINI)) && set_address) {
|
||||||
|
// Ack from set_address command sent - now update address
|
||||||
|
UDADDR = set_address;
|
||||||
|
set_address = 0;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
receive_pos = needcopy;
|
if (ueint & (1<<USB_CDC_EP_BULK_OUT)) {
|
||||||
|
UENUM = USB_CDC_EP_BULK_OUT;
|
||||||
|
UEINTX = ~(1<<RXOUTI);
|
||||||
|
usb_notify_bulk_out();
|
||||||
|
}
|
||||||
|
if (ueint & (1<<USB_CDC_EP_BULK_IN)) {
|
||||||
|
UENUM = USB_CDC_EP_BULK_IN;
|
||||||
|
UEINTX = ~((1<<RXOUTI) | (1<<TXINI));
|
||||||
|
usb_notify_bulk_in();
|
||||||
|
}
|
||||||
|
UENUM = old_uenum;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Process any incoming commands
|
|
||||||
void
|
void
|
||||||
console_task(void)
|
usbserial_shutdown(void)
|
||||||
{
|
{
|
||||||
console_check_input();
|
UEIENX = 1<<RXSTPE;
|
||||||
uint_fast8_t pop_count;
|
|
||||||
int8_t ret = command_find_and_dispatch(receive_buf, receive_pos, &pop_count);
|
|
||||||
if (ret)
|
|
||||||
console_pop_input(pop_count);
|
|
||||||
}
|
|
||||||
DECL_TASK(console_task);
|
|
||||||
|
|
||||||
// Encode and transmit a "response" message
|
|
||||||
void
|
|
||||||
console_sendf(const struct command_encoder *ce, va_list args)
|
|
||||||
{
|
|
||||||
// Generate message
|
|
||||||
static uint8_t buf[MESSAGE_MAX];
|
|
||||||
uint8_t msglen = command_encode_and_frame(buf, ce, args);
|
|
||||||
|
|
||||||
// Transmit message
|
|
||||||
usb_serial_write((void*)buf, msglen);
|
|
||||||
usb_serial_flush_output();
|
|
||||||
}
|
}
|
||||||
|
DECL_SHUTDOWN(usbserial_shutdown);
|
||||||
|
|
Loading…
Reference in New Issue