mirror of
				git://git.kernel.org/pub/scm/linux/kernel/git/torvalds/linux.git
				synced 2025-09-04 20:19:47 +08:00 
			
		
		
		
	 c8c0ba4fe2
			
		
	
	
		c8c0ba4fe2
		
	
	
	
	
		
			
			Since the tag_8021q tagger is software-defined, it has no means by itself for retrieving hardware timestamps of PTP event messages. Because we do want to support PTP on ocelot even with tag_8021q, we need to use the CPU port module for that. The RX timestamp is present in the Extraction Frame Header. And because we can't use NPI mode which redirects the CPU queues to an "external CPU" (meaning the ARM CPU running Linux), then we need to poll the CPU port module through the MMIO registers to retrieve TX and RX timestamps. Sadly, on NXP LS1028A, the Felix switch was integrated into the SoC without wiring the extraction IRQ line to the ARM GIC. So, if we want to be notified of any PTP packets received on the CPU port module, we have a problem. There is a possible workaround, which is to use the Ethernet CPU port as a notification channel that packets are available on the CPU port module as well. When a PTP packet is received by the DSA tagger (without timestamp, of course), we go to the CPU extraction queues, poll for it there, then we drop the original Ethernet packet and masquerade the packet retrieved over MMIO (plus the timestamp) as the original when we inject it up the stack. Create a quirk in struct felix is selected by the Felix driver (but not by Seville, since that doesn't support PTP at all). We want to do this such that the workaround is minimally invasive for future switches that don't require this workaround. The only traffic for which we need timestamps is PTP traffic, so add a redirection rule to the CPU port module for this. Currently we only have the need for PTP over L2, so redirection rules for UDP ports 319 and 320 are TBD for now. Note that for the workaround of matching of PTP-over-Ethernet-port with PTP-over-MMIO queues to work properly, both channels need to be absolutely lossless. There are two parts to achieving that: - We keep flow control enabled on the tag_8021q CPU port - We put the DSA master interface in promiscuous mode, so it will never drop a PTP frame (for the profiles we are interested in, these are sent to the multicast MAC addresses of 01-80-c2-00-00-0e and 01-1b-19-00-00-00). Signed-off-by: Vladimir Oltean <vladimir.oltean@nxp.com> Reviewed-by: Florian Fainelli <f.fainelli@gmail.com> Signed-off-by: David S. Miller <davem@davemloft.net>
		
			
				
	
	
		
			71 lines
		
	
	
		
			2.3 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			71 lines
		
	
	
		
			2.3 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| /* SPDX-License-Identifier: GPL-2.0 */
 | |
| /* Copyright 2019 NXP Semiconductors
 | |
|  */
 | |
| #ifndef _MSCC_FELIX_H
 | |
| #define _MSCC_FELIX_H
 | |
| 
 | |
| #define ocelot_to_felix(o)		container_of((o), struct felix, ocelot)
 | |
| 
 | |
| /* Platform-specific information */
 | |
| struct felix_info {
 | |
| 	const struct resource		*target_io_res;
 | |
| 	const struct resource		*port_io_res;
 | |
| 	const struct resource		*imdio_res;
 | |
| 	const struct reg_field		*regfields;
 | |
| 	const u32 *const		*map;
 | |
| 	const struct ocelot_ops		*ops;
 | |
| 	int				num_mact_rows;
 | |
| 	const struct ocelot_stat_layout	*stats_layout;
 | |
| 	unsigned int			num_stats;
 | |
| 	int				num_ports;
 | |
| 	int				num_tx_queues;
 | |
| 	struct vcap_props		*vcap;
 | |
| 	int				switch_pci_bar;
 | |
| 	int				imdio_pci_bar;
 | |
| 	const struct ptp_clock_info	*ptp_caps;
 | |
| 
 | |
| 	/* Some Ocelot switches are integrated into the SoC without the
 | |
| 	 * extraction IRQ line connected to the ARM GIC. By enabling this
 | |
| 	 * workaround, the few packets that are delivered to the CPU port
 | |
| 	 * module (currently only PTP) are copied not only to the hardware CPU
 | |
| 	 * port module, but also to the 802.1Q Ethernet CPU port, and polling
 | |
| 	 * the extraction registers is triggered once the DSA tagger sees a PTP
 | |
| 	 * frame. The Ethernet frame is only used as a notification: it is
 | |
| 	 * dropped, and the original frame is extracted over MMIO and annotated
 | |
| 	 * with the RX timestamp.
 | |
| 	 */
 | |
| 	bool				quirk_no_xtr_irq;
 | |
| 
 | |
| 	int	(*mdio_bus_alloc)(struct ocelot *ocelot);
 | |
| 	void	(*mdio_bus_free)(struct ocelot *ocelot);
 | |
| 	void	(*phylink_validate)(struct ocelot *ocelot, int port,
 | |
| 				    unsigned long *supported,
 | |
| 				    struct phylink_link_state *state);
 | |
| 	int	(*prevalidate_phy_mode)(struct ocelot *ocelot, int port,
 | |
| 					phy_interface_t phy_mode);
 | |
| 	int	(*port_setup_tc)(struct dsa_switch *ds, int port,
 | |
| 				 enum tc_setup_type type, void *type_data);
 | |
| 	void	(*port_sched_speed_set)(struct ocelot *ocelot, int port,
 | |
| 					u32 speed);
 | |
| };
 | |
| 
 | |
| extern const struct dsa_switch_ops felix_switch_ops;
 | |
| 
 | |
| /* DSA glue / front-end for struct ocelot */
 | |
| struct felix {
 | |
| 	struct dsa_switch		*ds;
 | |
| 	const struct felix_info		*info;
 | |
| 	struct ocelot			ocelot;
 | |
| 	struct mii_bus			*imdio;
 | |
| 	struct lynx_pcs			**pcs;
 | |
| 	resource_size_t			switch_base;
 | |
| 	resource_size_t			imdio_base;
 | |
| 	struct dsa_8021q_context	*dsa_8021q_ctx;
 | |
| 	enum dsa_tag_protocol		tag_proto;
 | |
| };
 | |
| 
 | |
| struct net_device *felix_port_to_netdev(struct ocelot *ocelot, int port);
 | |
| int felix_netdev_to_port(struct net_device *dev);
 | |
| 
 | |
| #endif
 |