atsamd5x: improve USBCDC
Этот коммит содержится в:
родитель
d274e299ac
коммит
01b917fb11
1 изменённых файлов: 86 добавлений и 22 удалений
|
@ -11,6 +11,7 @@ import (
|
||||||
"device/arm"
|
"device/arm"
|
||||||
"device/sam"
|
"device/sam"
|
||||||
"runtime/interrupt"
|
"runtime/interrupt"
|
||||||
|
"runtime/volatile"
|
||||||
"unsafe"
|
"unsafe"
|
||||||
)
|
)
|
||||||
|
|
||||||
|
@ -1734,35 +1735,92 @@ func (pwm PWM) getMux() PinMode {
|
||||||
// USBCDC is the USB CDC aka serial over USB interface on the SAMD21.
|
// USBCDC is the USB CDC aka serial over USB interface on the SAMD21.
|
||||||
type USBCDC struct {
|
type USBCDC struct {
|
||||||
Buffer *RingBuffer
|
Buffer *RingBuffer
|
||||||
|
TxIdx volatile.Register8
|
||||||
|
waitTxc bool
|
||||||
|
sent bool
|
||||||
}
|
}
|
||||||
|
|
||||||
// WriteByte writes a byte of data to the USB CDC interface.
|
const (
|
||||||
func (usbcdc USBCDC) WriteByte(c byte) error {
|
usbcdcTxSizeMask uint8 = 0x3F
|
||||||
// Supposedly to handle problem with Windows USB serial ports?
|
usbcdcTxBankMask uint8 = ^usbcdcTxSizeMask
|
||||||
if usbLineInfo.lineState > 0 {
|
usbcdcTxBank1st uint8 = 0x00
|
||||||
// set the data
|
usbcdcTxBank2nd uint8 = usbcdcTxSizeMask + 1
|
||||||
udd_ep_in_cache_buffer[usb_CDC_ENDPOINT_IN][0] = c
|
)
|
||||||
|
|
||||||
usbEndpointDescriptors[usb_CDC_ENDPOINT_IN].DeviceDescBank[1].ADDR.Set(uint32(uintptr(unsafe.Pointer(&udd_ep_in_cache_buffer[usb_CDC_ENDPOINT_IN]))))
|
// Flush flushes buffered data.
|
||||||
|
func (usbcdc *USBCDC) Flush() error {
|
||||||
|
if usbLineInfo.lineState > 0 {
|
||||||
|
idx := usbcdc.TxIdx.Get()
|
||||||
|
sz := idx & usbcdcTxSizeMask
|
||||||
|
bk := idx & usbcdcTxBankMask
|
||||||
|
if 0 < sz {
|
||||||
|
|
||||||
|
if usbcdc.waitTxc {
|
||||||
|
// waiting for the next flush(), because the transmission is not complete
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
usbcdc.waitTxc = true
|
||||||
|
|
||||||
|
// set the data
|
||||||
|
usbEndpointDescriptors[usb_CDC_ENDPOINT_IN].DeviceDescBank[1].ADDR.Set(uint32(uintptr(unsafe.Pointer(&udd_ep_in_cache_buffer[usb_CDC_ENDPOINT_IN][bk]))))
|
||||||
|
if bk == usbcdcTxBank1st {
|
||||||
|
usbcdc.TxIdx.Set(usbcdcTxBank2nd)
|
||||||
|
} else {
|
||||||
|
usbcdc.TxIdx.Set(usbcdcTxBank1st)
|
||||||
|
}
|
||||||
|
|
||||||
// clean multi packet size of bytes already sent
|
// clean multi packet size of bytes already sent
|
||||||
usbEndpointDescriptors[usb_CDC_ENDPOINT_IN].DeviceDescBank[1].PCKSIZE.ClearBits(usb_DEVICE_PCKSIZE_MULTI_PACKET_SIZE_Mask << usb_DEVICE_PCKSIZE_MULTI_PACKET_SIZE_Pos)
|
usbEndpointDescriptors[usb_CDC_ENDPOINT_IN].DeviceDescBank[1].PCKSIZE.ClearBits(usb_DEVICE_PCKSIZE_MULTI_PACKET_SIZE_Mask << usb_DEVICE_PCKSIZE_MULTI_PACKET_SIZE_Pos)
|
||||||
|
|
||||||
// set count of bytes to be sent
|
// set count of bytes to be sent
|
||||||
usbEndpointDescriptors[usb_CDC_ENDPOINT_IN].DeviceDescBank[1].PCKSIZE.SetBits((1 & usb_DEVICE_PCKSIZE_BYTE_COUNT_Mask) << usb_DEVICE_PCKSIZE_BYTE_COUNT_Pos)
|
usbEndpointDescriptors[usb_CDC_ENDPOINT_IN].DeviceDescBank[1].PCKSIZE.ClearBits(usb_DEVICE_PCKSIZE_BYTE_COUNT_Mask << usb_DEVICE_PCKSIZE_BYTE_COUNT_Pos)
|
||||||
|
usbEndpointDescriptors[usb_CDC_ENDPOINT_IN].DeviceDescBank[1].PCKSIZE.SetBits((uint32(sz) & usb_DEVICE_PCKSIZE_BYTE_COUNT_Mask) << usb_DEVICE_PCKSIZE_BYTE_COUNT_Pos)
|
||||||
|
|
||||||
// clear transfer complete flag
|
// clear transfer complete flag
|
||||||
setEPINTFLAG(usb_CDC_ENDPOINT_IN, sam.USB_DEVICE_ENDPOINT_EPINTFLAG_TRCPT1)
|
setEPINTFLAG(usb_CDC_ENDPOINT_IN, sam.USB_DEVICE_ENDPOINT_EPINTFLAG_TRCPT1)
|
||||||
|
|
||||||
// send data by setting bank ready
|
// send data by setting bank ready
|
||||||
setEPSTATUSSET(usb_CDC_ENDPOINT_IN, sam.USB_DEVICE_ENDPOINT_EPSTATUSSET_BK1RDY)
|
setEPSTATUSSET(usb_CDC_ENDPOINT_IN, sam.USB_DEVICE_ENDPOINT_EPSTATUSSET_BK1RDY)
|
||||||
|
UART0.sent = true
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
// wait for transfer to complete
|
// WriteByte writes a byte of data to the USB CDC interface.
|
||||||
timeout := 3000
|
func (usbcdc *USBCDC) WriteByte(c byte) error {
|
||||||
for (getEPINTFLAG(usb_CDC_ENDPOINT_IN) & sam.USB_DEVICE_ENDPOINT_EPINTFLAG_TRCPT1) == 0 {
|
// Supposedly to handle problem with Windows USB serial ports?
|
||||||
timeout--
|
if usbLineInfo.lineState > 0 {
|
||||||
if timeout == 0 {
|
ok := false
|
||||||
return errUSBCDCWriteByteTimeout
|
for {
|
||||||
|
mask := interrupt.Disable()
|
||||||
|
|
||||||
|
idx := UART0.TxIdx.Get()
|
||||||
|
if (idx & usbcdcTxSizeMask) < usbcdcTxSizeMask {
|
||||||
|
udd_ep_in_cache_buffer[usb_CDC_ENDPOINT_IN][idx] = c
|
||||||
|
UART0.TxIdx.Set(idx + 1)
|
||||||
|
ok = true
|
||||||
|
}
|
||||||
|
|
||||||
|
interrupt.Restore(mask)
|
||||||
|
|
||||||
|
if ok {
|
||||||
|
break
|
||||||
|
} else {
|
||||||
|
mask := interrupt.Disable()
|
||||||
|
if UART0.sent {
|
||||||
|
if UART0.waitTxc {
|
||||||
|
if (getEPINTFLAG(usb_CDC_ENDPOINT_IN) & sam.USB_DEVICE_ENDPOINT_EPINTFLAG_TRCPT1) != 0 {
|
||||||
|
setEPSTATUSCLR(usb_CDC_ENDPOINT_IN, sam.USB_DEVICE_ENDPOINT_EPSTATUSCLR_BK1RDY)
|
||||||
|
setEPINTFLAG(usb_CDC_ENDPOINT_IN, sam.USB_DEVICE_ENDPOINT_EPINTFLAG_TRCPT1)
|
||||||
|
UART0.waitTxc = false
|
||||||
|
UART0.Flush()
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
UART0.Flush()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
interrupt.Restore(mask)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -1971,9 +2029,15 @@ func handleUSBIRQ(interrupt.Interrupt) {
|
||||||
case usb_CDC_ENDPOINT_IN, usb_CDC_ENDPOINT_ACM:
|
case usb_CDC_ENDPOINT_IN, usb_CDC_ENDPOINT_ACM:
|
||||||
setEPSTATUSCLR(i, sam.USB_DEVICE_ENDPOINT_EPSTATUSCLR_BK1RDY)
|
setEPSTATUSCLR(i, sam.USB_DEVICE_ENDPOINT_EPSTATUSCLR_BK1RDY)
|
||||||
setEPINTFLAG(i, sam.USB_DEVICE_ENDPOINT_EPINTFLAG_TRCPT1)
|
setEPINTFLAG(i, sam.USB_DEVICE_ENDPOINT_EPINTFLAG_TRCPT1)
|
||||||
|
|
||||||
|
if i == usb_CDC_ENDPOINT_IN {
|
||||||
|
UART0.waitTxc = false
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
UART0.Flush()
|
||||||
}
|
}
|
||||||
|
|
||||||
func initEndpoint(ep, config uint32) {
|
func initEndpoint(ep, config uint32) {
|
||||||
|
|
Загрузка…
Создание таблицы
Сослаться в новой задаче