mirror of
				git://git.kernel.org/pub/scm/linux/kernel/git/torvalds/linux.git
				synced 2025-09-04 20:19:47 +08:00 
			
		
		
		
	can: peak_usb: pcan_usb_decode_error(): upgrade handling of bus state changes
This patch updates old code by using the can_change_state() function published since by the socket-can module. In particular, this new code better manages the change of bus state by also using the value of the error counters that the driver now systematically asks for when initializing the channel. Link: https://lore.kernel.org/r/20210715142842.35793-1-s.grosjean@peak-system.com Signed-off-by: Stephane Grosjean <s.grosjean@peak-system.com> Signed-off-by: Marc Kleine-Budde <mkl@pengutronix.de>
This commit is contained in:
		
							parent
							
								
									1763c54764
								
							
						
					
					
						commit
						c11dcee758
					
				| @ -452,145 +452,65 @@ static int pcan_usb_decode_error(struct pcan_usb_msg_context *mc, u8 n, | |||||||
| { | { | ||||||
| 	struct sk_buff *skb; | 	struct sk_buff *skb; | ||||||
| 	struct can_frame *cf; | 	struct can_frame *cf; | ||||||
| 	enum can_state new_state; | 	enum can_state new_state = CAN_STATE_ERROR_ACTIVE; | ||||||
| 
 | 
 | ||||||
| 	/* ignore this error until 1st ts received */ | 	/* ignore this error until 1st ts received */ | ||||||
| 	if (n == PCAN_USB_ERROR_QOVR) | 	if (n == PCAN_USB_ERROR_QOVR) | ||||||
| 		if (!mc->pdev->time_ref.tick_count) | 		if (!mc->pdev->time_ref.tick_count) | ||||||
| 			return 0; | 			return 0; | ||||||
| 
 | 
 | ||||||
| 	new_state = mc->pdev->dev.can.state; |  | ||||||
| 
 |  | ||||||
| 	switch (mc->pdev->dev.can.state) { |  | ||||||
| 	case CAN_STATE_ERROR_ACTIVE: |  | ||||||
| 		if (n & PCAN_USB_ERROR_BUS_LIGHT) { |  | ||||||
| 			new_state = CAN_STATE_ERROR_WARNING; |  | ||||||
| 			break; |  | ||||||
| 		} |  | ||||||
| 		fallthrough; |  | ||||||
| 
 |  | ||||||
| 	case CAN_STATE_ERROR_WARNING: |  | ||||||
| 		if (n & PCAN_USB_ERROR_BUS_HEAVY) { |  | ||||||
| 			new_state = CAN_STATE_ERROR_PASSIVE; |  | ||||||
| 			break; |  | ||||||
| 		} |  | ||||||
| 		if (n & PCAN_USB_ERROR_BUS_OFF) { |  | ||||||
| 			new_state = CAN_STATE_BUS_OFF; |  | ||||||
| 			break; |  | ||||||
| 		} |  | ||||||
| 		if (n & ~PCAN_USB_ERROR_BUS) { |  | ||||||
| 			/*
 |  | ||||||
| 			 * trick to bypass next comparison and process other |  | ||||||
| 			 * errors |  | ||||||
| 			 */ |  | ||||||
| 			new_state = CAN_STATE_MAX; |  | ||||||
| 			break; |  | ||||||
| 		} |  | ||||||
| 		if ((n & PCAN_USB_ERROR_BUS_LIGHT) == 0) { |  | ||||||
| 			/* no error (back to active state) */ |  | ||||||
| 			new_state = CAN_STATE_ERROR_ACTIVE; |  | ||||||
| 			break; |  | ||||||
| 		} |  | ||||||
| 		break; |  | ||||||
| 
 |  | ||||||
| 	case CAN_STATE_ERROR_PASSIVE: |  | ||||||
| 		if (n & PCAN_USB_ERROR_BUS_OFF) { |  | ||||||
| 			new_state = CAN_STATE_BUS_OFF; |  | ||||||
| 			break; |  | ||||||
| 		} |  | ||||||
| 		if (n & PCAN_USB_ERROR_BUS_LIGHT) { |  | ||||||
| 			new_state = CAN_STATE_ERROR_WARNING; |  | ||||||
| 			break; |  | ||||||
| 		} |  | ||||||
| 		if (n & ~PCAN_USB_ERROR_BUS) { |  | ||||||
| 			/*
 |  | ||||||
| 			 * trick to bypass next comparison and process other |  | ||||||
| 			 * errors |  | ||||||
| 			 */ |  | ||||||
| 			new_state = CAN_STATE_MAX; |  | ||||||
| 			break; |  | ||||||
| 		} |  | ||||||
| 
 |  | ||||||
| 		if ((n & PCAN_USB_ERROR_BUS_HEAVY) == 0) { |  | ||||||
| 			/* no error (back to warning state) */ |  | ||||||
| 			new_state = CAN_STATE_ERROR_WARNING; |  | ||||||
| 			break; |  | ||||||
| 		} |  | ||||||
| 		break; |  | ||||||
| 
 |  | ||||||
| 	default: |  | ||||||
| 		/* do nothing waiting for restart */ |  | ||||||
| 		return 0; |  | ||||||
| 	} |  | ||||||
| 
 |  | ||||||
| 	/* donot post any error if current state didn't change */ |  | ||||||
| 	if (mc->pdev->dev.can.state == new_state) |  | ||||||
| 		return 0; |  | ||||||
| 
 |  | ||||||
| 	/* allocate an skb to store the error frame */ | 	/* allocate an skb to store the error frame */ | ||||||
| 	skb = alloc_can_err_skb(mc->netdev, &cf); | 	skb = alloc_can_err_skb(mc->netdev, &cf); | ||||||
| 	if (!skb) |  | ||||||
| 		return -ENOMEM; |  | ||||||
| 
 | 
 | ||||||
| 	switch (new_state) { | 	if (n & PCAN_USB_ERROR_RXQOVR) { | ||||||
| 	case CAN_STATE_BUS_OFF: | 		/* data overrun interrupt */ | ||||||
| 		cf->can_id |= CAN_ERR_BUSOFF; | 		netdev_dbg(mc->netdev, "data overrun interrupt\n"); | ||||||
| 		mc->pdev->dev.can.can_stats.bus_off++; | 		mc->netdev->stats.rx_over_errors++; | ||||||
| 		can_bus_off(mc->netdev); | 		mc->netdev->stats.rx_errors++; | ||||||
| 		break; | 		if (cf) { | ||||||
| 
 |  | ||||||
| 	case CAN_STATE_ERROR_PASSIVE: |  | ||||||
| 			cf->can_id |= CAN_ERR_CRTL; | 			cf->can_id |= CAN_ERR_CRTL; | ||||||
| 		cf->data[1] = (mc->pdev->bec.txerr > mc->pdev->bec.rxerr) ? | 			cf->data[1] |= CAN_ERR_CRTL_RX_OVERFLOW; | ||||||
| 				CAN_ERR_CRTL_TX_PASSIVE : | 		} | ||||||
| 				CAN_ERR_CRTL_RX_PASSIVE; | 	} | ||||||
| 		cf->data[6] = mc->pdev->bec.txerr; |  | ||||||
| 		cf->data[7] = mc->pdev->bec.rxerr; |  | ||||||
| 
 | 
 | ||||||
| 		mc->pdev->dev.can.can_stats.error_passive++; |  | ||||||
| 		break; |  | ||||||
| 
 |  | ||||||
| 	case CAN_STATE_ERROR_WARNING: |  | ||||||
| 		cf->can_id |= CAN_ERR_CRTL; |  | ||||||
| 		cf->data[1] = (mc->pdev->bec.txerr > mc->pdev->bec.rxerr) ? |  | ||||||
| 				CAN_ERR_CRTL_TX_WARNING : |  | ||||||
| 				CAN_ERR_CRTL_RX_WARNING; |  | ||||||
| 		cf->data[6] = mc->pdev->bec.txerr; |  | ||||||
| 		cf->data[7] = mc->pdev->bec.rxerr; |  | ||||||
| 
 |  | ||||||
| 		mc->pdev->dev.can.can_stats.error_warning++; |  | ||||||
| 		break; |  | ||||||
| 
 |  | ||||||
| 	case CAN_STATE_ERROR_ACTIVE: |  | ||||||
| 		cf->can_id |= CAN_ERR_CRTL; |  | ||||||
| 		cf->data[1] = CAN_ERR_CRTL_ACTIVE; |  | ||||||
| 
 |  | ||||||
| 		/* sync local copies of rxerr/txerr counters */ |  | ||||||
| 		mc->pdev->bec.txerr = 0; |  | ||||||
| 		mc->pdev->bec.rxerr = 0; |  | ||||||
| 		break; |  | ||||||
| 
 |  | ||||||
| 	default: |  | ||||||
| 		/* CAN_STATE_MAX (trick to handle other errors) */ |  | ||||||
| 	if (n & PCAN_USB_ERROR_TXQFULL) | 	if (n & PCAN_USB_ERROR_TXQFULL) | ||||||
| 		netdev_dbg(mc->netdev, "device Tx queue full)\n"); | 		netdev_dbg(mc->netdev, "device Tx queue full)\n"); | ||||||
| 
 | 
 | ||||||
| 		if (n & PCAN_USB_ERROR_RXQOVR) { | 	if (n & PCAN_USB_ERROR_BUS_OFF) { | ||||||
| 			netdev_dbg(mc->netdev, "data overrun interrupt\n"); | 		new_state = CAN_STATE_BUS_OFF; | ||||||
| 			cf->can_id |= CAN_ERR_CRTL; | 	} else if (n & PCAN_USB_ERROR_BUS_HEAVY) { | ||||||
| 			cf->data[1] |= CAN_ERR_CRTL_RX_OVERFLOW; | 		new_state = ((mc->pdev->bec.txerr >= 128) || | ||||||
| 			mc->netdev->stats.rx_over_errors++; | 			     (mc->pdev->bec.rxerr >= 128)) ? | ||||||
| 			mc->netdev->stats.rx_errors++; | 				CAN_STATE_ERROR_PASSIVE : | ||||||
|  | 				CAN_STATE_ERROR_WARNING; | ||||||
|  | 	} else { | ||||||
|  | 		new_state = CAN_STATE_ERROR_ACTIVE; | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
|  | 	/* handle change of state */ | ||||||
|  | 	if (new_state != mc->pdev->dev.can.state) { | ||||||
|  | 		enum can_state tx_state = | ||||||
|  | 			(mc->pdev->bec.txerr >= mc->pdev->bec.rxerr) ? | ||||||
|  | 				new_state : 0; | ||||||
|  | 		enum can_state rx_state = | ||||||
|  | 			(mc->pdev->bec.txerr <= mc->pdev->bec.rxerr) ? | ||||||
|  | 				new_state : 0; | ||||||
|  | 
 | ||||||
|  | 		can_change_state(mc->netdev, cf, tx_state, rx_state); | ||||||
|  | 
 | ||||||
|  | 		if (new_state == CAN_STATE_BUS_OFF) { | ||||||
|  | 			can_bus_off(mc->netdev); | ||||||
|  | 		} else if (cf && (cf->can_id & CAN_ERR_CRTL)) { | ||||||
|  | 			/* Supply TX/RX error counters in case of
 | ||||||
|  | 			 * controller error. | ||||||
|  | 			 */ | ||||||
| 			cf->data[6] = mc->pdev->bec.txerr; | 			cf->data[6] = mc->pdev->bec.txerr; | ||||||
| 			cf->data[7] = mc->pdev->bec.rxerr; | 			cf->data[7] = mc->pdev->bec.rxerr; | ||||||
| 
 | 		} | ||||||
| 		new_state = mc->pdev->dev.can.state; |  | ||||||
| 		break; |  | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	mc->pdev->dev.can.state = new_state; | 	if (!skb) | ||||||
|  | 		return -ENOMEM; | ||||||
| 
 | 
 | ||||||
| 	if (status_len & PCAN_USB_STATUSLEN_TIMESTAMP) { | 	if (status_len & PCAN_USB_STATUSLEN_TIMESTAMP) { | ||||||
| 		struct skb_shared_hwtstamps *hwts = skb_hwtstamps(skb); | 		struct skb_shared_hwtstamps *hwts = skb_hwtstamps(skb); | ||||||
|  | |||||||
		Loading…
	
		Reference in New Issue
	
	Block a user
	 Stephane Grosjean
						Stephane Grosjean