mirror of
				git://git.kernel.org/pub/scm/linux/kernel/git/torvalds/linux.git
				synced 2025-09-04 20:19:47 +08:00 
			
		
		
		
	net: stmmac: fix taprio configuration when base_time is in the past
The Synopsys TSN MAC supports Qbv base times in the past, but only up to a
certain limit. As a result, a taprio qdisc configuration with a small
base time (for example when treating the base time as a simple phase
offset) is not applied by the hardware and silently ignored.
This was observed on an NXP i.MX8MPlus device, but likely affects all
TSN-variants of the MAC.
Fix the issue by making sure the base time is in the future, pushing it by
an integer amount of cycle times if needed. (a similar check is already
done in several other taprio implementations, see for example
drivers/net/ethernet/intel/igc/igc_tsn.c#L116 or
drivers/net/dsa/sja1105/sja1105_ptp.h#L39).
Fixes: b60189e039 ("net: stmmac: Integrate EST with TAPRIO scheduler API")
Signed-off-by: Yannick Vignon <yannick.vignon@nxp.com>
Link: https://lore.kernel.org/r/20210113131557.24651-2-yannick.vignon@oss.nxp.com
Signed-off-by: Jakub Kicinski <kuba@kernel.org>
			
			
This commit is contained in:
		
							parent
							
								
									b76889ff51
								
							
						
					
					
						commit
						fe28c53ed7
					
				| @ -599,7 +599,8 @@ static int tc_setup_taprio(struct stmmac_priv *priv, | ||||
| { | ||||
| 	u32 size, wid = priv->dma_cap.estwid, dep = priv->dma_cap.estdep; | ||||
| 	struct plat_stmmacenet_data *plat = priv->plat; | ||||
| 	struct timespec64 time; | ||||
| 	struct timespec64 time, current_time; | ||||
| 	ktime_t current_time_ns; | ||||
| 	bool fpe = false; | ||||
| 	int i, ret = 0; | ||||
| 	u64 ctr; | ||||
| @ -694,7 +695,22 @@ static int tc_setup_taprio(struct stmmac_priv *priv, | ||||
| 	} | ||||
| 
 | ||||
| 	/* Adjust for real system time */ | ||||
| 	time = ktime_to_timespec64(qopt->base_time); | ||||
| 	priv->ptp_clock_ops.gettime64(&priv->ptp_clock_ops, ¤t_time); | ||||
| 	current_time_ns = timespec64_to_ktime(current_time); | ||||
| 	if (ktime_after(qopt->base_time, current_time_ns)) { | ||||
| 		time = ktime_to_timespec64(qopt->base_time); | ||||
| 	} else { | ||||
| 		ktime_t base_time; | ||||
| 		s64 n; | ||||
| 
 | ||||
| 		n = div64_s64(ktime_sub_ns(current_time_ns, qopt->base_time), | ||||
| 			      qopt->cycle_time); | ||||
| 		base_time = ktime_add_ns(qopt->base_time, | ||||
| 					 (n + 1) * qopt->cycle_time); | ||||
| 
 | ||||
| 		time = ktime_to_timespec64(base_time); | ||||
| 	} | ||||
| 
 | ||||
| 	priv->plat->est->btr[0] = (u32)time.tv_nsec; | ||||
| 	priv->plat->est->btr[1] = (u32)time.tv_sec; | ||||
| 
 | ||||
|  | ||||
		Loading…
	
		Reference in New Issue
	
	Block a user
	 Yannick Vignon
						Yannick Vignon