Re: [PATCH v2] can: rcar_canfd: Add Renesas R-Car CAN FD driver

[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index]

 




On 03/03/2016 04:38 PM, Ramesh Shanmugasundaram wrote:
> This patch adds support for the CAN FD controller found in Renesas R-Car
> SoCs. The controller operates in CAN FD mode by default.
> 
> CAN FD mode supports both Classical CAN & CAN FD frame formats. The
> controller supports ISO 11898-1:2015 CAN FD format only.
> 
> This controller supports two channels and the driver can enable either
> or both of the channels.
> 
> Driver uses Rx FIFOs (one per channel) for reception & Common FIFOs (one
> per channel) for transmission. Rx filter rules are configured to the
> minimum (one per channel) and it accepts Standard, Extended, Data &
> Remote Frame combinations.
> 
> Note: There are few documentation errors in R-Car Gen3 Hardware User
> Manual v0.5E with respect to CAN FD controller. They are listed below:
> 
> 1. CAN FD interrupt numbers 29 & 30 are listed as per channel
> interrupts. However, they are common to both channels (i.e.) they are
> global and channel interrupts respectively.
> 
> 2. CANFD clock is derived from PLL1. This is not documented.
> 
> 3. CANFD clock is further divided by (1/2) within the CAN FD controller.
> This is not documented.
> 
> 4. The minimum value of NTSEG1 in RSCFDnCFDCmNCFG register is 2 Tq. It
> is mentioned 4 Tq in the manual.
> 
> 5. The maximum number of message RAM area the controller can use is 3584
> bytes. It is specified 10752 bytes in the manual.
> 
> Signed-off-by: Ramesh Shanmugasundaram <ramesh.shanmugasundaram@xxxxxxxxxxxxxx>
> ---
> Changes since v1:
> 	* Removed testmodes & debugfs code (suggested by Oliver H)
> 	* Fixed tx path race issue by introducing lock (suggested by Marc K)
> 	* Removed __maybe_unused attribute of rcar_canfd_of_table
> 
> Thanks,
> Ramesh
> ---
>  .../devicetree/bindings/net/can/rcar_canfd.txt     |   86 ++
>  drivers/net/can/Kconfig                            |   10 +
>  drivers/net/can/Makefile                           |    1 +
>  drivers/net/can/rcar_canfd.c                       | 1624 ++++++++++++++++++++
>  4 files changed, 1721 insertions(+)
>  create mode 100644 Documentation/devicetree/bindings/net/can/rcar_canfd.txt
>  create mode 100644 drivers/net/can/rcar_canfd.c
> 
> diff --git a/Documentation/devicetree/bindings/net/can/rcar_canfd.txt b/Documentation/devicetree/bindings/net/can/rcar_canfd.txt
> new file mode 100644
> index 0000000..4299bd8
> --- /dev/null
> +++ b/Documentation/devicetree/bindings/net/can/rcar_canfd.txt
> @@ -0,0 +1,86 @@
> +Renesas R-Car CAN FD controller Device Tree Bindings
> +----------------------------------------------------
> +
> +Required properties:
> +- compatible: Must contain one or more of the following:
> +  - "renesas,rcar-gen3-canfd" for R-Car Gen3 compatible controller.
> +  - "renesas,r8a7795-canfd" for R8A7795 (R-Car H3) compatible controller.
> +
> +  When compatible with the generic version, nodes must list the
> +  SoC-specific version corresponding to the platform first, followed by the
> +  family-specific and/or generic versions.
> +
> +- reg: physical base address and size of the R-Car CAN FD register map.
> +- interrupts: interrupt specifier for the Global & Channel interrupts
> +- clocks: phandles and clock specifiers for 3 clock inputs.
> +- clock-names: 3 clock input name strings: "fck", "canfd", "can_clk".
> +- pinctrl-0: pin control group to be used for this controller.
> +- pinctrl-names: must be "default".
> +
> +Required properties for "renesas,r8a7795-canfd" compatible:
> +In R8A7795 SoC, canfd clock is a div6 clock and can be used by both CAN
> +and CAN FD controller at the same time. It needs to be scaled to maximum
> +frequency if any of these controllers use it. This is done using the
> +below properties.
> +
> +- assigned-clocks: phandle of canfd clock.
> +- assigned-clock-rates: maximum frequency of this clock.
> +
> +Each channel is represented as a child node. They can be enabled/disabled
> +using "status" property.
> +
> +Example
> +-------
> +
> +SoC common .dtsi file:
> +
> +		canfd: canfd@e66c0000 {
> +			compatible = "renesas,r8a7795-canfd",
> +				     "renesas,rcar-gen3-canfd";
> +			reg = <0 0xe66c0000 0 0x8000>;
> +			interrupts = <GIC_SPI 29 IRQ_TYPE_LEVEL_HIGH>,
> +				   <GIC_SPI 30 IRQ_TYPE_LEVEL_HIGH>;
> +			clocks = <&cpg CPG_MOD 914>,
> +			       <&cpg CPG_CORE R8A7795_CLK_CANFD>,
> +			       <&can_clk>;
> +			clock-names = "fck", "canfd", "can_clk";
> +			assigned-clocks = <&cpg CPG_CORE R8A7795_CLK_CANFD>;
> +			assigned-clock-rates = <40000000>;
> +			power-domains = <&cpg>;
> +			status = "disabled";
> +
> +			channel0 {
> +				status = "disabled";
> +			};
> +
> +			channel1 {
> +				status = "disabled";
> +			};
> +		};
> +
> +Board specific .dts file:
> +
> +E.g. below enables Channel 1 alone in the board.
> +
> +&canfd {
> +        pinctrl-0 = <&canfd1_pins>;
> +        pinctrl-names = "default";
> +        status = "okay";
> +
> +	channel1 {
> +		status = "okay";
> +	};
> +};
> +
> +E.g. below enables Channel 0 alone in the board using External clock
> +as fCAN clock.
> +
> +&canfd {
> +        pinctrl-0 = <&canfd0_pins &can_clk_pins>;
> +        pinctrl-names = "default";
> +        status = "okay";
> +
> +	channel0 {
> +		status = "okay";
> +	};
> +};
> diff --git a/drivers/net/can/Kconfig b/drivers/net/can/Kconfig
> index 0d40aef..0ecb4fe 100644
> --- a/drivers/net/can/Kconfig
> +++ b/drivers/net/can/Kconfig
> @@ -114,6 +114,16 @@ config CAN_RCAR
>  	  To compile this driver as a module, choose M here: the module will
>  	  be called rcar_can.
>  
> +config CANFD_RCAR
> +	tristate "Renesas R-Car CAN FD controller"
> +	depends on ARCH_RENESAS || ARM
> +	---help---
> +	  Say Y here if you want to use CAN FD controller found on
> +	  Renesas R-Car SoCs.
> +
> +	  To compile this driver as a module, choose M here: the module will
> +	  be called rcar_canfd.
> +
>  config CAN_SUN4I
>  	tristate "Allwinner A10 CAN controller"
>  	depends on MACH_SUN4I || MACH_SUN7I || COMPILE_TEST
> diff --git a/drivers/net/can/Makefile b/drivers/net/can/Makefile
> index e3db0c8..401b150 100644
> --- a/drivers/net/can/Makefile
> +++ b/drivers/net/can/Makefile
> @@ -25,6 +25,7 @@ obj-$(CONFIG_CAN_JANZ_ICAN3)	+= janz-ican3.o
>  obj-$(CONFIG_CAN_MSCAN)		+= mscan/
>  obj-$(CONFIG_CAN_M_CAN)		+= m_can/
>  obj-$(CONFIG_CAN_RCAR)		+= rcar_can.o
> +obj-$(CONFIG_CANFD_RCAR)	+= rcar_canfd.o
>  obj-$(CONFIG_CAN_SJA1000)	+= sja1000/
>  obj-$(CONFIG_CAN_SUN4I)		+= sun4i_can.o
>  obj-$(CONFIG_CAN_TI_HECC)	+= ti_hecc.o
> diff --git a/drivers/net/can/rcar_canfd.c b/drivers/net/can/rcar_canfd.c
> new file mode 100644
> index 0000000..56e089d
> --- /dev/null
> +++ b/drivers/net/can/rcar_canfd.c
> @@ -0,0 +1,1624 @@
> +/* Renesas R-Car CAN FD device driver
> + *
> + * Copyright (C) 2015 Renesas Electronics Corp.
> + *
> + * This program is free software; you can redistribute  it and/or modify it
> + * under  the terms of  the GNU General  Public License as published by the
> + * Free Software Foundation;  either version 2 of the  License, or (at your
> + * option) any later version.
> + */
> +
> +#include <linux/module.h>
> +#include <linux/moduleparam.h>
> +#include <linux/kernel.h>
> +#include <linux/types.h>
> +#include <linux/interrupt.h>
> +#include <linux/errno.h>
> +#include <linux/netdevice.h>
> +#include <linux/platform_device.h>
> +#include <linux/can/led.h>
> +#include <linux/can/dev.h>
> +#include <linux/clk.h>
> +#include <linux/of.h>
> +#include <linux/of_device.h>
> +#include <linux/bitmap.h>
> +#include <linux/bitops.h>
> +#include <linux/iopoll.h>
> +
> +#define RCANFD_DRV_NAME			"rcar_canfd"
> +
> +#define RCANFD_FIFO_DEPTH		8	/* Tx FIFO depth */
> +#define RCANFD_NAPI_WEIGHT		8	/* Rx poll quota */
> +
> +#define RCANFD_NUM_CHANNELS		2
> +#define RCANFD_CHANNELS_MASK		0x3	/* Two channels max */

(BIT(RCANFD_NUM_CHANNELS) - 1

> +
> +/* Rx FIFO is a global resource of the controller. There are 8 such FIFOs
> + * available. Each channel gets a dedicated Rx FIFO (i.e.) the channel
> + * number is added to RFFIFO index.
> + */
> +#define RCANFD_RFFIFO_IDX		0
> +
> +/* Tx/Rx or Common FIFO is a per channel resource. Each channel has 3 Common
> + * FIFOs dedicated to them. Use the first (index 0) FIFO out of the 3 for Tx.
> + */
> +#define RCANFD_CFFIFO_IDX		0
> +
> +/* Global register bits */
> +#define RCANFD_GINTF_CANFD		BIT(0)
> +
> +#define RCANFD_GCFG_TPRI		BIT(0)
> +#define RCANFD_GCFG_DCE			BIT(1)
> +#define RCANFD_GCFG_DCS			BIT(4)
> +#define RCANFD_GCFG_CMPOC		BIT(5)
> +#define RCANFD_GCFG_EEFE		BIT(6)
> +
> +#define RCANFD_GCTR_SLPR		BIT(2)
> +#define RCANFD_GCTR_MODEMASK		(0x3)
> +#define RCANFD_GCTR_GOPM		(0x0)
> +#define RCANFD_GCTR_GRESET		(0x1)
> +#define RCANFD_GCTR_GHLT		(0x2)
> +
> +#define RCANFD_GCTR_DEIE		BIT(8)
> +#define RCANFD_GCTR_MEIE		BIT(9)
> +#define RCANFD_GCTR_THLEIE		BIT(10)
> +#define RCANFD_GCTR_CFMPOFIE		BIT(11)
> +#define RCANFD_GCTR_TSRST		BIT(16)
> +
> +#define RCANFD_GSTS_RAMINIT		BIT(3)
> +#define RCANFD_GSTS_SLP			BIT(2)
> +#define RCANFD_GSTS_HLT			BIT(1)
> +#define RCANFD_GSTS_RESET		BIT(0)
> +
> +#define RCANFD_GSTS_GNOPM		(BIT(0) | BIT(1) | BIT(2) | BIT(3))
> +
> +/* Channel register bits */
> +#define RCANFD_CCTR_CSLPR		BIT(2)
> +#define RCANFD_CCTR_MODEMASK		(0x3)
> +#define RCANFD_CCTR_COPM		(0x0)
> +#define RCANFD_CCTR_CRESET		(0x1)
> +#define RCANFD_CCTR_CHLT		(0x2)
> +#define RCANFD_CCTR_CTMASK		(0x3 << 25)
> +#define RCANFD_CCTR_CTMS_ILB		(0x3 << 25)
> +#define RCANFD_CCTR_CTME		BIT(24)
> +#define RCANFD_CCTR_ERRD		BIT(23)
> +#define RCANFD_CCTR_BOMMASK		(0x3 << 21)
> +#define RCANFD_CCTR_BOM_ENTRY		(0x1 << 21)
> +#define RCANFD_CCTR_TDCVFIE		BIT(19)
> +#define RCANFD_CCTR_SOCOIE		BIT(18)
> +#define RCANFD_CCTR_EOCOIE		BIT(17)
> +#define RCANFD_CCTR_TAIE		BIT(16)
> +#define RCANFD_CCTR_ALIE		BIT(15)
> +#define RCANFD_CCTR_BLIE		BIT(14)
> +#define RCANFD_CCTR_OLIE		BIT(13)
> +#define RCANFD_CCTR_BORIE		BIT(12)
> +#define RCANFD_CCTR_BOEIE		BIT(11)
> +#define RCANFD_CCTR_EPIE		BIT(10)
> +#define RCANFD_CCTR_EWIE		BIT(9)
> +#define RCANFD_CCTR_BEIE		BIT(8)
> +
> +#define RCANFD_CSTS_COM			BIT(7)
> +#define RCANFD_CSTS_REC			BIT(6)
> +#define RCANFD_CSTS_TRM			BIT(5)
> +#define RCANFD_CSTS_BO			BIT(4)
> +#define RCANFD_CSTS_EP			BIT(3)
> +#define RCANFD_CSTS_SLP			BIT(2)
> +#define RCANFD_CSTS_HALT		BIT(1)
> +#define RCANFD_CSTS_RESET		BIT(0)
> +
> +#define RCANFD_CSTS_TECCNT(x)		(((x) >> 24) & 0xff)
> +#define RCANFD_CSTS_RECCNT(x)		(((x) >> 16) & 0xff)
> +
> +/* Bit Configuration register */
> +#define RCANFD_BRP(x)			(((x) & 0x3ff) << 0)
> +#define RCANFD_SJW(x)			(((x) & 0x3) << 24)
> +#define RCANFD_TSEG1(x)			(((x) & 0xf) << 16)
> +#define RCANFD_TSEG2(x)			(((x) & 0x7) << 20)
> +
> +#define RCANFD_NR_BRP(x)		(((x) & 0x3ff) << 0)
> +#define RCANFD_NR_SJW(x)		(((x) & 0x1f) << 11)
> +#define RCANFD_NR_TSEG1(x)		(((x) & 0x7f) << 16)
> +#define RCANFD_NR_TSEG2(x)		(((x) & 0x1f) << 24)
> +
> +#define RCANFD_DR_BRP(x)		(((x) & 0xff) << 0)
> +#define RCANFD_DR_SJW(x)		(((x) & 0x7) << 24)
> +#define RCANFD_DR_TSEG1(x)		(((x) & 0xf) << 16)
> +#define RCANFD_DR_TSEG2(x)		(((x) & 0x7) << 20)
> +
> +/* Global Error flag bits */
> +#define RCANFD_GERFL_EEF1		BIT(17)
> +#define RCANFD_GERFL_EEF0		BIT(16)
> +#define RCANFD_GERFL_CMPOF		BIT(3)
> +#define RCANFD_GERFL_THLES		BIT(2)
> +#define RCANFD_GERFL_MES		BIT(1)
> +#define RCANFD_GERFL_DEF		BIT(0)
> +
> +#define RCANFD_GERFL_ERR(x)		((x) & (RCANFD_GERFL_EEF1 |\
> +						RCANFD_GERFL_EEF0 |\
> +						RCANFD_GERFL_MES |\
> +						RCANFD_GERFL_CMPOF))
> +
> +/* Channel Error flag bits */
> +#define RCANFD_CERFL_ADEF		BIT(14)
> +#define RCANFD_CERFL_B0EF		BIT(13)
> +#define RCANFD_CERFL_B1EF		BIT(12)
> +#define RCANFD_CERFL_CEF		BIT(11)
> +#define RCANFD_CERFL_AEF		BIT(10)
> +#define RCANFD_CERFL_FEF		BIT(9)
> +#define RCANFD_CERFL_SEF		BIT(8)
> +#define RCANFD_CERFL_ALEF		BIT(7)
> +#define RCANFD_CERFL_BLEF		BIT(6)
> +#define RCANFD_CERFL_OLEF		BIT(5)
> +#define RCANFD_CERFL_BOREF		BIT(4)
> +#define RCANFD_CERFL_BOEEF		BIT(3)
> +#define RCANFD_CERFL_EPEF		BIT(2)
> +#define RCANFD_CERFL_EWEF		BIT(1)
> +#define RCANFD_CERFL_BEF		BIT(0)
> +
> +#define RCANFD_CERFL_ERR(x)		((x) & (0x7fff)) /* above bits 14:0 */
> +
> +/* CAN FD specific registers */
> +#define RCANFD_DCFG_TDCE		BIT(9)
> +#define RCANFD_DCFG_TDCOC		BIT(8)
> +#define RCANFD_DCFG_TDCO(x)		(((x) & 0x7f) >> 16)
> +
> +#define RCANFD_DCSTS_TDCR(x)		(((x) >> 0) & 0x7f)
> +#define RCANFD_DCSTS_SOCCNT(x)		(((x) >> 24) & 0xff)
> +#define RCANFD_DCSTS_EOCCNT(x)		(((x) >> 16) & 0xff)
> +
> +#define RCANFD_DCSTS_TDCVF		BIT(7)
> +#define RCANFD_DCSTS_EOCO		BIT(8)
> +#define RCANFD_DCSTS_SOCO		BIT(9)
> +
> +/* Rx FIFO bits */
> +#define RCANFD_RFFIFO_RFIF		BIT(3)
> +#define RCANFD_RFFIFO_RFMLT		BIT(2)
> +#define RCANFD_RFFIFO_RFFLL		BIT(1)
> +#define RCANFD_RFFIFO_RFEMP		BIT(0)
> +
> +#define RCANFD_RFFIFO_RFESI		BIT(0)
> +#define RCANFD_RFFIFO_RFBRS		BIT(1)
> +#define RCANFD_RFFIFO_RFFDF		BIT(2)
> +
> +#define RCANFD_RFFIFO_RFIDE		BIT(31)
> +#define RCANFD_RFFIFO_RFRTR		BIT(30)
> +
> +#define RCANFD_RFFIFO_RFDLC(x)		(((x) >> 28) & 0xf)
> +#define RCANFD_RFFIFO_RFPTR(x)		(((x) >> 16) & 0xfff)
> +#define RCANFD_RFFIFO_RFTS(x)		(((x) >> 0) & 0xff)
> +
> +#define RCANFD_RFFIFO_RFIM		BIT(12)
> +#define RCANFD_RFFIFO_RFDC(x)		(((x) & 0x7) << 8)
> +#define RCANFD_RFFIFO_RFPLS(x)		(((x) & 0x7) << 4)
> +#define RCANFD_RFFIFO_RFIE		BIT(1)
> +#define RCANFD_RFFIFO_RFE		BIT(0)
> +
> +/* Common FIFO bits */
> +#define RCANFD_CMFIFO_TML(x)		(((x) & 0xf) << 20)
> +#define RCANFD_CMFIFO_M(x)		(((x) & 0x3) << 16)
> +#define RCANFD_CMFIFO_CFIM		BIT(12)
> +#define RCANFD_CMFIFO_DC(x)		(((x) & 0x7) << 8)
> +#define RCANFD_CMFIFO_PLS(x)		(((x) & 0x7) << 4)
> +#define RCANFD_CMFIFO_CFTXIE		BIT(2)
> +#define RCANFD_CMFIFO_CFE		BIT(0)
> +
> +#define RCANFD_CMFIFO_CFTXIF		BIT(4)
> +#define RCANFD_CMFIFO_CFMLT		BIT(2)
> +#define RCANFD_CMFIFO_CFFLL		BIT(1)
> +#define RCANFD_CMFIFO_CFEMP		BIT(0)
> +#define RCANFD_CMFIFO_CFMC(x)		(((x) >> 8) & 0xff)
> +
> +#define RCANFD_CMFIFO_CFESI		BIT(0)
> +#define RCANFD_CMFIFO_CFBRS		BIT(1)
> +#define RCANFD_CMFIFO_CFFDF		BIT(2)
> +
> +#define RCANFD_CMFIFO_CFIDE		BIT(31)
> +#define RCANFD_CMFIFO_CFRTR		BIT(30)
> +#define RCANFD_CMFIFO_CFID(x)		((x) & 0x1fffffff)
> +
> +#define RCANFD_CMFIFO_CFDLC(x)		(((x) & 0xf) << 28)
> +#define RCANFD_CMFIFO_CFPTR(x)		(((x) & 0xfff) << 16)
> +#define RCANFD_CMFIFO_CFTS(x)		(((x) & 0xff) << 0)
> +
> +/* Global Test Config register */
> +#define RCANFD_GTSTCFG_C0CBCE		BIT(0)
> +#define RCANFD_GTSTCFG_C1CBCE		BIT(1)
> +
> +#define RCANFD_GTSTCTR_ICBCTME		BIT(0)
> +
> +/* AFL Rx rules registers */
> +#define RCANFD_AFLCFG_SETRNC0(x)	(((x) & 0xff) << 24)
> +#define RCANFD_AFLCFG_SETRNC1(x)	(((x) & 0xff) << 16)

What about something like:

#define RCANFD_AFLCFG_SETRNC(n, x)	(((x) & 0xff) << (24 - n * 8))

This will save some if()s in the code

> +#define RCANFD_AFLCFG_GETRNC0(x)	(((x) >> 24) & 0xff)
> +#define RCANFD_AFLCFG_GETRNC1(x)	(((x) >> 16) & 0xff)
> +
> +#define RCANFD_AFL_PAGENUM(entry)	((entry) / 16)
> +
> +#define RCANFD_AFLCTR_AFLDAE		BIT(8)
> +#define RCANFD_AFLCTR_AFLPN(x)		((x) & 0x1f)
> +#define RCANFD_CHANNEL_NUMRULES		1	/* only one rule per channel */
> +#define RCANFD_AFLID_GAFLLB		BIT(29)
> +#define RCANFD_AFLPTR1_RFFIFO(x)	(1 << (x))
> +
> +/* Common register map offsets */
> +
> +/* Nominal rate registers */
> +#define RCANFD_CCFG(m)			(0x0000 + (0x10 * (m)))
> +#define RCANFD_CCTR(m)			(0x0004 + (0x10 * (m)))
> +#define RCANFD_CSTS(m)			(0x0008 + (0x10 * (m)))
> +#define RCANFD_CERFL(m)			(0x000C + (0x10 * (m)))
> +
> +/* Global registers */
> +#define RCANFD_GCFG			(0x0084)
> +#define RCANFD_GCTR			(0x0088)
> +#define RCANFD_GSTS			(0x008c)
> +#define RCANFD_GERFL			(0x0090)
> +#define RCANFD_GTSC			(0x0094)
> +#define RCANFD_GAFLECTR			(0x0098)
> +#define RCANFD_GAFLCFG0			(0x009c)
> +#define RCANFD_GAFLCFG1			(0x00a0)
> +#define RCANFD_RMNB			(0x00a4)
> +
> +#define RCANFD_RMND(y)			(0x00a8 + (0x04 * (y)))
> +
> +/* Rx FIFO Control registers */
> +#define RCANFD_RFCC(x)			(0x00b8 + (0x04 * (x)))
> +#define RCANFD_RFSTS(x)			(0x00d8 + (0x04 * (x)))
> +#define RCANFD_RFPCTR(x)		(0x00f8 + (0x04 * (x)))
> +
> +/* Common FIFO Control registers */
> +#define RCANFD_CFCC(ch, idx)		(0x0118 + (0x0c * (ch)) + \
> +					 (0x04 * (idx)))
> +#define RCANFD_CFSTS(ch, idx)		(0x0178 + (0x0c * (ch)) + \
> +					 (0x04 * (idx)))
> +#define RCANFD_CFPCTR(ch, idx)		(0x01d8 + (0x0c * (ch)) + \
> +					 (0x04 * (idx)))
> +
> +#define RCANFD_FESTS			(0x0238)
> +#define RCANFD_FFSTS			(0x023c)
> +#define RCANFD_FMSTS			(0x0240)
> +#define RCANFD_RFISTS			(0x0244)
> +#define RCANFD_CFRISTS			(0x0248)
> +#define RCANFD_CFTISTS			(0x024c)
> +
> +#define RCANFD_TMC(p)			(0x0250 + (0x01 * (p)))
> +#define RCANFD_TMSTS(p)			(0x02d0 + (0x01 * (p)))
> +
> +#define RCANFD_TMTRSTS(y)		(0x0350 + (0x04 * (y)))
> +#define RCANFD_TMTARSTS(y)		(0x0360 + (0x04 * (y)))
> +#define RCANFD_TMTCSTS(y)		(0x0370 + (0x04 * (y)))
> +#define RCANFD_TMTASTS(y)		(0x0380 + (0x04 * (y)))
> +#define RCANFD_TMIEC(y)			(0x0390 + (0x04 * (y)))
> +
> +#define RCANFD_TXQCC(m)			(0x03a0 + (0x04 * (m)))
> +#define RCANFD_TXQSTS(m)		(0x03c0 + (0x04 * (m)))
> +#define RCANFD_TXQPCTR(m)		(0x03e0 + (0x04 * (m)))
> +
> +#define RCANFD_THLCC(m)			(0x0400 + (0x04 * (m)))
> +#define RCANFD_THLSTS(m)		(0x0420 + (0x04 * (m)))
> +#define RCANFD_THLPCTR(m)		(0x0440 + (0x04 * (m)))
> +
> +#define RCANFD_GTINTSTS0		(0x0460)
> +#define RCANFD_GTINTSTS1		(0x0464)
> +#define RCANFD_GTSTCFG			(0x0468)
> +#define RCANFD_GTSTCTR			(0x046c)
> +#define RCANFD_GLOCKK			(0x047c)
> +#define RCANFD_GRMCFG			(0x04fc)
> +
> +/* Receive rule registers */
> +#define RCANFD_GAFLID(offset, j)	((offset) + (0x10 * (j)))
> +#define RCANFD_GAFLM(offset, j)		((offset) + 0x04 + (0x10 * (j)))
> +#define RCANFD_GAFLP0(offset, j)	((offset) + 0x08 + (0x10 * (j)))
> +#define RCANFD_GAFLP1(offset, j)	((offset) + 0x0c + (0x10 * (j)))
> +
> +/* CAN FD mode specific regsiter map */
> +
> +/* Data bitrate registers */
> +#define RCANFD_F_CDFG(m)		(0x0500 + (0x20 * (m)))
> +#define RCANFD_F_CFDCFG(m)		(0x0504 + (0x20 * (m)))
> +#define RCANFD_F_CFDCTR(m)		(0x0508 + (0x20 * (m)))
> +#define RCANFD_F_CFDSTS(m)		(0x050c + (0x20 * (m)))
> +#define RCANFD_F_CFDCRC(m)		(0x0510 + (0x20 * (m)))
> +
> +#define RCANFD_F_GAFL_OFFSET		(0x1000)
> +
> +#define RCANFD_F_RMID(q)		(0x2000 + (0x10 * (q)))
> +#define RCANFD_F_RMPTR(q)		(0x2004 + (0x10 * (q)))
> +#define RCANFD_F_RMFDSTS(q)		(0x2008 + (0x10 * (q)))
> +#define RCANFD_F_RMDF(q, b)		(0x200c + (0x04 * (b)) + (0x20 * (q)))
> +
> +/* Rx FIFO data registers */
> +#define RCANFD_F_RFOFFSET		(0x3000)
> +#define RCANFD_F_RFID(x)		(RCANFD_F_RFOFFSET + (0x80 * (x)))
> +#define RCANFD_F_RFPTR(x)		(RCANFD_F_RFOFFSET + 0x04 + \
> +					 (0x80 * (x)))
> +#define RCANFD_F_RFFDSTS(x)		(RCANFD_F_RFOFFSET + 0x08 + \
> +					 (0x80 * (x)))
> +#define RCANFD_F_RFDF(x, df)		(RCANFD_F_RFOFFSET + 0x0c + \
> +					 (0x80 * (x)) + (0x04 * (df)))
> +
> +/* Common FIFO data registers */
> +#define RCANFD_F_CFOFFSET		(0x3400)
> +#define RCANFD_F_CFID(ch, idx)		(RCANFD_F_CFOFFSET + (0x180 * (ch)) + \
> +					 (0x80 * (idx)))
> +#define RCANFD_F_CFPTR(ch, idx)		(RCANFD_F_CFOFFSET + 0x04 + \
> +					 (0x180 * (ch)) + (0x80 * (idx)))
> +#define RCANFD_F_CFFDCSTS(ch, idx)	(RCANFD_F_CFOFFSET + 0x08 + \
> +					 (0x180 * (ch)) + (0x80 * (idx)))
> +#define RCANFD_F_CFDF(ch, idx, df)	(RCANFD_F_CFOFFSET + 0x0c + \
> +					 (0x180 * (ch)) + (0x80 * (idx)) + \
> +					 (0x04 * (df)))
> +
> +#define RCANFD_F_TMID(p)		(0x4000 + (0x20 * (p)))
> +#define RCANFD_F_TMPTR(p)		(0x4004 + (0x20 * (p)))
> +#define RCANFD_F_TMFDCTR(p)		(0x4008 + (0x20 * (p)))
> +#define RCANFD_F_TMDF(p, b)		(0x400c + (0x20 * (p)) + (0x04 * (b)))
> +
> +#define RCANFD_F_THLACC(m)		(0x6000 + (0x04 * (m)))
> +#define RCANFD_F_RPGACC(r)		(0x6400 + (0x04 * (r)))
> +
> +struct rcar_canfd_global;
> +
> +/* Channel priv data */
> +struct rcar_canfd_channel {
> +	struct can_priv can;			/* Must be the first member */
> +	struct net_device *ndev;
> +	struct rcar_canfd_global *gpriv;	/* Controller reference */
> +	void __iomem *base;			/* Register base address */
> +#ifdef CONFIG_DEBUG_FS
> +	struct dentry *dev_root;
> +#endif /* CONFIG_DEBUG_FS */
> +	struct napi_struct napi;
> +	u8  tx_len[RCANFD_FIFO_DEPTH];		/* For net stats */
> +	u32 tx_head;				/* Incremented on xmit */
> +	u32 tx_tail;				/* Incremented on xmit done */
> +	u32 channel;				/* Channel number */
> +	spinlock_t tx_lock;			/* To protect tx path */
> +};
> +
> +/* Global priv data */
> +struct rcar_canfd_global {
> +	struct rcar_canfd_channel *ch[RCANFD_NUM_CHANNELS];
> +	void __iomem *base;		/* Register base address */
> +	struct platform_device *pdev;	/* Respective platform device */
> +	struct clk *clkp;		/* Peripheral clock */
> +	struct clk *can_clk;		/* fCAN clock */
> +	int clock_select;		/* CANFD or Ext clock */
        ^^^
please give the corresponding enum a proper name and use it here.

> +	unsigned long channels_mask;	/* Enabled channels mask */
> +	u32 freq;			/* fCAN clock frequency in Hz */

This value is not used outside of the probe function. You can pass the
freq to the individual channels via an argument.

> +};
> +
> +/* CAN FD mode nominal rate constants */
> +static const struct can_bittiming_const rcar_canfd_nom_bittiming_const = {
> +	.name = RCANFD_DRV_NAME,
> +	.tseg1_min = 2,
> +	.tseg1_max = 128,
> +	.tseg2_min = 2,
> +	.tseg2_max = 32,
> +	.sjw_max = 32,
> +	.brp_min = 1,
> +	.brp_max = 1024,
> +	.brp_inc = 1,
> +};
> +
> +/* CAN FD mode data rate constants */
> +static const struct can_bittiming_const rcar_canfd_data_bittiming_const = {
> +	.name = RCANFD_DRV_NAME,
> +	.tseg1_min = 2,
> +	.tseg1_max = 16,
> +	.tseg2_min = 2,
> +	.tseg2_max = 8,
> +	.sjw_max = 8,
> +	.brp_min = 1,
> +	.brp_max = 256,
> +	.brp_inc = 1,
> +};
> +
> +/* fCAN clock select register settings */
> +enum {
> +	RCANFD_CANFDCLK = 0,	/* CANFD clock */
> +	RCANFD_EXTCLK,		/* Externally input clock */
> +};
> +
> +/* Helper functions */
> +static inline void rcar_canfd_update(u32 mask, u32 val, u32 __iomem *reg)
> +{
> +	u32 data = readl(reg);
> +
> +	data &= ~mask;
> +	data |= (val & mask);
> +	writel(data, reg);
> +}
> +
> +#define rcar_canfd_read(priv, offset)			\
> +	readl(priv->base + (offset))
> +#define rcar_canfd_write(priv, offset, val)		\
> +	writel(val, priv->base + (offset))
> +#define rcar_canfd_set_bit(priv, reg, val)		\
> +	rcar_canfd_update(val, val, priv->base + (reg))
> +#define rcar_canfd_clear_bit(priv, reg, val)		\
> +	rcar_canfd_update(val, 0, priv->base + (reg))
> +#define rcar_canfd_update_bit(priv, reg, mask, val)	\
> +	rcar_canfd_update(mask, val, priv->base + (reg))

please use static inline functions instead of defines.

> +
> +static void rcar_canfd_get_data(struct canfd_frame *cf,
> +				struct rcar_canfd_channel *priv, u32 off)

Please use "struct rcar_canfd_channel *priv" as first argument, struct
canfd_frame *cf as second. Remove off, as the offset is already defined
by the channel.

> +{
> +	u32 i, lwords;
> +
> +	lwords = cf->len / sizeof(u32);
> +	if (cf->len % sizeof(u32))
> +		lwords++;

Use DIV_ROUND_UP() instead of open coding it.

> +	for (i = 0; i < lwords; i++)
> +		*((u32 *)cf->data + i) =
> +			rcar_canfd_read(priv, off + (i * sizeof(u32)));
> +}
> +
> +static void rcar_canfd_put_data(struct canfd_frame *cf,
> +				struct rcar_canfd_channel *priv, u32 off)

same here

> +{
> +	u32 i, j, lwords, leftover;
> +	u32 data = 0;
> +
> +	lwords = cf->len / sizeof(u32);
> +	leftover = cf->len % sizeof(u32);
> +	for (i = 0; i < lwords; i++)
> +		rcar_canfd_write(priv, off + (i * sizeof(u32)),
> +				 *((u32 *)cf->data + i));

Here you don't convert the endianess...
> +
> +	if (leftover) {
> +		u8 *p = (u8 *)((u32 *)cf->data + i);
> +
> +		for (j = 0; j < leftover; j++)
> +			data |= p[j] << (j * 8);

...here you do an implicit endianess conversion. "data" is little
endian, while p[j] is big endian.

> +		rcar_canfd_write(priv, off + (i * sizeof(u32)), data);
> +	}

Have you tested to send CAN frames with len != n*4 against a different
controller?

> +}
> +
> +static void rcar_canfd_tx_failure_cleanup(struct net_device *ndev)
> +{
> +	u32 i;
> +
> +	for (i = 0; i < RCANFD_FIFO_DEPTH; i++)
> +		can_free_echo_skb(ndev, i);
> +}
> +
> +static int rcar_canfd_reset_controller(struct rcar_canfd_global *gpriv)
> +{
> +	u32 sts, ch;
> +	int err;
> +
> +	/* Check RAMINIT flag as CAN RAM initialization takes place
> +	 * after the MCU reset
> +	 */
> +	err = readl_poll_timeout((gpriv->base + RCANFD_GSTS), sts,
> +				 !(sts & RCANFD_GSTS_RAMINIT), 2, 500000);
> +	if (err) {
> +		dev_dbg(&gpriv->pdev->dev, "global raminit failed\n");
> +		return err;
> +	}
> +
> +	/* Transition to Global Reset mode */
> +	rcar_canfd_clear_bit(gpriv, RCANFD_GCTR, RCANFD_GCTR_SLPR);
> +	rcar_canfd_update_bit(gpriv, RCANFD_GCTR,
> +			      RCANFD_GCTR_MODEMASK, RCANFD_GCTR_GRESET);
> +
> +	/* Ensure Global reset mode */
> +	err = readl_poll_timeout((gpriv->base + RCANFD_GSTS), sts,
> +				 (sts & RCANFD_GSTS_RESET), 2, 500000);
> +	if (err) {
> +		dev_dbg(&gpriv->pdev->dev, "global reset failed\n");
> +		return err;
> +	}
> +
> +	/* Reset Global error flags */
> +	rcar_canfd_write(gpriv, RCANFD_GERFL, 0x0);
> +
> +	/* Set the controller into FD mode */
> +	rcar_canfd_set_bit(gpriv, RCANFD_GRMCFG, RCANFD_GINTF_CANFD);
> +
> +	/* Transition all Channels to reset mode */
> +	for_each_set_bit(ch, &gpriv->channels_mask, RCANFD_NUM_CHANNELS) {
> +		rcar_canfd_clear_bit(gpriv, RCANFD_CCTR(ch), RCANFD_CCTR_CSLPR);
> +
> +		rcar_canfd_update_bit(gpriv, RCANFD_CCTR(ch),
> +				      RCANFD_CCTR_MODEMASK,
> +				      RCANFD_CCTR_CRESET);
> +
> +		/* Ensure Channel reset mode */
> +		err = readl_poll_timeout((gpriv->base + RCANFD_CSTS(ch)), sts,
> +					 (sts & RCANFD_CSTS_RESET), 2, 500000);
> +		if (err) {
> +			dev_dbg(&gpriv->pdev->dev,
> +				"channel %u reset failed\n", ch);
> +			return err;
> +		}
> +	}
> +	return 0;
> +}
> +
> +static void rcar_canfd_configure_controller(struct rcar_canfd_global *gpriv)
> +{
> +	u32 cfg, ch;
> +
> +	/* Global Configuration settings */
> +	cfg = RCANFD_GCFG_EEFE;		/* ECC error flag enabled */
> +
> +	/* Set External Clock if selected */
> +	if (gpriv->clock_select != RCANFD_CANFDCLK)
> +		cfg |= RCANFD_GCFG_DCS;
> +
> +	/* Truncate payload to configured message size RFPLS */
> +	cfg |= RCANFD_GCFG_CMPOC;
> +
> +	rcar_canfd_set_bit(gpriv, RCANFD_GCFG, cfg);
> +
> +	/* Channel configuration settings */
> +	for_each_set_bit(ch, &gpriv->channels_mask, RCANFD_NUM_CHANNELS) {
> +		rcar_canfd_set_bit(gpriv, RCANFD_CCTR(ch), RCANFD_CCTR_ERRD);
> +		rcar_canfd_update_bit(gpriv, RCANFD_CCTR(ch),
> +				      RCANFD_CCTR_BOMMASK,
> +				      RCANFD_CCTR_BOM_ENTRY);
> +	}
> +}
> +
> +static void rcar_canfd_configure_afl_rules(struct rcar_canfd_global *gpriv,
> +					   u32 ch)
> +{
> +	u32 cfg;
> +	int start, page, num_rules = RCANFD_CHANNEL_NUMRULES;
> +	u32 ridx = ch + RCANFD_RFFIFO_IDX;
> +
> +	if (ch == 0) {
> +		start = 0; /* Start from 0th rule */
> +	} else {
> +		/* Get number of existing rules and adjust */
> +		cfg = rcar_canfd_read(gpriv, RCANFD_GAFLCFG0);
> +		start = RCANFD_AFLCFG_GETRNC0(cfg);
> +	}
> +
> +	/* Enable write access to entry */
> +	page = RCANFD_AFL_PAGENUM(start);
> +	rcar_canfd_set_bit(gpriv, RCANFD_GAFLECTR,
> +			   (RCANFD_AFLCTR_AFLPN(page) | RCANFD_AFLCTR_AFLDAE));
> +
> +	/* Write number of rules for channel */
> +	if (ch == 0)
> +		rcar_canfd_set_bit(gpriv, RCANFD_GAFLCFG0,
> +				   RCANFD_AFLCFG_SETRNC0(num_rules));
> +	else
> +		rcar_canfd_set_bit(gpriv, RCANFD_GAFLCFG0,
> +				   RCANFD_AFLCFG_SETRNC1(num_rules));
> +
> +	/* Accept all IDs */
> +	rcar_canfd_write(gpriv, RCANFD_GAFLID(RCANFD_F_GAFL_OFFSET, start), 0);
> +	/* IDE or RTR is not considered for matching */
> +	rcar_canfd_write(gpriv, RCANFD_GAFLM(RCANFD_F_GAFL_OFFSET, start), 0);
> +	/* Any data length accepted */
> +	rcar_canfd_write(gpriv, RCANFD_GAFLP0(RCANFD_F_GAFL_OFFSET, start), 0);
> +	/* Place the msg in corresponding Rx FIFO entry */
> +	rcar_canfd_write(gpriv, RCANFD_GAFLP1(RCANFD_F_GAFL_OFFSET, start),
> +			 RCANFD_AFLPTR1_RFFIFO(ridx));
> +
> +	/* Disable write access to page */
> +	rcar_canfd_clear_bit(gpriv, RCANFD_GAFLECTR, RCANFD_AFLCTR_AFLDAE);
> +}
> +
> +static void rcar_canfd_configure_rx(struct rcar_canfd_global *gpriv, u32 ch)
> +{
> +	/* Rx FIFO is used for reception */
> +	u32 cfg;
> +	u16 rfdc, rfpls;
> +
> +	/* Select Rx FIFO based on channel */
> +	u32 ridx = ch + RCANFD_RFFIFO_IDX;
> +
> +	rfdc = 2;		/* b010 - 8 messages Rx FIFO depth */
> +	rfpls = 7;		/* b111 - Max 64 bytes payload */
> +
> +	cfg = (RCANFD_RFFIFO_RFIM | RCANFD_RFFIFO_RFDC(rfdc) |
> +		RCANFD_RFFIFO_RFPLS(rfpls) | RCANFD_RFFIFO_RFIE);
> +	rcar_canfd_write(gpriv, RCANFD_RFCC(ridx), cfg);
> +}
> +
> +static void rcar_canfd_configure_tx(struct rcar_canfd_global *gpriv, u32 ch)
> +{
> +	/* Tx/Rx(Common) FIFO configured in Tx mode is
> +	 * used for transmission
> +	 *
> +	 * Each channel has 3 Common FIFO dedicated to them.
> +	 * Use the 1st (index 0) out of 3
> +	 */
> +	u32 cfg;
> +	u16 cftml, cfm, cfdc, cfpls;
> +
> +	cftml = 0;		/* 0th buffer */
> +	cfm = 1;		/* b01 - Transmit mode */
> +	cfdc = 2;		/* b010 - 8 messages Tx FIFO depth */
> +	cfpls = 7;		/* b111 - Max 64 bytes payload */
> +
> +	cfg = (RCANFD_CMFIFO_TML(cftml) | RCANFD_CMFIFO_M(cfm) |
> +		RCANFD_CMFIFO_CFIM | RCANFD_CMFIFO_DC(cfdc) |
> +		RCANFD_CMFIFO_PLS(cfpls) | RCANFD_CMFIFO_CFTXIE);
> +	rcar_canfd_write(gpriv, RCANFD_CFCC(ch, RCANFD_CFFIFO_IDX), cfg);
> +
> +	/* Clear FD mode specific control/status register */
> +	rcar_canfd_write(gpriv, RCANFD_F_CFFDCSTS(ch, RCANFD_CFFIFO_IDX), 0);
> +}
> +
> +static void rcar_canfd_enable_global_interrupts(struct rcar_canfd_global *gpriv)
> +{
> +	u32 ctr;
> +
> +	/* Clear any stray error interrupt flags */
> +	rcar_canfd_write(gpriv, RCANFD_GERFL, 0);
> +
> +	/* Global interrupts setup */
> +	ctr = RCANFD_GCTR_MEIE;
> +	ctr |= RCANFD_GCTR_CFMPOFIE;
> +
> +	rcar_canfd_set_bit(gpriv, RCANFD_GCTR, ctr);
> +}
> +
> +static void rcar_canfd_disable_global_interrupts(struct rcar_canfd_global
> +						 *gpriv)
> +{
> +	/* Disable all interrupts */
> +	rcar_canfd_write(gpriv, RCANFD_GCTR, 0);
> +
> +	/* Clear any stray error interrupt flags */
> +	rcar_canfd_write(gpriv, RCANFD_GERFL, 0);
> +}
> +
> +static void rcar_canfd_enable_channel_interrupts(struct rcar_canfd_channel
> +						 *priv)
> +{
> +	u32 ctr, ch = priv->channel;
> +
> +	/* Clear any stray error flags */
> +	rcar_canfd_write(priv, RCANFD_CERFL(ch), 0);
> +
> +	/* Channel interrupts setup */
> +	ctr = (RCANFD_CCTR_TAIE |
> +	       RCANFD_CCTR_ALIE | RCANFD_CCTR_BLIE |
> +	       RCANFD_CCTR_OLIE | RCANFD_CCTR_BORIE |
> +	       RCANFD_CCTR_BOEIE | RCANFD_CCTR_EPIE |
> +	       RCANFD_CCTR_EWIE | RCANFD_CCTR_BEIE);
> +	rcar_canfd_set_bit(priv, RCANFD_CCTR(ch), ctr);
> +}
> +
> +static void rcar_canfd_disable_channel_interrupts(struct rcar_canfd_channel
> +						  *priv)
> +{
> +	u32 ctr, ch = priv->channel;
> +
> +	ctr = (RCANFD_CCTR_TAIE |
> +	       RCANFD_CCTR_ALIE | RCANFD_CCTR_BLIE |
> +	       RCANFD_CCTR_OLIE | RCANFD_CCTR_BORIE |
> +	       RCANFD_CCTR_BOEIE | RCANFD_CCTR_EPIE |
> +	       RCANFD_CCTR_EWIE | RCANFD_CCTR_BEIE);
> +	rcar_canfd_clear_bit(priv, RCANFD_CCTR(ch), ctr);
> +
> +	/* Clear any stray error flags */
> +	rcar_canfd_write(priv, RCANFD_CERFL(ch), 0);
> +}
> +
> +static void rcar_canfd_global_error(struct net_device *ndev)
> +{
> +	struct rcar_canfd_channel *priv = netdev_priv(ndev);
> +	struct net_device_stats *stats = &ndev->stats;
> +	u32 ch = priv->channel;
> +	u32 gerfl, sts;
> +	u32 ridx = ch + RCANFD_RFFIFO_IDX;
> +
> +	gerfl = rcar_canfd_read(priv, RCANFD_GERFL);
> +	if ((gerfl & RCANFD_GERFL_EEF0) && (ch == 0)) {
> +		netdev_dbg(ndev, "Ch0: ECC Error flag\n");
> +		stats->tx_dropped++;
> +	}
> +	if ((gerfl & RCANFD_GERFL_EEF1) && (ch == 1)) {
> +		netdev_dbg(ndev, "Ch1: ECC Error flag\n");
> +		stats->tx_dropped++;
> +	}
> +	if (gerfl & RCANFD_GERFL_MES) {
> +		sts = rcar_canfd_read(priv,
> +				      RCANFD_CFSTS(ch, RCANFD_CFFIFO_IDX));
> +		if (sts & RCANFD_CMFIFO_CFMLT) {
> +			netdev_dbg(ndev, "Tx Message Lost flag\n");
> +			stats->tx_dropped++;
> +			rcar_canfd_write(priv,
> +					 RCANFD_CFSTS(ch, RCANFD_CFFIFO_IDX),
> +					 sts & ~RCANFD_CMFIFO_CFMLT);
> +		}
> +
> +		sts = rcar_canfd_read(priv, RCANFD_RFSTS(ridx));
> +		if (sts & RCANFD_RFFIFO_RFMLT) {
> +			netdev_dbg(ndev, "Rx Message Lost flag\n");
> +			stats->rx_dropped++;
> +			rcar_canfd_write(priv, RCANFD_RFSTS(ridx),
> +					 sts & ~RCANFD_RFFIFO_RFMLT);
> +		}
> +	}
> +	if (gerfl & RCANFD_GERFL_CMPOF) {
> +		/* Message Lost flag will be set for respective channel
> +		 * when this condition happens with counters and flags
> +		 * already updated.
> +		 */
> +		netdev_dbg(ndev, "global payload overflow interrupt\n");
> +	}
> +
> +	/* Clear all global error interrupts. Only affected channels bits
> +	 * get cleared
> +	 */
> +	rcar_canfd_write(priv, RCANFD_GERFL, 0);
> +}
> +
> +static void rcar_canfd_error(struct net_device *ndev)
> +{
> +	struct rcar_canfd_channel *priv = netdev_priv(ndev);
> +	struct net_device_stats *stats = &ndev->stats;
> +	struct can_frame *cf;
> +	struct sk_buff *skb;
> +	u32 cerfl, csts;
> +	u32 txerr = 0, rxerr = 0;
> +	u32 ch = priv->channel;
> +
> +	/* Propagate the error condition to the CAN stack */
> +	skb = alloc_can_err_skb(ndev, &cf);
> +	if (!skb) {
> +		stats->rx_dropped++;
> +		return;
> +	}
> +
> +	/* Channel error interrupt */
> +	cerfl = rcar_canfd_read(priv, RCANFD_CERFL(ch));
> +	csts = rcar_canfd_read(priv, RCANFD_CSTS(ch));
> +	txerr = RCANFD_CSTS_TECCNT(csts);
> +	rxerr = RCANFD_CSTS_RECCNT(csts);
> +
> +	netdev_dbg(ndev, "ch erfl %x sts %x txerr %u rxerr %u\n",
> +		   cerfl, csts, txerr, rxerr);
> +
> +	if (cerfl & RCANFD_CERFL_BEF) {
> +		netdev_dbg(ndev, "Bus error\n");
> +		cf->can_id |= CAN_ERR_BUSERROR | CAN_ERR_PROT;
> +		cf->data[2] = CAN_ERR_PROT_UNSPEC;
> +		priv->can.can_stats.bus_error++;
> +	}
> +	if (cerfl & RCANFD_CERFL_ADEF) {
> +		netdev_dbg(ndev, "ACK Delimiter Error\n");
> +		stats->tx_errors++;
> +		cf->data[3] |= CAN_ERR_PROT_LOC_ACK_DEL;
> +	}
> +	if (cerfl & RCANFD_CERFL_B0EF) {
> +		netdev_dbg(ndev, "Bit Error (dominant)\n");
> +		stats->tx_errors++;
> +		cf->data[2] |= CAN_ERR_PROT_BIT0;
> +	}
> +	if (cerfl & RCANFD_CERFL_B1EF) {
> +		netdev_dbg(ndev, "Bit Error (recessive)\n");
> +		stats->tx_errors++;
> +		cf->data[2] |= CAN_ERR_PROT_BIT1;
> +	}
> +	if (cerfl & RCANFD_CERFL_CEF) {
> +		netdev_dbg(ndev, "CRC Error\n");
> +		stats->rx_errors++;
> +		cf->data[3] |= CAN_ERR_PROT_LOC_CRC_SEQ;
> +	}
> +	if (cerfl & RCANFD_CERFL_AEF) {
> +		netdev_dbg(ndev, "ACK Error\n");
> +		stats->tx_errors++;
> +		cf->can_id |= CAN_ERR_ACK;
> +		cf->data[3] |= CAN_ERR_PROT_LOC_ACK;
> +	}
> +	if (cerfl & RCANFD_CERFL_FEF) {
> +		netdev_dbg(ndev, "Form Error\n");
> +		stats->rx_errors++;
> +		cf->data[2] |= CAN_ERR_PROT_FORM;
> +	}
> +	if (cerfl & RCANFD_CERFL_SEF) {
> +		netdev_dbg(ndev, "Stuff Error\n");
> +		stats->rx_errors++;
> +		cf->data[2] |= CAN_ERR_PROT_STUFF;
> +	}
> +	if (cerfl & RCANFD_CERFL_ALEF) {
> +		netdev_dbg(ndev, "Arbitration lost Error\n");
> +		priv->can.can_stats.arbitration_lost++;
> +		cf->can_id |= CAN_ERR_LOSTARB;
> +		cf->data[0] |= CAN_ERR_LOSTARB_UNSPEC;
> +	}
> +	if (cerfl & RCANFD_CERFL_BLEF) {
> +		netdev_dbg(ndev, "Bus Lock Error\n");
> +		stats->rx_errors++;
> +		cf->can_id |= CAN_ERR_BUSERROR;
> +	}
> +	if (cerfl & RCANFD_CERFL_EWEF) {
> +		netdev_dbg(ndev, "Error warning interrupt\n");
> +		priv->can.state = CAN_STATE_ERROR_WARNING;
> +		priv->can.can_stats.error_warning++;
> +		cf->can_id |= CAN_ERR_CRTL;
> +		cf->data[1] = txerr > rxerr ? CAN_ERR_CRTL_TX_WARNING :
> +			CAN_ERR_CRTL_RX_WARNING;
> +		cf->data[6] = txerr;
> +		cf->data[7] = rxerr;
> +	}
> +	if (cerfl & RCANFD_CERFL_EPEF) {
> +		netdev_dbg(ndev, "Error passive interrupt\n");
> +		priv->can.state = CAN_STATE_ERROR_PASSIVE;
> +		priv->can.can_stats.error_passive++;
> +		cf->can_id |= CAN_ERR_CRTL;
> +		cf->data[1] = txerr > rxerr ? CAN_ERR_CRTL_TX_PASSIVE :
> +			CAN_ERR_CRTL_RX_PASSIVE;
> +		cf->data[6] = txerr;
> +		cf->data[7] = rxerr;
> +	}
> +	if (cerfl & RCANFD_CERFL_BOEEF) {
> +		netdev_dbg(ndev, "Bus-off entry interrupt\n");
> +		rcar_canfd_tx_failure_cleanup(ndev);
> +		priv->can.state = CAN_STATE_BUS_OFF;
> +		priv->can.can_stats.bus_off++;
> +		can_bus_off(ndev);
> +		cf->can_id |= CAN_ERR_BUSOFF;
> +	}
> +	if (cerfl & RCANFD_CERFL_OLEF) {
> +		netdev_dbg(ndev,
> +			   "Overload Frame Transmission error interrupt\n");
> +		stats->tx_errors++;
> +		cf->can_id |= CAN_ERR_PROT;
> +		cf->data[2] |= CAN_ERR_PROT_OVERLOAD;
> +	}
> +
> +	/* Clear all channel error interrupts */
> +	rcar_canfd_write(priv, RCANFD_CERFL(ch), 0);
> +	stats->rx_packets++;
> +	stats->rx_bytes += cf->can_dlc;
> +	netif_rx(skb);
> +}
> +
> +static void rcar_canfd_tx_done(struct net_device *ndev)
> +{
> +	struct rcar_canfd_channel *priv = netdev_priv(ndev);
> +	struct net_device_stats *stats = &ndev->stats;
> +	u32 sts;
> +	unsigned long flags;
> +	u32 ch = priv->channel;
> +
> +	do {

You should iterare over all pending CAN frames:

> 	for (/* nix */; (priv->tx_head - priv->tx_tail) > 0; priv->tx_tail++) {


> +		u8 unsent, sent;
> +
> +		sent = priv->tx_tail % RCANFD_FIFO_DEPTH;

and check here, if that packet has really been tramsitted. Exit the loop
otherweise.

> +		stats->tx_packets++;
> +		stats->tx_bytes += priv->tx_len[sent];
> +		priv->tx_len[sent] = 0;
> +		can_get_echo_skb(ndev, sent);
> +
> +		spin_lock_irqsave(&priv->tx_lock, flags);

What does the tx_lock protect? The tx path is per channel, isn't it?

> +		priv->tx_tail++;
> +		sts = rcar_canfd_read(priv,
> +				      RCANFD_CFSTS(ch, RCANFD_CFFIFO_IDX));
> +		unsent = RCANFD_CMFIFO_CFMC(sts);
> +
> +		/* Wake producer only when there is room */
> +		if (unsent != RCANFD_FIFO_DEPTH)
> +			netif_wake_queue(ndev);

Move the netif_wake_queue() out of the loop.

> +
> +		if (priv->tx_head - priv->tx_tail <= unsent) {
> +			spin_unlock_irqrestore(&priv->tx_lock, flags);
> +			break;
> +		}
> +		spin_unlock_irqrestore(&priv->tx_lock, flags);
> +
> +	} while (1);
> +
> +	/* Clear interrupt */
> +	rcar_canfd_write(priv, RCANFD_CFSTS(ch, RCANFD_CFFIFO_IDX),
> +			 sts & ~RCANFD_CMFIFO_CFTXIF);
> +	can_led_event(ndev, CAN_LED_EVENT_TX);
> +}
> +
> +static irqreturn_t rcar_canfd_global_interrupt(int irq, void *dev_id)
> +{
> +	struct rcar_canfd_global *gpriv = dev_id;
> +	struct net_device *ndev;
> +	struct rcar_canfd_channel *priv;
> +	u32 sts, gerfl;
> +	u32 ch, ridx;
> +
> +	/* Global error interrupts still indicate a condition specific
> +	 * to a channel. RxFIFO interrupt is a global interrupt.
> +	 */
> +	for_each_set_bit(ch, &gpriv->channels_mask, RCANFD_NUM_CHANNELS) {
> +		priv = gpriv->ch[ch];
> +		ndev = priv->ndev;
> +		ridx = ch + RCANFD_RFFIFO_IDX;
> +
> +		/* Global error interrupts */
> +		gerfl = rcar_canfd_read(priv, RCANFD_GERFL);
> +		if (RCANFD_GERFL_ERR(gerfl))
> +			rcar_canfd_global_error(ndev);
> +
> +		/* Handle Rx interrupts */
> +		sts = rcar_canfd_read(priv, RCANFD_RFSTS(ridx));
> +		if (sts & RCANFD_RFFIFO_RFIF) {
> +			if (napi_schedule_prep(&priv->napi)) {
> +				/* Disable Rx FIFO interrupts */
> +				rcar_canfd_clear_bit(priv,
> +						     RCANFD_RFCC(ridx),
> +						     RCANFD_RFFIFO_RFIE);
> +				__napi_schedule(&priv->napi);
> +			}
> +		}
> +	}
> +	return IRQ_HANDLED;
> +}
> +
> +static irqreturn_t rcar_canfd_channel_interrupt(int irq, void *dev_id)
> +{
> +	struct rcar_canfd_global *gpriv = dev_id;
> +	struct net_device *ndev;
> +	struct rcar_canfd_channel *priv;
> +	u32 sts, cerfl, ch;
> +
> +	/* Common FIFO is a per channel resource */
> +	for_each_set_bit(ch, &gpriv->channels_mask, RCANFD_NUM_CHANNELS) {
> +		priv = gpriv->ch[ch];
> +		ndev = priv->ndev;
> +
> +		/* Channel error interrupts */
> +		cerfl = rcar_canfd_read(priv, RCANFD_CERFL(ch));
> +		if (RCANFD_CERFL_ERR(cerfl))
> +			rcar_canfd_error(ndev);
> +
> +		/* Handle Tx interrupts */
> +		sts = rcar_canfd_read(priv,
> +				      RCANFD_CFSTS(ch, RCANFD_CFFIFO_IDX));
> +		if (sts & RCANFD_CMFIFO_CFTXIF)
> +			rcar_canfd_tx_done(ndev);
> +	}
> +	return IRQ_HANDLED;
> +}
> +
> +static void rcar_canfd_set_bittiming(struct net_device *dev)
> +{
> +	struct rcar_canfd_channel *priv = netdev_priv(dev);
> +	const struct can_bittiming *bt = &priv->can.bittiming;
> +	const struct can_bittiming *dbt = &priv->can.data_bittiming;
> +	u16 brp, sjw, tseg1, tseg2;
> +	u32 cfg;
> +	u32 ch = priv->channel;
> +
> +	/* Nominal bit timing settings */
> +	brp = bt->brp - 1;
> +	sjw = bt->sjw - 1;
> +	tseg1 = bt->prop_seg + bt->phase_seg1 - 1;
> +	tseg2 = bt->phase_seg2 - 1;
> +
> +	cfg = (RCANFD_NR_TSEG1(tseg1) | RCANFD_NR_BRP(brp) |
> +	       RCANFD_NR_SJW(sjw) | RCANFD_NR_TSEG2(tseg2));
> +
> +	rcar_canfd_write(priv, RCANFD_CCFG(ch), cfg);
> +	netdev_dbg(priv->ndev, "nrate: brp %u, sjw %u, tseg1 %u, tseg2 %u\n",
> +		   brp, sjw, tseg1, tseg2);
> +
> +	/* Data bit timing settings */
> +	brp = dbt->brp - 1;
> +	sjw = dbt->sjw - 1;
> +	tseg1 = dbt->prop_seg + dbt->phase_seg1 - 1;
> +	tseg2 = dbt->phase_seg2 - 1;
> +
> +	cfg = (RCANFD_DR_TSEG1(tseg1) | RCANFD_DR_BRP(brp) |
> +	       RCANFD_DR_SJW(sjw) | RCANFD_DR_TSEG2(tseg2));
> +
> +	rcar_canfd_write(priv, RCANFD_F_CDFG(ch), cfg);
> +	netdev_dbg(priv->ndev, "drate: brp %u, sjw %u, tseg1 %u, tseg2 %u\n",
> +		   brp, sjw, tseg1, tseg2);
> +}
> +
> +static int rcar_canfd_start(struct net_device *ndev)
> +{
> +	struct rcar_canfd_channel *priv = netdev_priv(ndev);
> +	int err = -EOPNOTSUPP;
> +	u32 sts, ch = priv->channel;
> +	u32 ridx = ch + RCANFD_RFFIFO_IDX;
> +
> +	/* Ensure channel starts in FD mode */
> +	if (!(priv->can.ctrlmode & CAN_CTRLMODE_FD)) {
> +		netdev_err(ndev, "enable can fd mode for channel %d\n", ch);
> +		goto fail_mode;
> +	}
> +
> +	rcar_canfd_set_bittiming(ndev);
> +
> +	rcar_canfd_enable_channel_interrupts(priv);
> +
> +	/* Set channel to Operational mode */
> +	rcar_canfd_update_bit(priv, RCANFD_CCTR(ch),
> +			      RCANFD_CCTR_MODEMASK, RCANFD_CCTR_COPM);
> +
> +	/* Verify channel mode change */
> +	err = readl_poll_timeout((priv->base + RCANFD_CSTS(ch)), sts,
> +				 (sts & RCANFD_CSTS_COM), 2, 500000);
> +	if (err) {
> +		netdev_err(ndev, "channel %u communication state failed\n", ch);
> +		goto fail_mode_change;
> +	}
> +
> +	/* Enable Common & Rx FIFO */
> +	rcar_canfd_set_bit(priv, RCANFD_CFCC(ch, RCANFD_CFFIFO_IDX),
> +			   RCANFD_CMFIFO_CFE);
> +	rcar_canfd_set_bit(priv, RCANFD_RFCC(ridx), RCANFD_RFFIFO_RFE);
> +
> +	priv->can.state = CAN_STATE_ERROR_ACTIVE;
> +	return 0;
> +
> +fail_mode_change:
> +	rcar_canfd_disable_channel_interrupts(priv);
> +fail_mode:
> +	return err;
> +}
> +
> +static int rcar_canfd_open(struct net_device *ndev)
> +{
> +	struct rcar_canfd_channel *priv = netdev_priv(ndev);
> +	struct rcar_canfd_global *gpriv = priv->gpriv;
> +	int err;
> +
> +	/* Peripheral clock is already enabled in probe */
> +	err = clk_prepare_enable(gpriv->can_clk);
> +	if (err) {
> +		netdev_err(ndev, "failed to enable CAN clock, error %d\n", err);
> +		goto out_clock;
> +	}
> +
> +	err = open_candev(ndev);
> +	if (err) {
> +		netdev_err(ndev, "open_candev() failed, error %d\n", err);
> +		goto out_can_clock;
> +	}
> +
> +	napi_enable(&priv->napi);
> +	err = rcar_canfd_start(ndev);
> +	if (err)
> +		goto out_close;
> +	netif_start_queue(ndev);
> +	can_led_event(ndev, CAN_LED_EVENT_OPEN);
> +	return 0;
> +out_close:
> +	napi_disable(&priv->napi);
> +	close_candev(ndev);
> +out_can_clock:
> +	clk_disable_unprepare(gpriv->can_clk);
> +out_clock:
> +	return err;
> +}
> +
> +static void rcar_canfd_stop(struct net_device *ndev)
> +{
> +	struct rcar_canfd_channel *priv = netdev_priv(ndev);
> +	int err;
> +	u32 sts, ch = priv->channel;
> +	u32 ridx = ch + RCANFD_RFFIFO_IDX;
> +
> +	/* Transition to channel reset mode  */
> +	rcar_canfd_update_bit(priv, RCANFD_CCTR(ch),
> +			      RCANFD_CCTR_MODEMASK, RCANFD_CCTR_CRESET);
> +
> +	/* Check Channel reset mode */
> +	err = readl_poll_timeout((priv->base + RCANFD_CSTS(ch)), sts,
> +				 (sts & RCANFD_CSTS_RESET), 2, 500000);
> +	if (err)
> +		netdev_err(ndev, "channel %u reset failed\n", ch);
> +
> +	rcar_canfd_disable_channel_interrupts(priv);
> +
> +	/* Disable Common & Rx FIFO */
> +	rcar_canfd_clear_bit(priv, RCANFD_CFCC(ch, RCANFD_CFFIFO_IDX),
> +			     RCANFD_CMFIFO_CFE);
> +	rcar_canfd_clear_bit(priv, RCANFD_RFCC(ridx), RCANFD_RFFIFO_RFE);
> +
> +	/* Set the state as STOPPED */
> +	priv->can.state = CAN_STATE_STOPPED;
> +}
> +
> +static int rcar_canfd_close(struct net_device *ndev)
> +{
> +	struct rcar_canfd_channel *priv = netdev_priv(ndev);
> +	struct rcar_canfd_global *gpriv = priv->gpriv;
> +
> +	netif_stop_queue(ndev);
> +	rcar_canfd_stop(ndev);
> +	napi_disable(&priv->napi);
> +	clk_disable_unprepare(gpriv->can_clk);
> +	close_candev(ndev);
> +	can_led_event(ndev, CAN_LED_EVENT_STOP);
> +	return 0;
> +}
> +
> +static netdev_tx_t rcar_canfd_start_xmit(struct sk_buff *skb,
> +					 struct net_device *ndev)
> +{
> +	struct rcar_canfd_channel *priv = netdev_priv(ndev);
> +	struct canfd_frame *cf = (struct canfd_frame *)skb->data;
> +	u32 sts, id, ptr;
> +	unsigned long flags;
> +	u32 ch = priv->channel;
> +
> +	if (can_dropped_invalid_skb(ndev, skb))
> +		return NETDEV_TX_OK;
> +
> +	if (cf->can_id & CAN_EFF_FLAG) {
> +		id = cf->can_id & CAN_EFF_MASK;
> +		id |= RCANFD_CMFIFO_CFIDE;
> +	} else {
> +		id = cf->can_id & CAN_SFF_MASK;
> +	}
> +
> +	if (cf->can_id & CAN_RTR_FLAG)
> +		id |= RCANFD_CMFIFO_CFRTR;
> +
> +	rcar_canfd_write(priv, RCANFD_F_CFID(ch, RCANFD_CFFIFO_IDX),
> +			 id);
> +	ptr = RCANFD_CMFIFO_CFDLC(can_len2dlc(cf->len));

ptr usually means pointer, better call it dlc.

> +	rcar_canfd_write(priv, RCANFD_F_CFPTR(ch, RCANFD_CFFIFO_IDX),
> +			 ptr);
> +
> +	sts = rcar_canfd_read(priv, RCANFD_F_CFFDCSTS(ch, RCANFD_CFFIFO_IDX));
> +	if (can_is_canfd_skb(skb)) {
> +		/* CAN FD frame format */
> +		sts |= RCANFD_CMFIFO_CFFDF;
> +		if (cf->flags & CANFD_BRS)
> +			sts |= RCANFD_CMFIFO_CFBRS;
> +		else
> +			sts &= ~RCANFD_CMFIFO_CFBRS;
> +	} else {
> +		sts &= ~RCANFD_CMFIFO_CFFDF;
> +	}
> +	rcar_canfd_write(priv, RCANFD_F_CFFDCSTS(ch, RCANFD_CFFIFO_IDX), sts);
> +
> +	rcar_canfd_put_data(cf, priv,
> +			    RCANFD_F_CFDF(ch, RCANFD_CFFIFO_IDX, 0));
> +
> +	priv->tx_len[priv->tx_head % RCANFD_FIFO_DEPTH] = cf->len;
> +	can_put_echo_skb(skb, ndev, priv->tx_head % RCANFD_FIFO_DEPTH);
> +
> +	spin_lock_irqsave(&priv->tx_lock, flags);
> +	priv->tx_head++;
> +
> +	/* Start Tx: Write 0xff to CFPC to increment the CPU-side
> +	 * pointer for the Common FIFO
> +	 */
> +	rcar_canfd_write(priv, RCANFD_CFPCTR(ch, RCANFD_CFFIFO_IDX), 0xff);
> +
> +	/* Stop the queue if we've filled all FIFO entries */
> +	if (priv->tx_head - priv->tx_tail >= RCANFD_FIFO_DEPTH)
> +		netif_stop_queue(ndev);

Please move the check of stop_queue, before starting the send.

> +
> +	spin_unlock_irqrestore(&priv->tx_lock, flags);
> +	return NETDEV_TX_OK;
> +}
> +
> +static void rcar_canfd_rx_pkt(struct rcar_canfd_channel *priv)
> +{
> +	struct net_device_stats *stats = &priv->ndev->stats;
> +	struct canfd_frame *cf;
> +	struct sk_buff *skb;
> +	u32 sts = 0, id, ptr;
> +	u32 ch = priv->channel;
> +	u32 ridx = ch + RCANFD_RFFIFO_IDX;
> +
> +	id = rcar_canfd_read(priv, RCANFD_F_RFID(ridx));
> +	ptr = rcar_canfd_read(priv, RCANFD_F_RFPTR(ridx));
> +
> +	sts = rcar_canfd_read(priv, RCANFD_F_RFFDSTS(ridx));
> +	if (sts & RCANFD_RFFIFO_RFFDF)
> +		skb = alloc_canfd_skb(priv->ndev, &cf);
> +	else
> +		skb = alloc_can_skb(priv->ndev,
> +				    (struct can_frame **)&cf);
> +
> +	if (!skb) {
> +		stats->rx_dropped++;
> +		return;
> +	}
> +
> +	if (sts & RCANFD_RFFIFO_RFFDF)
> +		cf->len = can_dlc2len(RCANFD_RFFIFO_RFDLC(ptr));
> +	else
> +		cf->len = get_can_dlc(RCANFD_RFFIFO_RFDLC(ptr));
> +
> +	if (sts & RCANFD_RFFIFO_RFESI) {
> +		cf->flags |= CANFD_ESI;
> +		netdev_dbg(priv->ndev, "ESI Error\n");
> +	}
> +
> +	if (id & RCANFD_RFFIFO_RFIDE)
> +		cf->can_id = (id & CAN_EFF_MASK) | CAN_EFF_FLAG;
> +	else
> +		cf->can_id = id & CAN_SFF_MASK;
> +
> +	if (!(sts & RCANFD_RFFIFO_RFFDF) && (id & RCANFD_RFFIFO_RFRTR)) {
> +		cf->can_id |= CAN_RTR_FLAG;
> +	} else {
> +		if (sts & RCANFD_RFFIFO_RFBRS)
> +			cf->flags |= CANFD_BRS;
> +
> +		rcar_canfd_get_data(cf, priv, RCANFD_F_RFDF(ridx, 0));
> +	}
> +
> +	/* Write 0xff to RFPC to increment the CPU-side
> +	 * pointer of the Rx FIFO
> +	 */
> +	rcar_canfd_write(priv, RCANFD_RFPCTR(ridx), 0xff);
> +
> +	can_led_event(priv->ndev, CAN_LED_EVENT_RX);
> +
> +	stats->rx_bytes += cf->len;
> +	stats->rx_packets++;
> +	netif_receive_skb(skb);
> +}
> +
> +static int rcar_canfd_rx_poll(struct napi_struct *napi, int quota)
> +{
> +	struct rcar_canfd_channel *priv =
> +		container_of(napi, struct rcar_canfd_channel, napi);
> +	int num_pkts;
> +	u32 sts;
> +	u32 ch = priv->channel;
> +	u32 ridx = ch + RCANFD_RFFIFO_IDX;
> +
> +	for (num_pkts = 0; num_pkts < quota; num_pkts++) {
> +		sts = rcar_canfd_read(priv, RCANFD_RFSTS(ridx));
> +		/* Clear interrupt bit */
> +		if (sts & RCANFD_RFFIFO_RFIF)
> +			rcar_canfd_write(priv, RCANFD_RFSTS(ridx),
> +					 sts & ~RCANFD_RFFIFO_RFIF);
> +
> +		/* Check FIFO empty condition */
> +		if (sts & RCANFD_RFFIFO_RFEMP)
> +			break;
> +
> +		rcar_canfd_rx_pkt(priv);

This sequence looks strange. You first conditionally ack the interrupt
then you check for empty fifo, then read the CAN frame. I would assume
that you first check if there's a CAN frame, read it and then clear the
interrupt.

> +	}
> +
> +	/* All packets processed */
> +	if (num_pkts < quota) {
> +		napi_complete(napi);
> +		/* Enable Rx FIFO interrupts */
> +		rcar_canfd_set_bit(priv, RCANFD_RFCC(ridx), RCANFD_RFFIFO_RFIE);
> +	}
> +	return num_pkts;
> +}
> +
> +static int rcar_canfd_do_set_mode(struct net_device *ndev, enum can_mode mode)
> +{
> +	int err;
> +
> +	switch (mode) {
> +	case CAN_MODE_START:
> +		err = rcar_canfd_start(ndev);
> +		if (err)
> +			return err;
> +		netif_wake_queue(ndev);
> +		return 0;
> +	default:
> +		return -EOPNOTSUPP;
> +	}
> +}
> +
> +static int rcar_canfd_get_berr_counter(const struct net_device *dev,
> +				       struct can_berr_counter *bec)
> +{
> +	struct rcar_canfd_channel *priv = netdev_priv(dev);
> +	u32 val, ch = priv->channel;
> +
> +	/* Peripheral clock is already enabled in probe */
> +	val = rcar_canfd_read(priv, RCANFD_CSTS(ch));
> +	bec->txerr = RCANFD_CSTS_TECCNT(val);
> +	bec->rxerr = RCANFD_CSTS_RECCNT(val);
> +	return 0;
> +}
> +
> +static const struct net_device_ops rcar_canfd_netdev_ops = {
> +	.ndo_open = rcar_canfd_open,
> +	.ndo_stop = rcar_canfd_close,
> +	.ndo_start_xmit = rcar_canfd_start_xmit,
> +	.ndo_change_mtu = can_change_mtu,
> +};
> +
> +static int rcar_canfd_channel_probe(struct rcar_canfd_global *gpriv, u32 ch)
> +{
> +	struct platform_device *pdev = gpriv->pdev;
> +	struct rcar_canfd_channel *priv;
> +	struct net_device *ndev;
> +	int err = -ENODEV;
> +
> +	ndev = alloc_candev(sizeof(*priv), RCANFD_FIFO_DEPTH);
> +	if (!ndev) {
> +		dev_err(&pdev->dev, "alloc_candev() failed\n");
> +		err = -ENOMEM;
> +		goto fail;
> +	}
> +	priv = netdev_priv(ndev);
> +
> +	ndev->netdev_ops = &rcar_canfd_netdev_ops;
> +	ndev->flags |= IFF_ECHO;
> +	priv->ndev = ndev;
> +	priv->base = gpriv->base;
> +	priv->channel = ch;
> +	priv->can.clock.freq = gpriv->freq;
> +	dev_info(&pdev->dev, "can_clk rate is %u\n", priv->can.clock.freq);
> +
> +	priv->can.bittiming_const = &rcar_canfd_nom_bittiming_const;
> +	priv->can.data_bittiming_const =
> +		&rcar_canfd_data_bittiming_const;
> +
> +	/* Controller starts in CAN FD mode, which supports classical CAN and
> +	 * CAN FD frames. For classical CAN frames only configurations, data
> +	 * bitrate should be configured the same as nominal bitrate.
> +	 */
> +	priv->can.ctrlmode_supported = CAN_CTRLMODE_BERR_REPORTING |
> +		CAN_CTRLMODE_FD;
> +
> +	priv->can.do_set_mode = rcar_canfd_do_set_mode;
> +	priv->can.do_get_berr_counter = rcar_canfd_get_berr_counter;
> +	priv->gpriv = gpriv;
> +	SET_NETDEV_DEV(ndev, &pdev->dev);
> +
> +	netif_napi_add(ndev, &priv->napi, rcar_canfd_rx_poll,
> +		       RCANFD_NAPI_WEIGHT);
> +	err = register_candev(ndev);
> +	if (err) {
> +		dev_err(&pdev->dev,
> +			"register_candev() failed, error %d\n", err);
> +		goto fail_candev;
> +	}
> +	spin_lock_init(&priv->tx_lock);
> +	devm_can_led_init(ndev);
> +	gpriv->ch[priv->channel] = priv;
> +	dev_info(&pdev->dev, "device registered (channel %u)\n", priv->channel);
> +	return 0;
> +
> +fail_candev:
> +	netif_napi_del(&priv->napi);
> +	free_candev(ndev);
> +fail:
> +	return err;
> +}
> +
> +static void rcar_canfd_channel_remove(struct rcar_canfd_global *gpriv, u32 ch)
> +{
> +	struct rcar_canfd_channel *priv = gpriv->ch[ch];
> +
> +	if (priv) {
> +		unregister_candev(priv->ndev);
> +		netif_napi_del(&priv->napi);
> +		free_candev(priv->ndev);
> +	}
> +}
> +
> +static int rcar_canfd_probe(struct platform_device *pdev)
> +{
> +	struct resource *mem;
> +	void __iomem *addr;
> +	u32 sts, ch;
> +	struct rcar_canfd_global *gpriv;
> +	struct device_node *of_child;
> +	unsigned long channels_mask = 0;
> +	int err, ch_irq, g_irq;
> +
> +	of_child = of_get_child_by_name(pdev->dev.of_node, "channel0");
> +	if (of_child && of_device_is_available(of_child))
> +		channels_mask |= BIT(0);	/* Channel 0 */
> +
> +	of_child = of_get_child_by_name(pdev->dev.of_node, "channel1");
> +	if (of_child && of_device_is_available(of_child))
> +		channels_mask |= BIT(1);	/* Channel 1 */
> +
> +	ch_irq = platform_get_irq(pdev, 0);
> +	if (ch_irq < 0) {
> +		dev_err(&pdev->dev, "no Channel IRQ resource\n");
> +		err = ch_irq;
> +		goto fail_dev;
> +	}
> +
> +	g_irq = platform_get_irq(pdev, 1);
> +	if (g_irq < 0) {
> +		dev_err(&pdev->dev, "no Global IRQ resource\n");
> +		err = g_irq;
> +		goto fail_dev;
> +	}
> +
> +	/* Global controller context */
> +	gpriv = devm_kzalloc(&pdev->dev, sizeof(*gpriv), GFP_KERNEL);
> +	if (!gpriv) {
> +		err = -ENOMEM;
> +		goto fail_dev;
> +	}
> +	gpriv->pdev = pdev;
> +	gpriv->channels_mask = channels_mask;
> +
> +	/* Peripheral clock */
> +	gpriv->clkp = devm_clk_get(&pdev->dev, "fck");
> +	if (IS_ERR(gpriv->clkp)) {
> +		err = PTR_ERR(gpriv->clkp);
> +		dev_err(&pdev->dev, "cannot get peripheral clock, error %d\n",
> +			err);
> +		goto fail_dev;
> +	}
> +
> +	/* fCAN clock: Pick External clock. If not available fallback to
> +	 * CANFD clock
> +	 */
> +	gpriv->can_clk = devm_clk_get(&pdev->dev, "can_clk");
> +	if (IS_ERR(gpriv->can_clk)) {
> +		gpriv->can_clk = devm_clk_get(&pdev->dev, "canfd");
> +		if (IS_ERR(gpriv->can_clk)) {
> +			err = PTR_ERR(gpriv->can_clk);
> +			dev_err(&pdev->dev,
> +				"cannot get canfd clock, error %d\n", err);
> +			goto fail_dev;
> +		}
> +		gpriv->clock_select = RCANFD_CANFDCLK;
> +
> +	} else {
> +		gpriv->clock_select = RCANFD_EXTCLK;
> +	}
> +	gpriv->freq = clk_get_rate(gpriv->can_clk);
> +
> +	if (gpriv->clock_select == RCANFD_CANFDCLK)
> +		/* CANFD clock is further divided by (1/2) within the IP */
> +		gpriv->freq /= 2;
> +
> +	mem = platform_get_resource(pdev, IORESOURCE_MEM, 0);
> +	addr = devm_ioremap_resource(&pdev->dev, mem);
> +	if (IS_ERR(addr)) {
> +		err = PTR_ERR(addr);
> +		goto fail_dev;
> +	}
> +	gpriv->base = addr;
> +
> +	/* Request IRQ that's common for both channels */
> +	err = devm_request_irq(&pdev->dev, ch_irq,
> +			       rcar_canfd_channel_interrupt, 0,
> +			       "canfd.chn", gpriv);
> +	if (err) {
> +		dev_err(&pdev->dev, "devm_request_irq(%d) failed, error %d\n",
> +			ch_irq, err);
> +		goto fail_dev;
> +	}
> +	err = devm_request_irq(&pdev->dev, g_irq,
> +			       rcar_canfd_global_interrupt, 0,
> +			       "canfd.gbl", gpriv);
> +	if (err) {
> +		dev_err(&pdev->dev, "devm_request_irq(%d) failed, error %d\n",
> +			g_irq, err);
> +		goto fail_dev;
> +	}
> +
> +	/* Enable peripheral clock for register access */
> +	err = clk_prepare_enable(gpriv->clkp);
> +	if (err) {
> +		dev_err(&pdev->dev,
> +			"failed to enable peripheral clock, error %d\n", err);
> +		goto fail_dev;
> +	}
> +
> +	err = rcar_canfd_reset_controller(gpriv);
> +	if (err) {
> +		dev_err(&pdev->dev, "reset controller failed\n");
> +		goto fail_clk;
> +	}
> +
> +	/* Controller in Global reset & Channel reset mode */
> +	rcar_canfd_configure_controller(gpriv);
> +
> +	/* Configure per channel attributes */
> +	for_each_set_bit(ch, &gpriv->channels_mask, RCANFD_NUM_CHANNELS) {
> +		/* Configure Channel's Rx fifo */
> +		rcar_canfd_configure_rx(gpriv, ch);
> +
> +		/* Configure Channel's Tx (Common) fifo */
> +		rcar_canfd_configure_tx(gpriv, ch);
> +
> +		/* Configure receive rules */
> +		rcar_canfd_configure_afl_rules(gpriv, ch);
> +	}
> +
> +	/* Configure common interrupts */
> +	rcar_canfd_enable_global_interrupts(gpriv);
> +
> +	/* Start Global operation mode */
> +	rcar_canfd_update_bit(gpriv, RCANFD_GCTR, RCANFD_GCTR_MODEMASK,
> +			      RCANFD_GCTR_GOPM);
> +
> +	/* Verify mode change */
> +	err = readl_poll_timeout((gpriv->base + RCANFD_GSTS), sts,
> +				 !(sts & RCANFD_GSTS_GNOPM), 2, 500000);
> +	if (err) {
> +		dev_err(&pdev->dev, "global operational mode failed\n");
> +		goto fail_mode;
> +	}
> +
> +	for_each_set_bit(ch, &gpriv->channels_mask, RCANFD_NUM_CHANNELS) {
> +		err = rcar_canfd_channel_probe(gpriv, ch);
> +		if (err)
> +			goto fail_channel;
> +	}

Should the CAN IP core be clocked the whole time? What about shuting
down the clock and enabling it on ifup?

> +
> +	platform_set_drvdata(pdev, gpriv);
> +	dev_info(&pdev->dev, "global operational state (clk %d)\n",
> +		 gpriv->clock_select);
> +	return 0;
> +
> +fail_channel:
> +	for_each_set_bit(ch, &gpriv->channels_mask, RCANFD_NUM_CHANNELS)
> +		rcar_canfd_channel_remove(gpriv, ch);
> +fail_mode:
> +	rcar_canfd_disable_global_interrupts(gpriv);
> +fail_clk:
> +	clk_disable_unprepare(gpriv->clkp);
> +fail_dev:
> +	return err;
> +}
> +
> +static int rcar_canfd_remove(struct platform_device *pdev)
> +{
> +	struct rcar_canfd_global *gpriv = platform_get_drvdata(pdev);
> +	struct rcar_canfd_channel *priv;
> +	u32 ch;
> +
> +	rcar_canfd_reset_controller(gpriv);
> +	rcar_canfd_disable_global_interrupts(gpriv);
> +
> +	for_each_set_bit(ch, &gpriv->channels_mask, RCANFD_NUM_CHANNELS) {
> +		priv = gpriv->ch[ch];
> +		if (priv) {

This should always be true.

> +			rcar_canfd_disable_channel_interrupts(priv);
> +			unregister_candev(priv->ndev);
> +			netif_napi_del(&priv->napi);
> +			free_candev(priv->ndev);

Please make use of rcar_canfd_channel_remove(), as you already have the
function.

> +		}
> +	}
> +
> +	/* Enter global sleep mode */
> +	rcar_canfd_set_bit(gpriv, RCANFD_GCTR, RCANFD_GCTR_SLPR);
> +	clk_disable_unprepare(gpriv->clkp);
> +	return 0;
> +}
> +
> +static int __maybe_unused rcar_canfd_suspend(struct device *dev)
> +{
> +	return 0;
> +}
> +
> +static int __maybe_unused rcar_canfd_resume(struct device *dev)
> +{
> +	return 0;
> +}
> +
> +static SIMPLE_DEV_PM_OPS(rcar_canfd_pm_ops, rcar_canfd_suspend,
> +			 rcar_canfd_resume);
> +
> +static const struct of_device_id rcar_canfd_of_table[] = {
> +	{ .compatible = "renesas,rcar-gen3-canfd" },
> +	{ }
> +};
> +
> +MODULE_DEVICE_TABLE(of, rcar_canfd_of_table);
> +
> +static struct platform_driver rcar_canfd_driver = {
> +	.driver = {
> +		.name = RCANFD_DRV_NAME,
> +		.of_match_table = of_match_ptr(rcar_canfd_of_table),
> +		.pm = &rcar_canfd_pm_ops,
> +	},
> +	.probe = rcar_canfd_probe,
> +	.remove = rcar_canfd_remove,
> +};
> +
> +module_platform_driver(rcar_canfd_driver);
> +
> +MODULE_AUTHOR("Ramesh Shanmugasundaram <ramesh.shanmugasundaram@xxxxxxxxxxxxxx>");
> +MODULE_LICENSE("GPL");
> +MODULE_DESCRIPTION("CAN FD driver for Renesas R-Car SoC");
> +MODULE_ALIAS("platform:" RCANFD_DRV_NAME);

regards,
Marc

-- 
Pengutronix e.K.                  | Marc Kleine-Budde           |
Industrial Linux Solutions        | Phone: +49-231-2826-924     |
Vertretung West/Dortmund          | Fax:   +49-5121-206917-5555 |
Amtsgericht Hildesheim, HRA 2686  | http://www.pengutronix.de   |



Attachment: signature.asc
Description: OpenPGP digital signature


[Index of Archives]     [Device Tree Compilter]     [Device Tree Spec]     [Linux Driver Backports]     [Video for Linux]     [Linux USB Devel]     [Linux PCI Devel]     [Linux Audio Users]     [Linux Kernel]     [Linux SCSI]     [XFree86]     [Yosemite Backpacking]
  Powered by Linux