forked from mfulz_github/qmk_firmware
		
	Fix HID class device driver -- if a SetIDle request is issued with the LSB of wValue set to zero, the idle period must be set for all HID interfaces.
Fix Keyboard and Mouse demos, Idle period is now multiplied by 4 as the period is read into and sent out of the device to ensure it is always stored as a multiple of 1ms. Fixes Keyboard demo using an initial Idle period of 2s rather than 500ms (thanks to Brian Dickman). Move out the internal device serial descriptor reading routine into a seperate static function, rather than being part of USB_Device_GetDescriptor.
This commit is contained in:
		
							parent
							
								
									da138684e4
								
							
						
					
					
						commit
						f85a53a31b
					
				| @ -225,8 +225,8 @@ void EVENT_USB_UnhandledControlPacket(void) | |||||||
| 			{ | 			{ | ||||||
| 				Endpoint_ClearSETUP(); | 				Endpoint_ClearSETUP(); | ||||||
| 				 | 				 | ||||||
| 				/* Get idle period in MSB */ | 				/* Get idle period in MSB, IdleCount must be multiplied by 4 to get number of milliseconds */ | ||||||
| 				IdleCount = (USB_ControlRequest.wValue >> 8); | 				IdleCount = ((USB_ControlRequest.wValue & 0xFF00) >> 6); | ||||||
| 				 | 				 | ||||||
| 				/* Acknowledge status stage */ | 				/* Acknowledge status stage */ | ||||||
| 				while (!(Endpoint_IsINReady())); | 				while (!(Endpoint_IsINReady())); | ||||||
| @ -239,8 +239,8 @@ void EVENT_USB_UnhandledControlPacket(void) | |||||||
| 			{		 | 			{		 | ||||||
| 				Endpoint_ClearSETUP(); | 				Endpoint_ClearSETUP(); | ||||||
| 				 | 				 | ||||||
| 				/* Write the current idle duration to the host */ | 				/* Write the current idle duration to the host, must be divided by 4 before sent to host */ | ||||||
| 				Endpoint_Write_Byte(IdleCount); | 				Endpoint_Write_Byte(IdleCount >> 2); | ||||||
| 				 | 				 | ||||||
| 				/* Send the flag to the host */ | 				/* Send the flag to the host */ | ||||||
| 				Endpoint_ClearIN(); | 				Endpoint_ClearIN(); | ||||||
| @ -329,8 +329,8 @@ void SendNextReport(void) | |||||||
| 	/* Check if the idle period is set and has elapsed */ | 	/* Check if the idle period is set and has elapsed */ | ||||||
| 	if ((IdleCount != HID_IDLE_CHANGESONLY) && (!(IdleMSRemaining))) | 	if ((IdleCount != HID_IDLE_CHANGESONLY) && (!(IdleMSRemaining))) | ||||||
| 	{ | 	{ | ||||||
| 		/* Reset the idle time remaining counter, must multiply by 4 to get the duration in milliseconds */ | 		/* Reset the idle time remaining counter */ | ||||||
| 		IdleMSRemaining = (IdleCount << 2); | 		IdleMSRemaining = IdleCount; | ||||||
| 		 | 		 | ||||||
| 		/* Idle period is set and has elapsed, must send a report to the host */ | 		/* Idle period is set and has elapsed, must send a report to the host */ | ||||||
| 		SendReport = true; | 		SendReport = true; | ||||||
|  | |||||||
| @ -197,8 +197,8 @@ void EVENT_USB_UnhandledControlPacket(void) | |||||||
| 			{ | 			{ | ||||||
| 				Endpoint_ClearSETUP(); | 				Endpoint_ClearSETUP(); | ||||||
| 				 | 				 | ||||||
| 				/* Get idle period in MSB */ | 				/* Get idle period in MSB, must multiply by 4 to get the duration in milliseconds */ | ||||||
| 				IdleCount = (USB_ControlRequest.wValue >> 8); | 				IdleCount = ((USB_ControlRequest.wValue & 0xFF00) >> 6); | ||||||
| 				 | 				 | ||||||
| 				/* Acknowledge status stage */ | 				/* Acknowledge status stage */ | ||||||
| 				while (!(Endpoint_IsINReady())); | 				while (!(Endpoint_IsINReady())); | ||||||
| @ -211,8 +211,8 @@ void EVENT_USB_UnhandledControlPacket(void) | |||||||
| 			{		 | 			{		 | ||||||
| 				Endpoint_ClearSETUP(); | 				Endpoint_ClearSETUP(); | ||||||
| 				 | 				 | ||||||
| 				/* Write the current idle duration to the host */ | 				/* Write the current idle duration to the host, must be divided by 4 before sent to host */ | ||||||
| 				Endpoint_Write_Byte(IdleCount); | 				Endpoint_Write_Byte(IdleCount >> 2); | ||||||
| 				 | 				 | ||||||
| 				/* Send the flag to the host */ | 				/* Send the flag to the host */ | ||||||
| 				Endpoint_ClearIN(); | 				Endpoint_ClearIN(); | ||||||
| @ -289,8 +289,8 @@ void SendNextReport(void) | |||||||
| 	/* Check if the idle period is set and has elapsed */ | 	/* Check if the idle period is set and has elapsed */ | ||||||
| 	if ((IdleCount != HID_IDLE_CHANGESONLY) && (!(IdleMSRemaining))) | 	if ((IdleCount != HID_IDLE_CHANGESONLY) && (!(IdleMSRemaining))) | ||||||
| 	{ | 	{ | ||||||
| 		/* Reset the idle time remaining counter, must multiply by 4 to get the duration in milliseconds */ | 		/* Reset the idle time remaining counter */ | ||||||
| 		IdleMSRemaining = (IdleCount << 2); | 		IdleMSRemaining = IdleCount; | ||||||
| 		 | 		 | ||||||
| 		/* Idle period is set and has elapsed, must send a report to the host */ | 		/* Idle period is set and has elapsed, must send a report to the host */ | ||||||
| 		SendReport = true; | 		SendReport = true; | ||||||
|  | |||||||
| @ -38,8 +38,11 @@ void HID_Device_ProcessControlPacket(USB_ClassInfo_HID_Device_t* const HIDInterf | |||||||
| 	if (!(Endpoint_IsSETUPReceived())) | 	if (!(Endpoint_IsSETUPReceived())) | ||||||
| 	  return; | 	  return; | ||||||
| 	   | 	   | ||||||
| 	if (USB_ControlRequest.wIndex != HIDInterfaceInfo->Config.InterfaceNumber) | 	if ((USB_ControlRequest.wIndex   != HIDInterfaceInfo->Config.InterfaceNumber) && | ||||||
|  | 	    (USB_ControlRequest.bRequest != SetIdle)) | ||||||
|  | 	{ | ||||||
| 		return; | 		return; | ||||||
|  | 	} | ||||||
| 
 | 
 | ||||||
| 	switch (USB_ControlRequest.bRequest) | 	switch (USB_ControlRequest.bRequest) | ||||||
| 	{ | 	{ | ||||||
| @ -104,14 +107,18 @@ void HID_Device_ProcessControlPacket(USB_ClassInfo_HID_Device_t* const HIDInterf | |||||||
| 			break; | 			break; | ||||||
| 		case REQ_SetIdle: | 		case REQ_SetIdle: | ||||||
| 			if (USB_ControlRequest.bmRequestType == (REQDIR_HOSTTODEVICE | REQTYPE_CLASS | REQREC_INTERFACE)) | 			if (USB_ControlRequest.bmRequestType == (REQDIR_HOSTTODEVICE | REQTYPE_CLASS | REQREC_INTERFACE)) | ||||||
|  | 			{ | ||||||
|  | 				if ((USB_ControlRequest.wIndex         == HIDInterfaceInfo->Config.InterfaceNumber) || | ||||||
|  | 				    (USB_ControlRequest.wValue & 0xFF) == 0) | ||||||
| 				{ | 				{ | ||||||
| 					Endpoint_ClearSETUP(); | 					Endpoint_ClearSETUP(); | ||||||
| 					 | 					 | ||||||
| 				HIDInterfaceInfo->State.IdleCount = ((USB_ControlRequest.wValue >> 8) << 2); | 					HIDInterfaceInfo->State.IdleCount = ((USB_ControlRequest.wValue & 0xFF00) >> 6); | ||||||
| 					 | 					 | ||||||
| 					while (!(Endpoint_IsINReady())); | 					while (!(Endpoint_IsINReady())); | ||||||
| 					Endpoint_ClearIN(); | 					Endpoint_ClearIN(); | ||||||
| 				} | 				} | ||||||
|  | 			} | ||||||
| 			 | 			 | ||||||
| 			break; | 			break; | ||||||
| 		case REQ_GetIdle: | 		case REQ_GetIdle: | ||||||
|  | |||||||
| @ -174,20 +174,18 @@ void USB_Device_GetConfiguration(void) | |||||||
| 	Endpoint_ClearOUT(); | 	Endpoint_ClearOUT(); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| static void USB_Device_GetDescriptor(void) | #if !defined(NO_INTERNAL_SERIAL) && (defined(USB_SERIES_6_AVR) || defined(USB_SERIES_7_AVR)) | ||||||
|  | static void USB_Device_GetInternalSerialDescriptor(void) | ||||||
| { | { | ||||||
| 	void*    DescriptorPointer; |  | ||||||
| 	uint16_t DescriptorSize; |  | ||||||
| 	 |  | ||||||
| 	#if !defined(NO_INTERNAL_SERIAL) && (defined(USB_SERIES_6_AVR) || defined(USB_SERIES_7_AVR)) |  | ||||||
| 	if (USB_ControlRequest.wValue == ((DTYPE_String << 8) | USE_INTERNAL_SERIAL)) |  | ||||||
| 	{ |  | ||||||
| 	struct | 	struct | ||||||
| 	{ | 	{ | ||||||
| 		USB_Descriptor_Header_t Header; | 		USB_Descriptor_Header_t Header; | ||||||
| 		int16_t                 UnicodeString[12]; | 		int16_t                 UnicodeString[12]; | ||||||
| 	} SignatureDescriptor; | 	} SignatureDescriptor; | ||||||
| 	 | 	 | ||||||
|  | 	uint8_t SigReadAddress  = 0x0E;		 | ||||||
|  | 	bool    OddNibbleRead   = false; | ||||||
|  | 
 | ||||||
| 	#if defined(USE_NONSTANDARD_DESCRIPTOR_NAMES) | 	#if defined(USE_NONSTANDARD_DESCRIPTOR_NAMES) | ||||||
| 		SignatureDescriptor.Header.Size            = sizeof(SignatureDescriptor); | 		SignatureDescriptor.Header.Size            = sizeof(SignatureDescriptor); | ||||||
| 		SignatureDescriptor.Header.Type            = DTYPE_String; | 		SignatureDescriptor.Header.Type            = DTYPE_String; | ||||||
| @ -196,14 +194,11 @@ static void USB_Device_GetDescriptor(void) | |||||||
| 		SignatureDescriptor.Header.bDescriptorType = DTYPE_String; | 		SignatureDescriptor.Header.bDescriptorType = DTYPE_String; | ||||||
| 	#endif | 	#endif | ||||||
| 
 | 
 | ||||||
| 		uint8_t  SigReadAddress  = 0x0E;		 |  | ||||||
| 		bool     OddRead         = false; |  | ||||||
| 
 |  | ||||||
| 	for (uint8_t SerialCharNum = 0; SerialCharNum < 12; SerialCharNum++) | 	for (uint8_t SerialCharNum = 0; SerialCharNum < 12; SerialCharNum++) | ||||||
| 	{ | 	{ | ||||||
| 		uint8_t SerialByte = boot_signature_byte_get(SigReadAddress); | 		uint8_t SerialByte = boot_signature_byte_get(SigReadAddress); | ||||||
| 		 | 		 | ||||||
| 			if (OddRead) | 		if (OddNibbleRead) | ||||||
| 		{ | 		{ | ||||||
| 			SerialByte >>= 4; | 			SerialByte >>= 4; | ||||||
| 			SigReadAddress++; | 			SigReadAddress++; | ||||||
| @ -213,7 +208,7 @@ static void USB_Device_GetDescriptor(void) | |||||||
| 			SerialByte &= 0x0F; | 			SerialByte &= 0x0F; | ||||||
| 		} | 		} | ||||||
| 		 | 		 | ||||||
| 			OddRead = !(OddRead); | 		OddNibbleRead = !(OddNibbleRead); | ||||||
| 
 | 
 | ||||||
| 		if (SerialByte < 0x0A) | 		if (SerialByte < 0x0A) | ||||||
| 		  SerialByte += '0'; | 		  SerialByte += '0'; | ||||||
| @ -226,7 +221,18 @@ static void USB_Device_GetDescriptor(void) | |||||||
| 	Endpoint_ClearSETUP(); | 	Endpoint_ClearSETUP(); | ||||||
| 	Endpoint_Write_Control_Stream_LE(&SignatureDescriptor, sizeof(SignatureDescriptor)); | 	Endpoint_Write_Control_Stream_LE(&SignatureDescriptor, sizeof(SignatureDescriptor)); | ||||||
| 	Endpoint_ClearOUT(); | 	Endpoint_ClearOUT(); | ||||||
|  | } | ||||||
|  | #endif | ||||||
| 
 | 
 | ||||||
|  | static void USB_Device_GetDescriptor(void) | ||||||
|  | { | ||||||
|  | 	void*    DescriptorPointer; | ||||||
|  | 	uint16_t DescriptorSize; | ||||||
|  | 	 | ||||||
|  | 	#if !defined(NO_INTERNAL_SERIAL) && (defined(USB_SERIES_6_AVR) || defined(USB_SERIES_7_AVR)) | ||||||
|  | 	if (USB_ControlRequest.wValue == ((DTYPE_String << 8) | USE_INTERNAL_SERIAL)) | ||||||
|  | 	{ | ||||||
|  | 		USB_Device_GetInternalSerialDescriptor(); | ||||||
| 		return; | 		return; | ||||||
| 	} | 	} | ||||||
| 	#endif | 	#endif | ||||||
|  | |||||||
| @ -99,6 +99,10 @@ | |||||||
| 				static void USB_Device_GetDescriptor(void); | 				static void USB_Device_GetDescriptor(void); | ||||||
| 				static void USB_Device_GetStatus(void); | 				static void USB_Device_GetStatus(void); | ||||||
| 				static void USB_Device_ClearSetFeature(void); | 				static void USB_Device_ClearSetFeature(void); | ||||||
|  | 				 | ||||||
|  | 				#if !defined(NO_INTERNAL_SERIAL) && (defined(USB_SERIES_6_AVR) || defined(USB_SERIES_7_AVR)) | ||||||
|  | 					static void USB_Device_GetInternalSerialDescriptor(void); | ||||||
|  | 				#endif				 | ||||||
| 			#endif | 			#endif | ||||||
| 	#endif | 	#endif | ||||||
| 
 | 
 | ||||||
|  | |||||||
| @ -55,6 +55,7 @@ | |||||||
|   *  - Fixed USB_Host_SendControlRequest() not re-suspending the USB bus when initial device ready-wait fails |   *  - Fixed USB_Host_SendControlRequest() not re-suspending the USB bus when initial device ready-wait fails | ||||||
|   *  - Fixed USB Pad regulator not being disabled on some AVR models when the USB_OPT_REG_DISABLED option is used |   *  - Fixed USB Pad regulator not being disabled on some AVR models when the USB_OPT_REG_DISABLED option is used | ||||||
|   *  - Fixed Host mode to Device mode UID change not causing a USB Disconnect event when a device was connected |   *  - Fixed Host mode to Device mode UID change not causing a USB Disconnect event when a device was connected | ||||||
|  |   *  - Fixed Mouse/Keyboard demos not performing the correct arithmetic on the Idle period at the right times (thanks to Brian Dickman) | ||||||
|   * |   * | ||||||
|   * |   * | ||||||
|   *  \section Sec_ChangeLog090605 Version 090605 |   *  \section Sec_ChangeLog090605 Version 090605 | ||||||
|  | |||||||
		Loading…
	
	
			
			x
			
			
		
	
		Reference in New Issue
	
	Block a user
	 Dean Camera
						Dean Camera