atsamd2x: improve USBCDC
Этот коммит содержится в:
		
							родитель
							
								
									681be2d4e3
								
							
						
					
					
						коммит
						ba634c4cc7
					
				
					 1 изменённых файлов: 86 добавлений и 22 удалений
				
			
		|  | @ -11,6 +11,7 @@ import ( | ||||||
| 	"device/arm" | 	"device/arm" | ||||||
| 	"device/sam" | 	"device/sam" | ||||||
| 	"runtime/interrupt" | 	"runtime/interrupt" | ||||||
|  | 	"runtime/volatile" | ||||||
| 	"unsafe" | 	"unsafe" | ||||||
| ) | ) | ||||||
| 
 | 
 | ||||||
|  | @ -1464,36 +1465,93 @@ func (pwm PWM) setChannel(timer *sam.TCC_Type, val uint32) { | ||||||
| 
 | 
 | ||||||
| // 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 | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | const ( | ||||||
|  | 	usbcdcTxSizeMask uint8 = 0x3F | ||||||
|  | 	usbcdcTxBankMask uint8 = ^usbcdcTxSizeMask | ||||||
|  | 	usbcdcTxBank1st  uint8 = 0x00 | ||||||
|  | 	usbcdcTxBank2nd  uint8 = usbcdcTxSizeMask + 1 | ||||||
|  | ) | ||||||
|  | 
 | ||||||
|  | // 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 | ||||||
|  | 			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 | ||||||
|  | 			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 | ||||||
|  | 			setEPINTFLAG(usb_CDC_ENDPOINT_IN, sam.USB_DEVICE_EPINTFLAG_TRCPT1) | ||||||
|  | 
 | ||||||
|  | 			// send data by setting bank ready | ||||||
|  | 			setEPSTATUSSET(usb_CDC_ENDPOINT_IN, sam.USB_DEVICE_EPSTATUSSET_BK1RDY) | ||||||
|  | 			UART0.sent = true | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 	return nil | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| // WriteByte writes a byte of data to the USB CDC interface. | // WriteByte writes a byte of data to the USB CDC interface. | ||||||
| func (usbcdc USBCDC) WriteByte(c byte) error { | func (usbcdc *USBCDC) WriteByte(c byte) error { | ||||||
| 	// Supposedly to handle problem with Windows USB serial ports? | 	// Supposedly to handle problem with Windows USB serial ports? | ||||||
| 	if usbLineInfo.lineState > 0 { | 	if usbLineInfo.lineState > 0 { | ||||||
| 		// set the data | 		ok := false | ||||||
| 		udd_ep_in_cache_buffer[usb_CDC_ENDPOINT_IN][0] = c | 		for { | ||||||
|  | 			mask := interrupt.Disable() | ||||||
| 
 | 
 | ||||||
| 		usbEndpointDescriptors[usb_CDC_ENDPOINT_IN].DeviceDescBank[1].ADDR.Set(uint32(uintptr(unsafe.Pointer(&udd_ep_in_cache_buffer[usb_CDC_ENDPOINT_IN])))) | 			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 | ||||||
|  | 			} | ||||||
| 
 | 
 | ||||||
| 		// clean multi packet size of bytes already sent | 			interrupt.Restore(mask) | ||||||
| 		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 | 			if ok { | ||||||
| 		usbEndpointDescriptors[usb_CDC_ENDPOINT_IN].DeviceDescBank[1].PCKSIZE.SetBits((1 & usb_DEVICE_PCKSIZE_BYTE_COUNT_Mask) << usb_DEVICE_PCKSIZE_BYTE_COUNT_Pos) | 				break | ||||||
| 
 | 			} else { | ||||||
| 		// clear transfer complete flag | 				mask := interrupt.Disable() | ||||||
| 		setEPINTFLAG(usb_CDC_ENDPOINT_IN, sam.USB_DEVICE_EPINTFLAG_TRCPT1) | 				if UART0.sent { | ||||||
| 
 | 					if UART0.waitTxc { | ||||||
| 		// send data by setting bank ready | 						if (getEPINTFLAG(usb_CDC_ENDPOINT_IN) & sam.USB_DEVICE_EPINTFLAG_TRCPT1) != 0 { | ||||||
| 		setEPSTATUSSET(usb_CDC_ENDPOINT_IN, sam.USB_DEVICE_EPSTATUSSET_BK1RDY) | 							setEPSTATUSCLR(usb_CDC_ENDPOINT_IN, sam.USB_DEVICE_EPSTATUSCLR_BK1RDY) | ||||||
| 
 | 							setEPINTFLAG(usb_CDC_ENDPOINT_IN, sam.USB_DEVICE_EPINTFLAG_TRCPT1) | ||||||
| 		// wait for transfer to complete | 							UART0.waitTxc = false | ||||||
| 		timeout := 3000 | 							UART0.Flush() | ||||||
| 		for (getEPINTFLAG(usb_CDC_ENDPOINT_IN) & sam.USB_DEVICE_EPINTFLAG_TRCPT1) == 0 { | 						} | ||||||
| 			timeout-- | 					} else { | ||||||
| 			if timeout == 0 { | 						UART0.Flush() | ||||||
| 				return errUSBCDCWriteByteTimeout | 					} | ||||||
|  | 				} | ||||||
|  | 				interrupt.Restore(mask) | ||||||
| 			} | 			} | ||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
|  | @ -1700,9 +1758,15 @@ func handleUSB(intr 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_EPSTATUSCLR_BK1RDY) | 				setEPSTATUSCLR(i, sam.USB_DEVICE_EPSTATUSCLR_BK1RDY) | ||||||
| 				setEPINTFLAG(i, sam.USB_DEVICE_EPINTFLAG_TRCPT1) | 				setEPINTFLAG(i, sam.USB_DEVICE_EPINTFLAG_TRCPT1) | ||||||
|  | 
 | ||||||
|  | 				if i == usb_CDC_ENDPOINT_IN { | ||||||
|  | 					UART0.waitTxc = false | ||||||
|  | 				} | ||||||
| 			} | 			} | ||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
|  | 
 | ||||||
|  | 	UART0.Flush() | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func initEndpoint(ep, config uint32) { | func initEndpoint(ep, config uint32) { | ||||||
|  |  | ||||||
		Загрузка…
	
	Создание таблицы
		
		Сослаться в новой задаче
	
	 sago35
						sago35