|  | // SPDX-License-Identifier: GPL-2.0 | 
|  | /* Copyright (c) 2018-2019, Vladimir Oltean <olteanv@gmail.com> | 
|  | */ | 
|  | #include "sja1105.h" | 
|  |  | 
|  | /* In the dynamic configuration interface, the switch exposes a register-like | 
|  | * view of some of the static configuration tables. | 
|  | * Many times the field organization of the dynamic tables is abbreviated (not | 
|  | * all fields are dynamically reconfigurable) and different from the static | 
|  | * ones, but the key reason for having it is that we can spare a switch reset | 
|  | * for settings that can be changed dynamically. | 
|  | * | 
|  | * This file creates a per-switch-family abstraction called | 
|  | * struct sja1105_dynamic_table_ops and two operations that work with it: | 
|  | * - sja1105_dynamic_config_write | 
|  | * - sja1105_dynamic_config_read | 
|  | * | 
|  | * Compared to the struct sja1105_table_ops from sja1105_static_config.c, | 
|  | * the dynamic accessors work with a compound buffer: | 
|  | * | 
|  | * packed_buf | 
|  | * | 
|  | * | | 
|  | * V | 
|  | * +-----------------------------------------+------------------+ | 
|  | * |              ENTRY BUFFER               |  COMMAND BUFFER  | | 
|  | * +-----------------------------------------+------------------+ | 
|  | * | 
|  | * <----------------------- packed_size ------------------------> | 
|  | * | 
|  | * The ENTRY BUFFER may or may not have the same layout, or size, as its static | 
|  | * configuration table entry counterpart. When it does, the same packing | 
|  | * function is reused (bar exceptional cases - see | 
|  | * sja1105pqrs_dyn_l2_lookup_entry_packing). | 
|  | * | 
|  | * The reason for the COMMAND BUFFER being at the end is to be able to send | 
|  | * a dynamic write command through a single SPI burst. By the time the switch | 
|  | * reacts to the command, the ENTRY BUFFER is already populated with the data | 
|  | * sent by the core. | 
|  | * | 
|  | * The COMMAND BUFFER is always SJA1105_SIZE_DYN_CMD bytes (one 32-bit word) in | 
|  | * size. | 
|  | * | 
|  | * Sometimes the ENTRY BUFFER does not really exist (when the number of fields | 
|  | * that can be reconfigured is small), then the switch repurposes some of the | 
|  | * unused 32 bits of the COMMAND BUFFER to hold ENTRY data. | 
|  | * | 
|  | * The key members of struct sja1105_dynamic_table_ops are: | 
|  | * - .entry_packing: A function that deals with packing an ENTRY structure | 
|  | *		     into an SPI buffer, or retrieving an ENTRY structure | 
|  | *		     from one. | 
|  | *		     The @packed_buf pointer it's given does always point to | 
|  | *		     the ENTRY portion of the buffer. | 
|  | * - .cmd_packing: A function that deals with packing/unpacking the COMMAND | 
|  | *		   structure to/from the SPI buffer. | 
|  | *		   It is given the same @packed_buf pointer as .entry_packing, | 
|  | *		   so most of the time, the @packed_buf points *behind* the | 
|  | *		   COMMAND offset inside the buffer. | 
|  | *		   To access the COMMAND portion of the buffer, the function | 
|  | *		   knows its correct offset. | 
|  | *		   Giving both functions the same pointer is handy because in | 
|  | *		   extreme cases (see sja1105pqrs_dyn_l2_lookup_entry_packing) | 
|  | *		   the .entry_packing is able to jump to the COMMAND portion, | 
|  | *		   or vice-versa (sja1105pqrs_l2_lookup_cmd_packing). | 
|  | * - .access: A bitmap of: | 
|  | *	OP_READ: Set if the hardware manual marks the ENTRY portion of the | 
|  | *		 dynamic configuration table buffer as R (readable) after | 
|  | *		 an SPI read command (the switch will populate the buffer). | 
|  | *	OP_WRITE: Set if the manual marks the ENTRY portion of the dynamic | 
|  | *		  table buffer as W (writable) after an SPI write command | 
|  | *		  (the switch will read the fields provided in the buffer). | 
|  | *	OP_DEL: Set if the manual says the VALIDENT bit is supported in the | 
|  | *		COMMAND portion of this dynamic config buffer (i.e. the | 
|  | *		specified entry can be invalidated through a SPI write | 
|  | *		command). | 
|  | *	OP_SEARCH: Set if the manual says that the index of an entry can | 
|  | *		   be retrieved in the COMMAND portion of the buffer based | 
|  | *		   on its ENTRY portion, as a result of a SPI write command. | 
|  | *		   Only the TCAM-based FDB table on SJA1105 P/Q/R/S supports | 
|  | *		   this. | 
|  | *	OP_VALID_ANYWAY: Reading some tables through the dynamic config | 
|  | *			 interface is possible even if the VALIDENT bit is not | 
|  | *			 set in the writeback. So don't error out in that case. | 
|  | * - .max_entry_count: The number of entries, counting from zero, that can be | 
|  | *		       reconfigured through the dynamic interface. If a static | 
|  | *		       table can be reconfigured at all dynamically, this | 
|  | *		       number always matches the maximum number of supported | 
|  | *		       static entries. | 
|  | * - .packed_size: The length in bytes of the compound ENTRY + COMMAND BUFFER. | 
|  | *		   Note that sometimes the compound buffer may contain holes in | 
|  | *		   it (see sja1105_vlan_lookup_cmd_packing). The @packed_buf is | 
|  | *		   contiguous however, so @packed_size includes any unused | 
|  | *		   bytes. | 
|  | * - .addr: The base SPI address at which the buffer must be written to the | 
|  | *	    switch's memory. When looking at the hardware manual, this must | 
|  | *	    always match the lowest documented address for the ENTRY, and not | 
|  | *	    that of the COMMAND, since the other 32-bit words will follow along | 
|  | *	    at the correct addresses. | 
|  | */ | 
|  |  | 
|  | #define SJA1105_SIZE_DYN_CMD					4 | 
|  |  | 
|  | #define SJA1105ET_SIZE_VL_LOOKUP_DYN_CMD			\ | 
|  | SJA1105_SIZE_DYN_CMD | 
|  |  | 
|  | #define SJA1105PQRS_SIZE_VL_LOOKUP_DYN_CMD			\ | 
|  | (SJA1105_SIZE_DYN_CMD + SJA1105_SIZE_VL_LOOKUP_ENTRY) | 
|  |  | 
|  | #define SJA1110_SIZE_VL_POLICING_DYN_CMD			\ | 
|  | (SJA1105_SIZE_DYN_CMD + SJA1105_SIZE_VL_POLICING_ENTRY) | 
|  |  | 
|  | #define SJA1105ET_SIZE_MAC_CONFIG_DYN_ENTRY			\ | 
|  | SJA1105_SIZE_DYN_CMD | 
|  |  | 
|  | #define SJA1105ET_SIZE_L2_LOOKUP_DYN_CMD			\ | 
|  | (SJA1105_SIZE_DYN_CMD + SJA1105ET_SIZE_L2_LOOKUP_ENTRY) | 
|  |  | 
|  | #define SJA1105PQRS_SIZE_L2_LOOKUP_DYN_CMD			\ | 
|  | (SJA1105_SIZE_DYN_CMD + SJA1105PQRS_SIZE_L2_LOOKUP_ENTRY) | 
|  |  | 
|  | #define SJA1110_SIZE_L2_LOOKUP_DYN_CMD				\ | 
|  | (SJA1105_SIZE_DYN_CMD + SJA1110_SIZE_L2_LOOKUP_ENTRY) | 
|  |  | 
|  | #define SJA1105_SIZE_VLAN_LOOKUP_DYN_CMD			\ | 
|  | (SJA1105_SIZE_DYN_CMD + 4 + SJA1105_SIZE_VLAN_LOOKUP_ENTRY) | 
|  |  | 
|  | #define SJA1110_SIZE_VLAN_LOOKUP_DYN_CMD			\ | 
|  | (SJA1105_SIZE_DYN_CMD + SJA1110_SIZE_VLAN_LOOKUP_ENTRY) | 
|  |  | 
|  | #define SJA1105_SIZE_L2_FORWARDING_DYN_CMD			\ | 
|  | (SJA1105_SIZE_DYN_CMD + SJA1105_SIZE_L2_FORWARDING_ENTRY) | 
|  |  | 
|  | #define SJA1105ET_SIZE_MAC_CONFIG_DYN_CMD			\ | 
|  | (SJA1105_SIZE_DYN_CMD + SJA1105ET_SIZE_MAC_CONFIG_DYN_ENTRY) | 
|  |  | 
|  | #define SJA1105PQRS_SIZE_MAC_CONFIG_DYN_CMD			\ | 
|  | (SJA1105_SIZE_DYN_CMD + SJA1105PQRS_SIZE_MAC_CONFIG_ENTRY) | 
|  |  | 
|  | #define SJA1105ET_SIZE_L2_LOOKUP_PARAMS_DYN_CMD			\ | 
|  | SJA1105_SIZE_DYN_CMD | 
|  |  | 
|  | #define SJA1105PQRS_SIZE_L2_LOOKUP_PARAMS_DYN_CMD		\ | 
|  | (SJA1105_SIZE_DYN_CMD + SJA1105PQRS_SIZE_L2_LOOKUP_PARAMS_ENTRY) | 
|  |  | 
|  | #define SJA1110_SIZE_L2_LOOKUP_PARAMS_DYN_CMD		\ | 
|  | (SJA1105_SIZE_DYN_CMD + SJA1110_SIZE_L2_LOOKUP_PARAMS_ENTRY) | 
|  |  | 
|  | #define SJA1105ET_SIZE_GENERAL_PARAMS_DYN_CMD			\ | 
|  | SJA1105_SIZE_DYN_CMD | 
|  |  | 
|  | #define SJA1105PQRS_SIZE_GENERAL_PARAMS_DYN_CMD			\ | 
|  | (SJA1105_SIZE_DYN_CMD + SJA1105PQRS_SIZE_GENERAL_PARAMS_ENTRY) | 
|  |  | 
|  | #define SJA1110_SIZE_GENERAL_PARAMS_DYN_CMD			\ | 
|  | (SJA1105_SIZE_DYN_CMD + SJA1110_SIZE_GENERAL_PARAMS_ENTRY) | 
|  |  | 
|  | #define SJA1105PQRS_SIZE_AVB_PARAMS_DYN_CMD			\ | 
|  | (SJA1105_SIZE_DYN_CMD + SJA1105PQRS_SIZE_AVB_PARAMS_ENTRY) | 
|  |  | 
|  | #define SJA1105_SIZE_RETAGGING_DYN_CMD				\ | 
|  | (SJA1105_SIZE_DYN_CMD + SJA1105_SIZE_RETAGGING_ENTRY) | 
|  |  | 
|  | #define SJA1105ET_SIZE_CBS_DYN_CMD				\ | 
|  | (SJA1105_SIZE_DYN_CMD + SJA1105ET_SIZE_CBS_ENTRY) | 
|  |  | 
|  | #define SJA1105PQRS_SIZE_CBS_DYN_CMD				\ | 
|  | (SJA1105_SIZE_DYN_CMD + SJA1105PQRS_SIZE_CBS_ENTRY) | 
|  |  | 
|  | #define SJA1110_SIZE_XMII_PARAMS_DYN_CMD			\ | 
|  | SJA1110_SIZE_XMII_PARAMS_ENTRY | 
|  |  | 
|  | #define SJA1110_SIZE_L2_POLICING_DYN_CMD			\ | 
|  | (SJA1105_SIZE_DYN_CMD + SJA1105_SIZE_L2_POLICING_ENTRY) | 
|  |  | 
|  | #define SJA1110_SIZE_L2_FORWARDING_PARAMS_DYN_CMD		\ | 
|  | SJA1105_SIZE_L2_FORWARDING_PARAMS_ENTRY | 
|  |  | 
|  | #define SJA1105_MAX_DYN_CMD_SIZE				\ | 
|  | SJA1110_SIZE_GENERAL_PARAMS_DYN_CMD | 
|  |  | 
|  | struct sja1105_dyn_cmd { | 
|  | bool search; | 
|  | u64 valid; | 
|  | u64 rdwrset; | 
|  | u64 errors; | 
|  | u64 valident; | 
|  | u64 index; | 
|  | }; | 
|  |  | 
|  | enum sja1105_hostcmd { | 
|  | SJA1105_HOSTCMD_SEARCH = 1, | 
|  | SJA1105_HOSTCMD_READ = 2, | 
|  | SJA1105_HOSTCMD_WRITE = 3, | 
|  | SJA1105_HOSTCMD_INVALIDATE = 4, | 
|  | }; | 
|  |  | 
|  | /* Command and entry overlap */ | 
|  | static void | 
|  | sja1105et_vl_lookup_cmd_packing(void *buf, struct sja1105_dyn_cmd *cmd, | 
|  | enum packing_op op) | 
|  | { | 
|  | const int size = SJA1105_SIZE_DYN_CMD; | 
|  |  | 
|  | sja1105_packing(buf, &cmd->valid,   31, 31, size, op); | 
|  | sja1105_packing(buf, &cmd->errors,  30, 30, size, op); | 
|  | sja1105_packing(buf, &cmd->rdwrset, 29, 29, size, op); | 
|  | sja1105_packing(buf, &cmd->index,    9,  0, size, op); | 
|  | } | 
|  |  | 
|  | /* Command and entry are separate */ | 
|  | static void | 
|  | sja1105pqrs_vl_lookup_cmd_packing(void *buf, struct sja1105_dyn_cmd *cmd, | 
|  | enum packing_op op) | 
|  | { | 
|  | u8 *p = buf + SJA1105_SIZE_VL_LOOKUP_ENTRY; | 
|  | const int size = SJA1105_SIZE_DYN_CMD; | 
|  |  | 
|  | sja1105_packing(p, &cmd->valid,   31, 31, size, op); | 
|  | sja1105_packing(p, &cmd->errors,  30, 30, size, op); | 
|  | sja1105_packing(p, &cmd->rdwrset, 29, 29, size, op); | 
|  | sja1105_packing(p, &cmd->index,    9,  0, size, op); | 
|  | } | 
|  |  | 
|  | static void | 
|  | sja1110_vl_lookup_cmd_packing(void *buf, struct sja1105_dyn_cmd *cmd, | 
|  | enum packing_op op) | 
|  | { | 
|  | u8 *p = buf + SJA1105PQRS_SIZE_L2_LOOKUP_ENTRY; | 
|  | const int size = SJA1105_SIZE_DYN_CMD; | 
|  |  | 
|  | sja1105_packing(p, &cmd->valid,   31, 31, size, op); | 
|  | sja1105_packing(p, &cmd->rdwrset, 30, 30, size, op); | 
|  | sja1105_packing(p, &cmd->errors,  29, 29, size, op); | 
|  | sja1105_packing(p, &cmd->index,   11,  0, size, op); | 
|  | } | 
|  |  | 
|  | static size_t sja1105et_vl_lookup_entry_packing(void *buf, void *entry_ptr, | 
|  | enum packing_op op) | 
|  | { | 
|  | struct sja1105_vl_lookup_entry *entry = entry_ptr; | 
|  | const int size = SJA1105ET_SIZE_VL_LOOKUP_DYN_CMD; | 
|  |  | 
|  | sja1105_packing(buf, &entry->egrmirr,  21, 17, size, op); | 
|  | sja1105_packing(buf, &entry->ingrmirr, 16, 16, size, op); | 
|  | return size; | 
|  | } | 
|  |  | 
|  | static void | 
|  | sja1110_vl_policing_cmd_packing(void *buf, struct sja1105_dyn_cmd *cmd, | 
|  | enum packing_op op) | 
|  | { | 
|  | u8 *p = buf + SJA1105_SIZE_VL_LOOKUP_ENTRY; | 
|  | const int size = SJA1105_SIZE_DYN_CMD; | 
|  |  | 
|  | sja1105_packing(p, &cmd->valid,   31, 31, size, op); | 
|  | sja1105_packing(p, &cmd->rdwrset, 30, 30, size, op); | 
|  | sja1105_packing(p, &cmd->index,   11,  0, size, op); | 
|  | } | 
|  |  | 
|  | static void | 
|  | sja1105pqrs_common_l2_lookup_cmd_packing(void *buf, struct sja1105_dyn_cmd *cmd, | 
|  | enum packing_op op, int entry_size) | 
|  | { | 
|  | const int size = SJA1105_SIZE_DYN_CMD; | 
|  | u8 *p = buf + entry_size; | 
|  | u64 hostcmd; | 
|  |  | 
|  | sja1105_packing(p, &cmd->valid,    31, 31, size, op); | 
|  | sja1105_packing(p, &cmd->rdwrset,  30, 30, size, op); | 
|  | sja1105_packing(p, &cmd->errors,   29, 29, size, op); | 
|  | sja1105_packing(p, &cmd->valident, 27, 27, size, op); | 
|  |  | 
|  | /* VALIDENT is supposed to indicate "keep or not", but in SJA1105 E/T, | 
|  | * using it to delete a management route was unsupported. UM10944 | 
|  | * said about it: | 
|  | * | 
|  | *   In case of a write access with the MGMTROUTE flag set, | 
|  | *   the flag will be ignored. It will always be found cleared | 
|  | *   for read accesses with the MGMTROUTE flag set. | 
|  | * | 
|  | * SJA1105 P/Q/R/S keeps the same behavior w.r.t. VALIDENT, but there | 
|  | * is now another flag called HOSTCMD which does more stuff (quoting | 
|  | * from UM11040): | 
|  | * | 
|  | *   A write request is accepted only when HOSTCMD is set to write host | 
|  | *   or invalid. A read request is accepted only when HOSTCMD is set to | 
|  | *   search host or read host. | 
|  | * | 
|  | * So it is possible to translate a RDWRSET/VALIDENT combination into | 
|  | * HOSTCMD so that we keep the dynamic command API in place, and | 
|  | * at the same time achieve compatibility with the management route | 
|  | * command structure. | 
|  | */ | 
|  | if (cmd->rdwrset == SPI_READ) { | 
|  | if (cmd->search) | 
|  | hostcmd = SJA1105_HOSTCMD_SEARCH; | 
|  | else | 
|  | hostcmd = SJA1105_HOSTCMD_READ; | 
|  | } else { | 
|  | /* SPI_WRITE */ | 
|  | if (cmd->valident) | 
|  | hostcmd = SJA1105_HOSTCMD_WRITE; | 
|  | else | 
|  | hostcmd = SJA1105_HOSTCMD_INVALIDATE; | 
|  | } | 
|  | sja1105_packing(p, &hostcmd, 25, 23, size, op); | 
|  | } | 
|  |  | 
|  | static void | 
|  | sja1105pqrs_l2_lookup_cmd_packing(void *buf, struct sja1105_dyn_cmd *cmd, | 
|  | enum packing_op op) | 
|  | { | 
|  | int entry_size = SJA1105PQRS_SIZE_L2_LOOKUP_ENTRY; | 
|  |  | 
|  | sja1105pqrs_common_l2_lookup_cmd_packing(buf, cmd, op, entry_size); | 
|  |  | 
|  | /* Hack - The hardware takes the 'index' field within | 
|  | * struct sja1105_l2_lookup_entry as the index on which this command | 
|  | * will operate. However it will ignore everything else, so 'index' | 
|  | * is logically part of command but physically part of entry. | 
|  | * Populate the 'index' entry field from within the command callback, | 
|  | * such that our API doesn't need to ask for a full-blown entry | 
|  | * structure when e.g. a delete is requested. | 
|  | */ | 
|  | sja1105_packing(buf, &cmd->index, 15, 6, entry_size, op); | 
|  | } | 
|  |  | 
|  | static void | 
|  | sja1110_l2_lookup_cmd_packing(void *buf, struct sja1105_dyn_cmd *cmd, | 
|  | enum packing_op op) | 
|  | { | 
|  | int entry_size = SJA1110_SIZE_L2_LOOKUP_ENTRY; | 
|  |  | 
|  | sja1105pqrs_common_l2_lookup_cmd_packing(buf, cmd, op, entry_size); | 
|  |  | 
|  | sja1105_packing(buf, &cmd->index, 10, 1, entry_size, op); | 
|  | } | 
|  |  | 
|  | /* The switch is so retarded that it makes our command/entry abstraction | 
|  | * crumble apart. | 
|  | * | 
|  | * On P/Q/R/S, the switch tries to say whether a FDB entry | 
|  | * is statically programmed or dynamically learned via a flag called LOCKEDS. | 
|  | * The hardware manual says about this fiels: | 
|  | * | 
|  | *   On write will specify the format of ENTRY. | 
|  | *   On read the flag will be found cleared at times the VALID flag is found | 
|  | *   set.  The flag will also be found cleared in response to a read having the | 
|  | *   MGMTROUTE flag set.  In response to a read with the MGMTROUTE flag | 
|  | *   cleared, the flag be set if the most recent access operated on an entry | 
|  | *   that was either loaded by configuration or through dynamic reconfiguration | 
|  | *   (as opposed to automatically learned entries). | 
|  | * | 
|  | * The trouble with this flag is that it's part of the *command* to access the | 
|  | * dynamic interface, and not part of the *entry* retrieved from it. | 
|  | * Otherwise said, for a sja1105_dynamic_config_read, LOCKEDS is supposed to be | 
|  | * an output from the switch into the command buffer, and for a | 
|  | * sja1105_dynamic_config_write, the switch treats LOCKEDS as an input | 
|  | * (hence we can write either static, or automatically learned entries, from | 
|  | * the core). | 
|  | * But the manual contradicts itself in the last phrase where it says that on | 
|  | * read, LOCKEDS will be set to 1 for all FDB entries written through the | 
|  | * dynamic interface (therefore, the value of LOCKEDS from the | 
|  | * sja1105_dynamic_config_write is not really used for anything, it'll store a | 
|  | * 1 anyway). | 
|  | * This means you can't really write a FDB entry with LOCKEDS=0 (automatically | 
|  | * learned) into the switch, which kind of makes sense. | 
|  | * As for reading through the dynamic interface, it doesn't make too much sense | 
|  | * to put LOCKEDS into the command, since the switch will inevitably have to | 
|  | * ignore it (otherwise a command would be like "read the FDB entry 123, but | 
|  | * only if it's dynamically learned" <- well how am I supposed to know?) and | 
|  | * just use it as an output buffer for its findings. But guess what... that's | 
|  | * what the entry buffer is for! | 
|  | * Unfortunately, what really breaks this abstraction is the fact that it | 
|  | * wasn't designed having the fact in mind that the switch can output | 
|  | * entry-related data as writeback through the command buffer. | 
|  | * However, whether a FDB entry is statically or dynamically learned *is* part | 
|  | * of the entry and not the command data, no matter what the switch thinks. | 
|  | * In order to do that, we'll need to wrap around the | 
|  | * sja1105pqrs_l2_lookup_entry_packing from sja1105_static_config.c, and take | 
|  | * a peek outside of the caller-supplied @buf (the entry buffer), to reach the | 
|  | * command buffer. | 
|  | */ | 
|  | static size_t | 
|  | sja1105pqrs_dyn_l2_lookup_entry_packing(void *buf, void *entry_ptr, | 
|  | enum packing_op op) | 
|  | { | 
|  | struct sja1105_l2_lookup_entry *entry = entry_ptr; | 
|  | u8 *cmd = buf + SJA1105PQRS_SIZE_L2_LOOKUP_ENTRY; | 
|  | const int size = SJA1105_SIZE_DYN_CMD; | 
|  |  | 
|  | sja1105_packing(cmd, &entry->lockeds, 28, 28, size, op); | 
|  |  | 
|  | return sja1105pqrs_l2_lookup_entry_packing(buf, entry_ptr, op); | 
|  | } | 
|  |  | 
|  | static size_t sja1110_dyn_l2_lookup_entry_packing(void *buf, void *entry_ptr, | 
|  | enum packing_op op) | 
|  | { | 
|  | struct sja1105_l2_lookup_entry *entry = entry_ptr; | 
|  | u8 *cmd = buf + SJA1110_SIZE_L2_LOOKUP_ENTRY; | 
|  | const int size = SJA1105_SIZE_DYN_CMD; | 
|  |  | 
|  | sja1105_packing(cmd, &entry->lockeds, 28, 28, size, op); | 
|  |  | 
|  | return sja1110_l2_lookup_entry_packing(buf, entry_ptr, op); | 
|  | } | 
|  |  | 
|  | static void | 
|  | sja1105et_l2_lookup_cmd_packing(void *buf, struct sja1105_dyn_cmd *cmd, | 
|  | enum packing_op op) | 
|  | { | 
|  | u8 *p = buf + SJA1105ET_SIZE_L2_LOOKUP_ENTRY; | 
|  | const int size = SJA1105_SIZE_DYN_CMD; | 
|  |  | 
|  | sja1105_packing(p, &cmd->valid,    31, 31, size, op); | 
|  | sja1105_packing(p, &cmd->rdwrset,  30, 30, size, op); | 
|  | sja1105_packing(p, &cmd->errors,   29, 29, size, op); | 
|  | sja1105_packing(p, &cmd->valident, 27, 27, size, op); | 
|  | /* Hack - see comments above. */ | 
|  | sja1105_packing(buf, &cmd->index, 29, 20, | 
|  | SJA1105ET_SIZE_L2_LOOKUP_ENTRY, op); | 
|  | } | 
|  |  | 
|  | static size_t sja1105et_dyn_l2_lookup_entry_packing(void *buf, void *entry_ptr, | 
|  | enum packing_op op) | 
|  | { | 
|  | struct sja1105_l2_lookup_entry *entry = entry_ptr; | 
|  | u8 *cmd = buf + SJA1105ET_SIZE_L2_LOOKUP_ENTRY; | 
|  | const int size = SJA1105_SIZE_DYN_CMD; | 
|  |  | 
|  | sja1105_packing(cmd, &entry->lockeds, 28, 28, size, op); | 
|  |  | 
|  | return sja1105et_l2_lookup_entry_packing(buf, entry_ptr, op); | 
|  | } | 
|  |  | 
|  | static void | 
|  | sja1105et_mgmt_route_cmd_packing(void *buf, struct sja1105_dyn_cmd *cmd, | 
|  | enum packing_op op) | 
|  | { | 
|  | u8 *p = buf + SJA1105ET_SIZE_L2_LOOKUP_ENTRY; | 
|  | u64 mgmtroute = 1; | 
|  |  | 
|  | sja1105et_l2_lookup_cmd_packing(buf, cmd, op); | 
|  | if (op == PACK) | 
|  | sja1105_pack(p, &mgmtroute, 26, 26, SJA1105_SIZE_DYN_CMD); | 
|  | } | 
|  |  | 
|  | static size_t sja1105et_mgmt_route_entry_packing(void *buf, void *entry_ptr, | 
|  | enum packing_op op) | 
|  | { | 
|  | struct sja1105_mgmt_entry *entry = entry_ptr; | 
|  | const size_t size = SJA1105ET_SIZE_L2_LOOKUP_ENTRY; | 
|  |  | 
|  | /* UM10944: To specify if a PTP egress timestamp shall be captured on | 
|  | * each port upon transmission of the frame, the LSB of VLANID in the | 
|  | * ENTRY field provided by the host must be set. | 
|  | * Bit 1 of VLANID then specifies the register where the timestamp for | 
|  | * this port is stored in. | 
|  | */ | 
|  | sja1105_packing(buf, &entry->tsreg,     85, 85, size, op); | 
|  | sja1105_packing(buf, &entry->takets,    84, 84, size, op); | 
|  | sja1105_packing(buf, &entry->macaddr,   83, 36, size, op); | 
|  | sja1105_packing(buf, &entry->destports, 35, 31, size, op); | 
|  | sja1105_packing(buf, &entry->enfport,   30, 30, size, op); | 
|  | return size; | 
|  | } | 
|  |  | 
|  | static void | 
|  | sja1105pqrs_mgmt_route_cmd_packing(void *buf, struct sja1105_dyn_cmd *cmd, | 
|  | enum packing_op op) | 
|  | { | 
|  | u8 *p = buf + SJA1105PQRS_SIZE_L2_LOOKUP_ENTRY; | 
|  | u64 mgmtroute = 1; | 
|  |  | 
|  | sja1105pqrs_l2_lookup_cmd_packing(buf, cmd, op); | 
|  | if (op == PACK) | 
|  | sja1105_pack(p, &mgmtroute, 26, 26, SJA1105_SIZE_DYN_CMD); | 
|  | } | 
|  |  | 
|  | static size_t sja1105pqrs_mgmt_route_entry_packing(void *buf, void *entry_ptr, | 
|  | enum packing_op op) | 
|  | { | 
|  | const size_t size = SJA1105PQRS_SIZE_L2_LOOKUP_ENTRY; | 
|  | struct sja1105_mgmt_entry *entry = entry_ptr; | 
|  |  | 
|  | /* In P/Q/R/S, enfport got renamed to mgmtvalid, but its purpose | 
|  | * is the same (driver uses it to confirm that frame was sent). | 
|  | * So just keep the name from E/T. | 
|  | */ | 
|  | sja1105_packing(buf, &entry->tsreg,     71, 71, size, op); | 
|  | sja1105_packing(buf, &entry->takets,    70, 70, size, op); | 
|  | sja1105_packing(buf, &entry->macaddr,   69, 22, size, op); | 
|  | sja1105_packing(buf, &entry->destports, 21, 17, size, op); | 
|  | sja1105_packing(buf, &entry->enfport,   16, 16, size, op); | 
|  | return size; | 
|  | } | 
|  |  | 
|  | /* In E/T, entry is at addresses 0x27-0x28. There is a 4 byte gap at 0x29, | 
|  | * and command is at 0x2a. Similarly in P/Q/R/S there is a 1 register gap | 
|  | * between entry (0x2d, 0x2e) and command (0x30). | 
|  | */ | 
|  | static void | 
|  | sja1105_vlan_lookup_cmd_packing(void *buf, struct sja1105_dyn_cmd *cmd, | 
|  | enum packing_op op) | 
|  | { | 
|  | u8 *p = buf + SJA1105_SIZE_VLAN_LOOKUP_ENTRY + 4; | 
|  | const int size = SJA1105_SIZE_DYN_CMD; | 
|  |  | 
|  | sja1105_packing(p, &cmd->valid,    31, 31, size, op); | 
|  | sja1105_packing(p, &cmd->rdwrset,  30, 30, size, op); | 
|  | sja1105_packing(p, &cmd->valident, 27, 27, size, op); | 
|  | /* Hack - see comments above, applied for 'vlanid' field of | 
|  | * struct sja1105_vlan_lookup_entry. | 
|  | */ | 
|  | sja1105_packing(buf, &cmd->index, 38, 27, | 
|  | SJA1105_SIZE_VLAN_LOOKUP_ENTRY, op); | 
|  | } | 
|  |  | 
|  | /* In SJA1110 there is no gap between the command and the data, yay... */ | 
|  | static void | 
|  | sja1110_vlan_lookup_cmd_packing(void *buf, struct sja1105_dyn_cmd *cmd, | 
|  | enum packing_op op) | 
|  | { | 
|  | u8 *p = buf + SJA1110_SIZE_VLAN_LOOKUP_ENTRY; | 
|  | const int size = SJA1105_SIZE_DYN_CMD; | 
|  | u64 type_entry = 0; | 
|  |  | 
|  | sja1105_packing(p, &cmd->valid,   31, 31, size, op); | 
|  | sja1105_packing(p, &cmd->rdwrset, 30, 30, size, op); | 
|  | sja1105_packing(p, &cmd->errors,  29, 29, size, op); | 
|  | /* Hack: treat 'vlanid' field of struct sja1105_vlan_lookup_entry as | 
|  | * cmd->index. | 
|  | */ | 
|  | sja1105_packing(buf, &cmd->index, 38, 27, | 
|  | SJA1110_SIZE_VLAN_LOOKUP_ENTRY, op); | 
|  |  | 
|  | /* But the VALIDENT bit has disappeared, now we are supposed to | 
|  | * invalidate an entry through the TYPE_ENTRY field of the entry.. | 
|  | * This is a hack to transform the non-zero quality of the TYPE_ENTRY | 
|  | * field into a VALIDENT bit. | 
|  | */ | 
|  | if (op == PACK && !cmd->valident) { | 
|  | sja1105_packing(buf, &type_entry, 40, 39, | 
|  | SJA1110_SIZE_VLAN_LOOKUP_ENTRY, PACK); | 
|  | } else if (op == UNPACK) { | 
|  | sja1105_packing(buf, &type_entry, 40, 39, | 
|  | SJA1110_SIZE_VLAN_LOOKUP_ENTRY, UNPACK); | 
|  | cmd->valident = !!type_entry; | 
|  | } | 
|  | } | 
|  |  | 
|  | static void | 
|  | sja1105_l2_forwarding_cmd_packing(void *buf, struct sja1105_dyn_cmd *cmd, | 
|  | enum packing_op op) | 
|  | { | 
|  | u8 *p = buf + SJA1105_SIZE_L2_FORWARDING_ENTRY; | 
|  | const int size = SJA1105_SIZE_DYN_CMD; | 
|  |  | 
|  | sja1105_packing(p, &cmd->valid,   31, 31, size, op); | 
|  | sja1105_packing(p, &cmd->errors,  30, 30, size, op); | 
|  | sja1105_packing(p, &cmd->rdwrset, 29, 29, size, op); | 
|  | sja1105_packing(p, &cmd->index,    4,  0, size, op); | 
|  | } | 
|  |  | 
|  | static void | 
|  | sja1110_l2_forwarding_cmd_packing(void *buf, struct sja1105_dyn_cmd *cmd, | 
|  | enum packing_op op) | 
|  | { | 
|  | u8 *p = buf + SJA1105_SIZE_L2_FORWARDING_ENTRY; | 
|  | const int size = SJA1105_SIZE_DYN_CMD; | 
|  |  | 
|  | sja1105_packing(p, &cmd->valid,   31, 31, size, op); | 
|  | sja1105_packing(p, &cmd->rdwrset, 30, 30, size, op); | 
|  | sja1105_packing(p, &cmd->errors,  29, 29, size, op); | 
|  | sja1105_packing(p, &cmd->index,    4,  0, size, op); | 
|  | } | 
|  |  | 
|  | static void | 
|  | sja1105et_mac_config_cmd_packing(void *buf, struct sja1105_dyn_cmd *cmd, | 
|  | enum packing_op op) | 
|  | { | 
|  | const int size = SJA1105_SIZE_DYN_CMD; | 
|  | /* Yup, user manual definitions are reversed */ | 
|  | u8 *reg1 = buf + 4; | 
|  |  | 
|  | sja1105_packing(reg1, &cmd->valid, 31, 31, size, op); | 
|  | sja1105_packing(reg1, &cmd->index, 26, 24, size, op); | 
|  | } | 
|  |  | 
|  | static size_t sja1105et_mac_config_entry_packing(void *buf, void *entry_ptr, | 
|  | enum packing_op op) | 
|  | { | 
|  | const int size = SJA1105ET_SIZE_MAC_CONFIG_DYN_ENTRY; | 
|  | struct sja1105_mac_config_entry *entry = entry_ptr; | 
|  | /* Yup, user manual definitions are reversed */ | 
|  | u8 *reg1 = buf + 4; | 
|  | u8 *reg2 = buf; | 
|  |  | 
|  | sja1105_packing(reg1, &entry->speed,     30, 29, size, op); | 
|  | sja1105_packing(reg1, &entry->drpdtag,   23, 23, size, op); | 
|  | sja1105_packing(reg1, &entry->drpuntag,  22, 22, size, op); | 
|  | sja1105_packing(reg1, &entry->retag,     21, 21, size, op); | 
|  | sja1105_packing(reg1, &entry->dyn_learn, 20, 20, size, op); | 
|  | sja1105_packing(reg1, &entry->egress,    19, 19, size, op); | 
|  | sja1105_packing(reg1, &entry->ingress,   18, 18, size, op); | 
|  | sja1105_packing(reg1, &entry->ing_mirr,  17, 17, size, op); | 
|  | sja1105_packing(reg1, &entry->egr_mirr,  16, 16, size, op); | 
|  | sja1105_packing(reg1, &entry->vlanprio,  14, 12, size, op); | 
|  | sja1105_packing(reg1, &entry->vlanid,    11,  0, size, op); | 
|  | sja1105_packing(reg2, &entry->tp_delin,  31, 16, size, op); | 
|  | sja1105_packing(reg2, &entry->tp_delout, 15,  0, size, op); | 
|  | /* MAC configuration table entries which can't be reconfigured: | 
|  | * top, base, enabled, ifg, maxage, drpnona664 | 
|  | */ | 
|  | /* Bogus return value, not used anywhere */ | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | static void | 
|  | sja1105pqrs_mac_config_cmd_packing(void *buf, struct sja1105_dyn_cmd *cmd, | 
|  | enum packing_op op) | 
|  | { | 
|  | const int size = SJA1105ET_SIZE_MAC_CONFIG_DYN_ENTRY; | 
|  | u8 *p = buf + SJA1105PQRS_SIZE_MAC_CONFIG_ENTRY; | 
|  |  | 
|  | sja1105_packing(p, &cmd->valid,   31, 31, size, op); | 
|  | sja1105_packing(p, &cmd->errors,  30, 30, size, op); | 
|  | sja1105_packing(p, &cmd->rdwrset, 29, 29, size, op); | 
|  | sja1105_packing(p, &cmd->index,    2,  0, size, op); | 
|  | } | 
|  |  | 
|  | static void | 
|  | sja1110_mac_config_cmd_packing(void *buf, struct sja1105_dyn_cmd *cmd, | 
|  | enum packing_op op) | 
|  | { | 
|  | u8 *p = buf + SJA1105PQRS_SIZE_MAC_CONFIG_ENTRY; | 
|  | const int size = SJA1105_SIZE_DYN_CMD; | 
|  |  | 
|  | sja1105_packing(p, &cmd->valid,   31, 31, size, op); | 
|  | sja1105_packing(p, &cmd->rdwrset, 30, 30, size, op); | 
|  | sja1105_packing(p, &cmd->errors,  29, 29, size, op); | 
|  | sja1105_packing(p, &cmd->index,    3,  0, size, op); | 
|  | } | 
|  |  | 
|  | static void | 
|  | sja1105et_l2_lookup_params_cmd_packing(void *buf, struct sja1105_dyn_cmd *cmd, | 
|  | enum packing_op op) | 
|  | { | 
|  | sja1105_packing(buf, &cmd->valid, 31, 31, | 
|  | SJA1105ET_SIZE_L2_LOOKUP_PARAMS_DYN_CMD, op); | 
|  | } | 
|  |  | 
|  | static size_t | 
|  | sja1105et_l2_lookup_params_entry_packing(void *buf, void *entry_ptr, | 
|  | enum packing_op op) | 
|  | { | 
|  | struct sja1105_l2_lookup_params_entry *entry = entry_ptr; | 
|  |  | 
|  | sja1105_packing(buf, &entry->poly, 7, 0, | 
|  | SJA1105ET_SIZE_L2_LOOKUP_PARAMS_DYN_CMD, op); | 
|  | /* Bogus return value, not used anywhere */ | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | static void | 
|  | sja1105pqrs_l2_lookup_params_cmd_packing(void *buf, | 
|  | struct sja1105_dyn_cmd *cmd, | 
|  | enum packing_op op) | 
|  | { | 
|  | u8 *p = buf + SJA1105PQRS_SIZE_L2_LOOKUP_PARAMS_ENTRY; | 
|  | const int size = SJA1105_SIZE_DYN_CMD; | 
|  |  | 
|  | sja1105_packing(p, &cmd->valid,   31, 31, size, op); | 
|  | sja1105_packing(p, &cmd->rdwrset, 30, 30, size, op); | 
|  | } | 
|  |  | 
|  | static void | 
|  | sja1110_l2_lookup_params_cmd_packing(void *buf, struct sja1105_dyn_cmd *cmd, | 
|  | enum packing_op op) | 
|  | { | 
|  | u8 *p = buf + SJA1110_SIZE_L2_LOOKUP_PARAMS_ENTRY; | 
|  | const int size = SJA1105_SIZE_DYN_CMD; | 
|  |  | 
|  | sja1105_packing(p, &cmd->valid,   31, 31, size, op); | 
|  | sja1105_packing(p, &cmd->rdwrset, 30, 30, size, op); | 
|  | sja1105_packing(p, &cmd->errors,  29, 29, size, op); | 
|  | } | 
|  |  | 
|  | static void | 
|  | sja1105et_general_params_cmd_packing(void *buf, struct sja1105_dyn_cmd *cmd, | 
|  | enum packing_op op) | 
|  | { | 
|  | const int size = SJA1105ET_SIZE_GENERAL_PARAMS_DYN_CMD; | 
|  |  | 
|  | sja1105_packing(buf, &cmd->valid,  31, 31, size, op); | 
|  | sja1105_packing(buf, &cmd->errors, 30, 30, size, op); | 
|  | } | 
|  |  | 
|  | static size_t | 
|  | sja1105et_general_params_entry_packing(void *buf, void *entry_ptr, | 
|  | enum packing_op op) | 
|  | { | 
|  | struct sja1105_general_params_entry *entry = entry_ptr; | 
|  | const int size = SJA1105ET_SIZE_GENERAL_PARAMS_DYN_CMD; | 
|  |  | 
|  | sja1105_packing(buf, &entry->mirr_port, 2, 0, size, op); | 
|  | /* Bogus return value, not used anywhere */ | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | static void | 
|  | sja1105pqrs_general_params_cmd_packing(void *buf, struct sja1105_dyn_cmd *cmd, | 
|  | enum packing_op op) | 
|  | { | 
|  | u8 *p = buf + SJA1105PQRS_SIZE_GENERAL_PARAMS_ENTRY; | 
|  | const int size = SJA1105_SIZE_DYN_CMD; | 
|  |  | 
|  | sja1105_packing(p, &cmd->valid,   31, 31, size, op); | 
|  | sja1105_packing(p, &cmd->errors,  30, 30, size, op); | 
|  | sja1105_packing(p, &cmd->rdwrset, 28, 28, size, op); | 
|  | } | 
|  |  | 
|  | static void | 
|  | sja1110_general_params_cmd_packing(void *buf, struct sja1105_dyn_cmd *cmd, | 
|  | enum packing_op op) | 
|  | { | 
|  | u8 *p = buf + SJA1110_SIZE_GENERAL_PARAMS_ENTRY; | 
|  | const int size = SJA1105_SIZE_DYN_CMD; | 
|  |  | 
|  | sja1105_packing(p, &cmd->valid,   31, 31, size, op); | 
|  | sja1105_packing(p, &cmd->rdwrset, 30, 30, size, op); | 
|  | sja1105_packing(p, &cmd->errors,  29, 29, size, op); | 
|  | } | 
|  |  | 
|  | static void | 
|  | sja1105pqrs_avb_params_cmd_packing(void *buf, struct sja1105_dyn_cmd *cmd, | 
|  | enum packing_op op) | 
|  | { | 
|  | u8 *p = buf + SJA1105PQRS_SIZE_AVB_PARAMS_ENTRY; | 
|  | const int size = SJA1105_SIZE_DYN_CMD; | 
|  |  | 
|  | sja1105_packing(p, &cmd->valid,   31, 31, size, op); | 
|  | sja1105_packing(p, &cmd->errors,  30, 30, size, op); | 
|  | sja1105_packing(p, &cmd->rdwrset, 29, 29, size, op); | 
|  | } | 
|  |  | 
|  | static void | 
|  | sja1105_retagging_cmd_packing(void *buf, struct sja1105_dyn_cmd *cmd, | 
|  | enum packing_op op) | 
|  | { | 
|  | u8 *p = buf + SJA1105_SIZE_RETAGGING_ENTRY; | 
|  | const int size = SJA1105_SIZE_DYN_CMD; | 
|  |  | 
|  | sja1105_packing(p, &cmd->valid,    31, 31, size, op); | 
|  | sja1105_packing(p, &cmd->errors,   30, 30, size, op); | 
|  | sja1105_packing(p, &cmd->valident, 29, 29, size, op); | 
|  | sja1105_packing(p, &cmd->rdwrset,  28, 28, size, op); | 
|  | sja1105_packing(p, &cmd->index,     5,  0, size, op); | 
|  | } | 
|  |  | 
|  | static void | 
|  | sja1110_retagging_cmd_packing(void *buf, struct sja1105_dyn_cmd *cmd, | 
|  | enum packing_op op) | 
|  | { | 
|  | u8 *p = buf + SJA1105_SIZE_RETAGGING_ENTRY; | 
|  | const int size = SJA1105_SIZE_DYN_CMD; | 
|  |  | 
|  | sja1105_packing(p, &cmd->valid,    31, 31, size, op); | 
|  | sja1105_packing(p, &cmd->rdwrset,  30, 30, size, op); | 
|  | sja1105_packing(p, &cmd->errors,   29, 29, size, op); | 
|  | sja1105_packing(p, &cmd->valident, 28, 28, size, op); | 
|  | sja1105_packing(p, &cmd->index,     4,  0, size, op); | 
|  | } | 
|  |  | 
|  | static void sja1105et_cbs_cmd_packing(void *buf, struct sja1105_dyn_cmd *cmd, | 
|  | enum packing_op op) | 
|  | { | 
|  | u8 *p = buf + SJA1105ET_SIZE_CBS_ENTRY; | 
|  | const int size = SJA1105_SIZE_DYN_CMD; | 
|  |  | 
|  | sja1105_packing(p, &cmd->valid, 31, 31, size, op); | 
|  | sja1105_packing(p, &cmd->index, 19, 16, size, op); | 
|  | } | 
|  |  | 
|  | static size_t sja1105et_cbs_entry_packing(void *buf, void *entry_ptr, | 
|  | enum packing_op op) | 
|  | { | 
|  | const size_t size = SJA1105ET_SIZE_CBS_ENTRY; | 
|  | struct sja1105_cbs_entry *entry = entry_ptr; | 
|  | u8 *cmd = buf + size; | 
|  | u32 *p = buf; | 
|  |  | 
|  | sja1105_packing(cmd, &entry->port, 5, 3, SJA1105_SIZE_DYN_CMD, op); | 
|  | sja1105_packing(cmd, &entry->prio, 2, 0, SJA1105_SIZE_DYN_CMD, op); | 
|  | sja1105_packing(p + 3, &entry->credit_lo,  31, 0, size, op); | 
|  | sja1105_packing(p + 2, &entry->credit_hi,  31, 0, size, op); | 
|  | sja1105_packing(p + 1, &entry->send_slope, 31, 0, size, op); | 
|  | sja1105_packing(p + 0, &entry->idle_slope, 31, 0, size, op); | 
|  | return size; | 
|  | } | 
|  |  | 
|  | static void sja1105pqrs_cbs_cmd_packing(void *buf, struct sja1105_dyn_cmd *cmd, | 
|  | enum packing_op op) | 
|  | { | 
|  | u8 *p = buf + SJA1105PQRS_SIZE_CBS_ENTRY; | 
|  | const int size = SJA1105_SIZE_DYN_CMD; | 
|  |  | 
|  | sja1105_packing(p, &cmd->valid,   31, 31, size, op); | 
|  | sja1105_packing(p, &cmd->rdwrset, 30, 30, size, op); | 
|  | sja1105_packing(p, &cmd->errors,  29, 29, size, op); | 
|  | sja1105_packing(p, &cmd->index,    3,  0, size, op); | 
|  | } | 
|  |  | 
|  | static void sja1110_cbs_cmd_packing(void *buf, struct sja1105_dyn_cmd *cmd, | 
|  | enum packing_op op) | 
|  | { | 
|  | u8 *p = buf + SJA1105PQRS_SIZE_CBS_ENTRY; | 
|  | const int size = SJA1105_SIZE_DYN_CMD; | 
|  |  | 
|  | sja1105_packing(p, &cmd->valid,   31, 31, size, op); | 
|  | sja1105_packing(p, &cmd->rdwrset, 30, 30, size, op); | 
|  | sja1105_packing(p, &cmd->errors,  29, 29, size, op); | 
|  | sja1105_packing(p, &cmd->index,    7,  0, size, op); | 
|  | } | 
|  |  | 
|  | static size_t sja1105pqrs_cbs_entry_packing(void *buf, void *entry_ptr, | 
|  | enum packing_op op) | 
|  | { | 
|  | const size_t size = SJA1105PQRS_SIZE_CBS_ENTRY; | 
|  | struct sja1105_cbs_entry *entry = entry_ptr; | 
|  |  | 
|  | sja1105_packing(buf, &entry->port,      159, 157, size, op); | 
|  | sja1105_packing(buf, &entry->prio,      156, 154, size, op); | 
|  | sja1105_packing(buf, &entry->credit_lo, 153, 122, size, op); | 
|  | sja1105_packing(buf, &entry->credit_hi, 121,  90, size, op); | 
|  | sja1105_packing(buf, &entry->send_slope, 89,  58, size, op); | 
|  | sja1105_packing(buf, &entry->idle_slope, 57,  26, size, op); | 
|  | return size; | 
|  | } | 
|  |  | 
|  | static size_t sja1110_cbs_entry_packing(void *buf, void *entry_ptr, | 
|  | enum packing_op op) | 
|  | { | 
|  | const size_t size = SJA1105PQRS_SIZE_CBS_ENTRY; | 
|  | struct sja1105_cbs_entry *entry = entry_ptr; | 
|  | u64 entry_type = SJA1110_CBS_SHAPER; | 
|  |  | 
|  | sja1105_packing(buf, &entry_type,       159, 159, size, op); | 
|  | sja1105_packing(buf, &entry->credit_lo, 151, 120, size, op); | 
|  | sja1105_packing(buf, &entry->credit_hi, 119,  88, size, op); | 
|  | sja1105_packing(buf, &entry->send_slope, 87,  56, size, op); | 
|  | sja1105_packing(buf, &entry->idle_slope, 55,  24, size, op); | 
|  | return size; | 
|  | } | 
|  |  | 
|  | static void sja1110_dummy_cmd_packing(void *buf, struct sja1105_dyn_cmd *cmd, | 
|  | enum packing_op op) | 
|  | { | 
|  | } | 
|  |  | 
|  | static void | 
|  | sja1110_l2_policing_cmd_packing(void *buf, struct sja1105_dyn_cmd *cmd, | 
|  | enum packing_op op) | 
|  | { | 
|  | u8 *p = buf + SJA1105_SIZE_L2_POLICING_ENTRY; | 
|  | const int size = SJA1105_SIZE_DYN_CMD; | 
|  |  | 
|  | sja1105_packing(p, &cmd->valid,   31, 31, size, op); | 
|  | sja1105_packing(p, &cmd->rdwrset, 30, 30, size, op); | 
|  | sja1105_packing(p, &cmd->errors,  29, 29, size, op); | 
|  | sja1105_packing(p, &cmd->index,    6,  0, size, op); | 
|  | } | 
|  |  | 
|  | #define OP_READ		BIT(0) | 
|  | #define OP_WRITE	BIT(1) | 
|  | #define OP_DEL		BIT(2) | 
|  | #define OP_SEARCH	BIT(3) | 
|  | #define OP_VALID_ANYWAY	BIT(4) | 
|  |  | 
|  | /* SJA1105E/T: First generation */ | 
|  | const struct sja1105_dynamic_table_ops sja1105et_dyn_ops[BLK_IDX_MAX_DYN] = { | 
|  | [BLK_IDX_VL_LOOKUP] = { | 
|  | .entry_packing = sja1105et_vl_lookup_entry_packing, | 
|  | .cmd_packing = sja1105et_vl_lookup_cmd_packing, | 
|  | .access = OP_WRITE, | 
|  | .max_entry_count = SJA1105_MAX_VL_LOOKUP_COUNT, | 
|  | .packed_size = SJA1105ET_SIZE_VL_LOOKUP_DYN_CMD, | 
|  | .addr = 0x35, | 
|  | }, | 
|  | [BLK_IDX_L2_LOOKUP] = { | 
|  | .entry_packing = sja1105et_dyn_l2_lookup_entry_packing, | 
|  | .cmd_packing = sja1105et_l2_lookup_cmd_packing, | 
|  | .access = (OP_READ | OP_WRITE | OP_DEL), | 
|  | .max_entry_count = SJA1105_MAX_L2_LOOKUP_COUNT, | 
|  | .packed_size = SJA1105ET_SIZE_L2_LOOKUP_DYN_CMD, | 
|  | .addr = 0x20, | 
|  | }, | 
|  | [BLK_IDX_MGMT_ROUTE] = { | 
|  | .entry_packing = sja1105et_mgmt_route_entry_packing, | 
|  | .cmd_packing = sja1105et_mgmt_route_cmd_packing, | 
|  | .access = (OP_READ | OP_WRITE | OP_VALID_ANYWAY), | 
|  | .max_entry_count = SJA1105_NUM_PORTS, | 
|  | .packed_size = SJA1105ET_SIZE_L2_LOOKUP_DYN_CMD, | 
|  | .addr = 0x20, | 
|  | }, | 
|  | [BLK_IDX_VLAN_LOOKUP] = { | 
|  | .entry_packing = sja1105_vlan_lookup_entry_packing, | 
|  | .cmd_packing = sja1105_vlan_lookup_cmd_packing, | 
|  | .access = (OP_WRITE | OP_DEL), | 
|  | .max_entry_count = SJA1105_MAX_VLAN_LOOKUP_COUNT, | 
|  | .packed_size = SJA1105_SIZE_VLAN_LOOKUP_DYN_CMD, | 
|  | .addr = 0x27, | 
|  | }, | 
|  | [BLK_IDX_L2_FORWARDING] = { | 
|  | .entry_packing = sja1105_l2_forwarding_entry_packing, | 
|  | .cmd_packing = sja1105_l2_forwarding_cmd_packing, | 
|  | .max_entry_count = SJA1105_MAX_L2_FORWARDING_COUNT, | 
|  | .access = OP_WRITE, | 
|  | .packed_size = SJA1105_SIZE_L2_FORWARDING_DYN_CMD, | 
|  | .addr = 0x24, | 
|  | }, | 
|  | [BLK_IDX_MAC_CONFIG] = { | 
|  | .entry_packing = sja1105et_mac_config_entry_packing, | 
|  | .cmd_packing = sja1105et_mac_config_cmd_packing, | 
|  | .max_entry_count = SJA1105_MAX_MAC_CONFIG_COUNT, | 
|  | .access = OP_WRITE, | 
|  | .packed_size = SJA1105ET_SIZE_MAC_CONFIG_DYN_CMD, | 
|  | .addr = 0x36, | 
|  | }, | 
|  | [BLK_IDX_L2_LOOKUP_PARAMS] = { | 
|  | .entry_packing = sja1105et_l2_lookup_params_entry_packing, | 
|  | .cmd_packing = sja1105et_l2_lookup_params_cmd_packing, | 
|  | .max_entry_count = SJA1105_MAX_L2_LOOKUP_PARAMS_COUNT, | 
|  | .access = OP_WRITE, | 
|  | .packed_size = SJA1105ET_SIZE_L2_LOOKUP_PARAMS_DYN_CMD, | 
|  | .addr = 0x38, | 
|  | }, | 
|  | [BLK_IDX_GENERAL_PARAMS] = { | 
|  | .entry_packing = sja1105et_general_params_entry_packing, | 
|  | .cmd_packing = sja1105et_general_params_cmd_packing, | 
|  | .max_entry_count = SJA1105_MAX_GENERAL_PARAMS_COUNT, | 
|  | .access = OP_WRITE, | 
|  | .packed_size = SJA1105ET_SIZE_GENERAL_PARAMS_DYN_CMD, | 
|  | .addr = 0x34, | 
|  | }, | 
|  | [BLK_IDX_RETAGGING] = { | 
|  | .entry_packing = sja1105_retagging_entry_packing, | 
|  | .cmd_packing = sja1105_retagging_cmd_packing, | 
|  | .max_entry_count = SJA1105_MAX_RETAGGING_COUNT, | 
|  | .access = (OP_WRITE | OP_DEL), | 
|  | .packed_size = SJA1105_SIZE_RETAGGING_DYN_CMD, | 
|  | .addr = 0x31, | 
|  | }, | 
|  | [BLK_IDX_CBS] = { | 
|  | .entry_packing = sja1105et_cbs_entry_packing, | 
|  | .cmd_packing = sja1105et_cbs_cmd_packing, | 
|  | .max_entry_count = SJA1105ET_MAX_CBS_COUNT, | 
|  | .access = OP_WRITE, | 
|  | .packed_size = SJA1105ET_SIZE_CBS_DYN_CMD, | 
|  | .addr = 0x2c, | 
|  | }, | 
|  | }; | 
|  |  | 
|  | /* SJA1105P/Q/R/S: Second generation */ | 
|  | const struct sja1105_dynamic_table_ops sja1105pqrs_dyn_ops[BLK_IDX_MAX_DYN] = { | 
|  | [BLK_IDX_VL_LOOKUP] = { | 
|  | .entry_packing = sja1105_vl_lookup_entry_packing, | 
|  | .cmd_packing = sja1105pqrs_vl_lookup_cmd_packing, | 
|  | .access = (OP_READ | OP_WRITE), | 
|  | .max_entry_count = SJA1105_MAX_VL_LOOKUP_COUNT, | 
|  | .packed_size = SJA1105PQRS_SIZE_VL_LOOKUP_DYN_CMD, | 
|  | .addr = 0x47, | 
|  | }, | 
|  | [BLK_IDX_L2_LOOKUP] = { | 
|  | .entry_packing = sja1105pqrs_dyn_l2_lookup_entry_packing, | 
|  | .cmd_packing = sja1105pqrs_l2_lookup_cmd_packing, | 
|  | .access = (OP_READ | OP_WRITE | OP_DEL | OP_SEARCH), | 
|  | .max_entry_count = SJA1105_MAX_L2_LOOKUP_COUNT, | 
|  | .packed_size = SJA1105PQRS_SIZE_L2_LOOKUP_DYN_CMD, | 
|  | .addr = 0x24, | 
|  | }, | 
|  | [BLK_IDX_MGMT_ROUTE] = { | 
|  | .entry_packing = sja1105pqrs_mgmt_route_entry_packing, | 
|  | .cmd_packing = sja1105pqrs_mgmt_route_cmd_packing, | 
|  | .access = (OP_READ | OP_WRITE | OP_DEL | OP_SEARCH | OP_VALID_ANYWAY), | 
|  | .max_entry_count = SJA1105_NUM_PORTS, | 
|  | .packed_size = SJA1105PQRS_SIZE_L2_LOOKUP_DYN_CMD, | 
|  | .addr = 0x24, | 
|  | }, | 
|  | [BLK_IDX_VLAN_LOOKUP] = { | 
|  | .entry_packing = sja1105_vlan_lookup_entry_packing, | 
|  | .cmd_packing = sja1105_vlan_lookup_cmd_packing, | 
|  | .access = (OP_READ | OP_WRITE | OP_DEL), | 
|  | .max_entry_count = SJA1105_MAX_VLAN_LOOKUP_COUNT, | 
|  | .packed_size = SJA1105_SIZE_VLAN_LOOKUP_DYN_CMD, | 
|  | .addr = 0x2D, | 
|  | }, | 
|  | [BLK_IDX_L2_FORWARDING] = { | 
|  | .entry_packing = sja1105_l2_forwarding_entry_packing, | 
|  | .cmd_packing = sja1105_l2_forwarding_cmd_packing, | 
|  | .max_entry_count = SJA1105_MAX_L2_FORWARDING_COUNT, | 
|  | .access = OP_WRITE, | 
|  | .packed_size = SJA1105_SIZE_L2_FORWARDING_DYN_CMD, | 
|  | .addr = 0x2A, | 
|  | }, | 
|  | [BLK_IDX_MAC_CONFIG] = { | 
|  | .entry_packing = sja1105pqrs_mac_config_entry_packing, | 
|  | .cmd_packing = sja1105pqrs_mac_config_cmd_packing, | 
|  | .max_entry_count = SJA1105_MAX_MAC_CONFIG_COUNT, | 
|  | .access = (OP_READ | OP_WRITE), | 
|  | .packed_size = SJA1105PQRS_SIZE_MAC_CONFIG_DYN_CMD, | 
|  | .addr = 0x4B, | 
|  | }, | 
|  | [BLK_IDX_L2_LOOKUP_PARAMS] = { | 
|  | .entry_packing = sja1105pqrs_l2_lookup_params_entry_packing, | 
|  | .cmd_packing = sja1105pqrs_l2_lookup_params_cmd_packing, | 
|  | .max_entry_count = SJA1105_MAX_L2_LOOKUP_PARAMS_COUNT, | 
|  | .access = (OP_READ | OP_WRITE), | 
|  | .packed_size = SJA1105PQRS_SIZE_L2_LOOKUP_PARAMS_DYN_CMD, | 
|  | .addr = 0x54, | 
|  | }, | 
|  | [BLK_IDX_AVB_PARAMS] = { | 
|  | .entry_packing = sja1105pqrs_avb_params_entry_packing, | 
|  | .cmd_packing = sja1105pqrs_avb_params_cmd_packing, | 
|  | .max_entry_count = SJA1105_MAX_AVB_PARAMS_COUNT, | 
|  | .access = (OP_READ | OP_WRITE), | 
|  | .packed_size = SJA1105PQRS_SIZE_AVB_PARAMS_DYN_CMD, | 
|  | .addr = 0x8003, | 
|  | }, | 
|  | [BLK_IDX_GENERAL_PARAMS] = { | 
|  | .entry_packing = sja1105pqrs_general_params_entry_packing, | 
|  | .cmd_packing = sja1105pqrs_general_params_cmd_packing, | 
|  | .max_entry_count = SJA1105_MAX_GENERAL_PARAMS_COUNT, | 
|  | .access = (OP_READ | OP_WRITE), | 
|  | .packed_size = SJA1105PQRS_SIZE_GENERAL_PARAMS_DYN_CMD, | 
|  | .addr = 0x3B, | 
|  | }, | 
|  | [BLK_IDX_RETAGGING] = { | 
|  | .entry_packing = sja1105_retagging_entry_packing, | 
|  | .cmd_packing = sja1105_retagging_cmd_packing, | 
|  | .max_entry_count = SJA1105_MAX_RETAGGING_COUNT, | 
|  | .access = (OP_READ | OP_WRITE | OP_DEL), | 
|  | .packed_size = SJA1105_SIZE_RETAGGING_DYN_CMD, | 
|  | .addr = 0x38, | 
|  | }, | 
|  | [BLK_IDX_CBS] = { | 
|  | .entry_packing = sja1105pqrs_cbs_entry_packing, | 
|  | .cmd_packing = sja1105pqrs_cbs_cmd_packing, | 
|  | .max_entry_count = SJA1105PQRS_MAX_CBS_COUNT, | 
|  | .access = OP_WRITE, | 
|  | .packed_size = SJA1105PQRS_SIZE_CBS_DYN_CMD, | 
|  | .addr = 0x32, | 
|  | }, | 
|  | }; | 
|  |  | 
|  | /* SJA1110: Third generation */ | 
|  | const struct sja1105_dynamic_table_ops sja1110_dyn_ops[BLK_IDX_MAX_DYN] = { | 
|  | [BLK_IDX_VL_LOOKUP] = { | 
|  | .entry_packing = sja1110_vl_lookup_entry_packing, | 
|  | .cmd_packing = sja1110_vl_lookup_cmd_packing, | 
|  | .access = (OP_READ | OP_WRITE | OP_VALID_ANYWAY), | 
|  | .max_entry_count = SJA1110_MAX_VL_LOOKUP_COUNT, | 
|  | .packed_size = SJA1105PQRS_SIZE_VL_LOOKUP_DYN_CMD, | 
|  | .addr = SJA1110_SPI_ADDR(0x124), | 
|  | }, | 
|  | [BLK_IDX_VL_POLICING] = { | 
|  | .entry_packing = sja1110_vl_policing_entry_packing, | 
|  | .cmd_packing = sja1110_vl_policing_cmd_packing, | 
|  | .access = (OP_READ | OP_WRITE | OP_VALID_ANYWAY), | 
|  | .max_entry_count = SJA1110_MAX_VL_POLICING_COUNT, | 
|  | .packed_size = SJA1110_SIZE_VL_POLICING_DYN_CMD, | 
|  | .addr = SJA1110_SPI_ADDR(0x310), | 
|  | }, | 
|  | [BLK_IDX_L2_LOOKUP] = { | 
|  | .entry_packing = sja1110_dyn_l2_lookup_entry_packing, | 
|  | .cmd_packing = sja1110_l2_lookup_cmd_packing, | 
|  | .access = (OP_READ | OP_WRITE | OP_DEL | OP_SEARCH), | 
|  | .max_entry_count = SJA1105_MAX_L2_LOOKUP_COUNT, | 
|  | .packed_size = SJA1110_SIZE_L2_LOOKUP_DYN_CMD, | 
|  | .addr = SJA1110_SPI_ADDR(0x8c), | 
|  | }, | 
|  | [BLK_IDX_VLAN_LOOKUP] = { | 
|  | .entry_packing = sja1110_vlan_lookup_entry_packing, | 
|  | .cmd_packing = sja1110_vlan_lookup_cmd_packing, | 
|  | .access = (OP_READ | OP_WRITE | OP_DEL), | 
|  | .max_entry_count = SJA1105_MAX_VLAN_LOOKUP_COUNT, | 
|  | .packed_size = SJA1110_SIZE_VLAN_LOOKUP_DYN_CMD, | 
|  | .addr = SJA1110_SPI_ADDR(0xb4), | 
|  | }, | 
|  | [BLK_IDX_L2_FORWARDING] = { | 
|  | .entry_packing = sja1110_l2_forwarding_entry_packing, | 
|  | .cmd_packing = sja1110_l2_forwarding_cmd_packing, | 
|  | .max_entry_count = SJA1110_MAX_L2_FORWARDING_COUNT, | 
|  | .access = (OP_READ | OP_WRITE | OP_VALID_ANYWAY), | 
|  | .packed_size = SJA1105_SIZE_L2_FORWARDING_DYN_CMD, | 
|  | .addr = SJA1110_SPI_ADDR(0xa8), | 
|  | }, | 
|  | [BLK_IDX_MAC_CONFIG] = { | 
|  | .entry_packing = sja1110_mac_config_entry_packing, | 
|  | .cmd_packing = sja1110_mac_config_cmd_packing, | 
|  | .max_entry_count = SJA1110_MAX_MAC_CONFIG_COUNT, | 
|  | .access = (OP_READ | OP_WRITE | OP_VALID_ANYWAY), | 
|  | .packed_size = SJA1105PQRS_SIZE_MAC_CONFIG_DYN_CMD, | 
|  | .addr = SJA1110_SPI_ADDR(0x134), | 
|  | }, | 
|  | [BLK_IDX_L2_LOOKUP_PARAMS] = { | 
|  | .entry_packing = sja1110_l2_lookup_params_entry_packing, | 
|  | .cmd_packing = sja1110_l2_lookup_params_cmd_packing, | 
|  | .max_entry_count = SJA1105_MAX_L2_LOOKUP_PARAMS_COUNT, | 
|  | .access = (OP_READ | OP_WRITE | OP_VALID_ANYWAY), | 
|  | .packed_size = SJA1110_SIZE_L2_LOOKUP_PARAMS_DYN_CMD, | 
|  | .addr = SJA1110_SPI_ADDR(0x158), | 
|  | }, | 
|  | [BLK_IDX_AVB_PARAMS] = { | 
|  | .entry_packing = sja1105pqrs_avb_params_entry_packing, | 
|  | .cmd_packing = sja1105pqrs_avb_params_cmd_packing, | 
|  | .max_entry_count = SJA1105_MAX_AVB_PARAMS_COUNT, | 
|  | .access = (OP_READ | OP_WRITE | OP_VALID_ANYWAY), | 
|  | .packed_size = SJA1105PQRS_SIZE_AVB_PARAMS_DYN_CMD, | 
|  | .addr = SJA1110_SPI_ADDR(0x2000C), | 
|  | }, | 
|  | [BLK_IDX_GENERAL_PARAMS] = { | 
|  | .entry_packing = sja1110_general_params_entry_packing, | 
|  | .cmd_packing = sja1110_general_params_cmd_packing, | 
|  | .max_entry_count = SJA1105_MAX_GENERAL_PARAMS_COUNT, | 
|  | .access = (OP_READ | OP_WRITE | OP_VALID_ANYWAY), | 
|  | .packed_size = SJA1110_SIZE_GENERAL_PARAMS_DYN_CMD, | 
|  | .addr = SJA1110_SPI_ADDR(0xe8), | 
|  | }, | 
|  | [BLK_IDX_RETAGGING] = { | 
|  | .entry_packing = sja1110_retagging_entry_packing, | 
|  | .cmd_packing = sja1110_retagging_cmd_packing, | 
|  | .max_entry_count = SJA1105_MAX_RETAGGING_COUNT, | 
|  | .access = (OP_READ | OP_WRITE | OP_DEL), | 
|  | .packed_size = SJA1105_SIZE_RETAGGING_DYN_CMD, | 
|  | .addr = SJA1110_SPI_ADDR(0xdc), | 
|  | }, | 
|  | [BLK_IDX_CBS] = { | 
|  | .entry_packing = sja1110_cbs_entry_packing, | 
|  | .cmd_packing = sja1110_cbs_cmd_packing, | 
|  | .max_entry_count = SJA1110_MAX_CBS_COUNT, | 
|  | .access = (OP_READ | OP_WRITE | OP_VALID_ANYWAY), | 
|  | .packed_size = SJA1105PQRS_SIZE_CBS_DYN_CMD, | 
|  | .addr = SJA1110_SPI_ADDR(0xc4), | 
|  | }, | 
|  | [BLK_IDX_XMII_PARAMS] = { | 
|  | .entry_packing = sja1110_xmii_params_entry_packing, | 
|  | .cmd_packing = sja1110_dummy_cmd_packing, | 
|  | .max_entry_count = SJA1105_MAX_XMII_PARAMS_COUNT, | 
|  | .access = (OP_READ | OP_VALID_ANYWAY), | 
|  | .packed_size = SJA1110_SIZE_XMII_PARAMS_DYN_CMD, | 
|  | .addr = SJA1110_SPI_ADDR(0x3c), | 
|  | }, | 
|  | [BLK_IDX_L2_POLICING] = { | 
|  | .entry_packing = sja1110_l2_policing_entry_packing, | 
|  | .cmd_packing = sja1110_l2_policing_cmd_packing, | 
|  | .max_entry_count = SJA1110_MAX_L2_POLICING_COUNT, | 
|  | .access = (OP_READ | OP_WRITE | OP_VALID_ANYWAY), | 
|  | .packed_size = SJA1110_SIZE_L2_POLICING_DYN_CMD, | 
|  | .addr = SJA1110_SPI_ADDR(0x2fc), | 
|  | }, | 
|  | [BLK_IDX_L2_FORWARDING_PARAMS] = { | 
|  | .entry_packing = sja1110_l2_forwarding_params_entry_packing, | 
|  | .cmd_packing = sja1110_dummy_cmd_packing, | 
|  | .max_entry_count = SJA1105_MAX_L2_FORWARDING_PARAMS_COUNT, | 
|  | .access = (OP_READ | OP_VALID_ANYWAY), | 
|  | .packed_size = SJA1110_SIZE_L2_FORWARDING_PARAMS_DYN_CMD, | 
|  | .addr = SJA1110_SPI_ADDR(0x20000), | 
|  | }, | 
|  | }; | 
|  |  | 
|  | #define SJA1105_DYNAMIC_CONFIG_SLEEP_US		10 | 
|  | #define SJA1105_DYNAMIC_CONFIG_TIMEOUT_US	100000 | 
|  |  | 
|  | static int | 
|  | sja1105_dynamic_config_poll_valid(struct sja1105_private *priv, | 
|  | const struct sja1105_dynamic_table_ops *ops, | 
|  | void *entry, bool check_valident, | 
|  | bool check_errors) | 
|  | { | 
|  | u8 packed_buf[SJA1105_MAX_DYN_CMD_SIZE] = {}; | 
|  | struct sja1105_dyn_cmd cmd = {}; | 
|  | int rc; | 
|  |  | 
|  | /* Read back the whole entry + command structure. */ | 
|  | rc = sja1105_xfer_buf(priv, SPI_READ, ops->addr, packed_buf, | 
|  | ops->packed_size); | 
|  | if (rc) | 
|  | return rc; | 
|  |  | 
|  | /* Unpack the command structure, and return it to the caller in case it | 
|  | * needs to perform further checks on it (VALIDENT). | 
|  | */ | 
|  | ops->cmd_packing(packed_buf, &cmd, UNPACK); | 
|  |  | 
|  | /* Hardware hasn't cleared VALID => still working on it */ | 
|  | if (cmd.valid) | 
|  | return -EAGAIN; | 
|  |  | 
|  | if (check_valident && !cmd.valident && !(ops->access & OP_VALID_ANYWAY)) | 
|  | return -ENOENT; | 
|  |  | 
|  | if (check_errors && cmd.errors) | 
|  | return -EINVAL; | 
|  |  | 
|  | /* Don't dereference possibly NULL pointer - maybe caller | 
|  | * only wanted to see whether the entry existed or not. | 
|  | */ | 
|  | if (entry) | 
|  | ops->entry_packing(packed_buf, entry, UNPACK); | 
|  |  | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | /* Poll the dynamic config entry's control area until the hardware has | 
|  | * cleared the VALID bit, which means we have confirmation that it has | 
|  | * finished processing the command. | 
|  | */ | 
|  | static int | 
|  | sja1105_dynamic_config_wait_complete(struct sja1105_private *priv, | 
|  | const struct sja1105_dynamic_table_ops *ops, | 
|  | void *entry, bool check_valident, | 
|  | bool check_errors) | 
|  | { | 
|  | int err, rc; | 
|  |  | 
|  | err = read_poll_timeout(sja1105_dynamic_config_poll_valid, | 
|  | rc, rc != -EAGAIN, | 
|  | SJA1105_DYNAMIC_CONFIG_SLEEP_US, | 
|  | SJA1105_DYNAMIC_CONFIG_TIMEOUT_US, | 
|  | false, priv, ops, entry, check_valident, | 
|  | check_errors); | 
|  | return err < 0 ? err : rc; | 
|  | } | 
|  |  | 
|  | /* Provides read access to the settings through the dynamic interface | 
|  | * of the switch. | 
|  | * @blk_idx	is used as key to select from the sja1105_dynamic_table_ops. | 
|  | *		The selection is limited by the hardware in respect to which | 
|  | *		configuration blocks can be read through the dynamic interface. | 
|  | * @index	is used to retrieve a particular table entry. If negative, | 
|  | *		(and if the @blk_idx supports the searching operation) a search | 
|  | *		is performed by the @entry parameter. | 
|  | * @entry	Type-casted to an unpacked structure that holds a table entry | 
|  | *		of the type specified in @blk_idx. | 
|  | *		Usually an output argument. If @index is negative, then this | 
|  | *		argument is used as input/output: it should be pre-populated | 
|  | *		with the element to search for. Entries which support the | 
|  | *		search operation will have an "index" field (not the @index | 
|  | *		argument to this function) and that is where the found index | 
|  | *		will be returned (or left unmodified - thus negative - if not | 
|  | *		found). | 
|  | */ | 
|  | int sja1105_dynamic_config_read(struct sja1105_private *priv, | 
|  | enum sja1105_blk_idx blk_idx, | 
|  | int index, void *entry) | 
|  | { | 
|  | const struct sja1105_dynamic_table_ops *ops; | 
|  | struct sja1105_dyn_cmd cmd = {0}; | 
|  | /* SPI payload buffer */ | 
|  | u8 packed_buf[SJA1105_MAX_DYN_CMD_SIZE] = {0}; | 
|  | int rc; | 
|  |  | 
|  | if (blk_idx >= BLK_IDX_MAX_DYN) | 
|  | return -ERANGE; | 
|  |  | 
|  | ops = &priv->info->dyn_ops[blk_idx]; | 
|  |  | 
|  | if (index >= 0 && index >= ops->max_entry_count) | 
|  | return -ERANGE; | 
|  | if (index < 0 && !(ops->access & OP_SEARCH)) | 
|  | return -EOPNOTSUPP; | 
|  | if (!(ops->access & OP_READ)) | 
|  | return -EOPNOTSUPP; | 
|  | if (ops->packed_size > SJA1105_MAX_DYN_CMD_SIZE) | 
|  | return -ERANGE; | 
|  | if (!ops->cmd_packing) | 
|  | return -EOPNOTSUPP; | 
|  | if (!ops->entry_packing) | 
|  | return -EOPNOTSUPP; | 
|  |  | 
|  | cmd.valid = true; /* Trigger action on table entry */ | 
|  | cmd.rdwrset = SPI_READ; /* Action is read */ | 
|  | if (index < 0) { | 
|  | /* Avoid copying a signed negative number to an u64 */ | 
|  | cmd.index = 0; | 
|  | cmd.search = true; | 
|  | } else { | 
|  | cmd.index = index; | 
|  | cmd.search = false; | 
|  | } | 
|  | cmd.valident = true; | 
|  | ops->cmd_packing(packed_buf, &cmd, PACK); | 
|  |  | 
|  | if (cmd.search) | 
|  | ops->entry_packing(packed_buf, entry, PACK); | 
|  |  | 
|  | /* Send SPI write operation: read config table entry */ | 
|  | mutex_lock(&priv->dynamic_config_lock); | 
|  | rc = sja1105_xfer_buf(priv, SPI_WRITE, ops->addr, packed_buf, | 
|  | ops->packed_size); | 
|  | if (rc < 0) | 
|  | goto out; | 
|  |  | 
|  | rc = sja1105_dynamic_config_wait_complete(priv, ops, entry, true, false); | 
|  | out: | 
|  | mutex_unlock(&priv->dynamic_config_lock); | 
|  |  | 
|  | return rc; | 
|  | } | 
|  |  | 
|  | int sja1105_dynamic_config_write(struct sja1105_private *priv, | 
|  | enum sja1105_blk_idx blk_idx, | 
|  | int index, void *entry, bool keep) | 
|  | { | 
|  | const struct sja1105_dynamic_table_ops *ops; | 
|  | struct sja1105_dyn_cmd cmd = {0}; | 
|  | /* SPI payload buffer */ | 
|  | u8 packed_buf[SJA1105_MAX_DYN_CMD_SIZE] = {0}; | 
|  | int rc; | 
|  |  | 
|  | if (blk_idx >= BLK_IDX_MAX_DYN) | 
|  | return -ERANGE; | 
|  |  | 
|  | ops = &priv->info->dyn_ops[blk_idx]; | 
|  |  | 
|  | if (index >= ops->max_entry_count) | 
|  | return -ERANGE; | 
|  | if (index < 0) | 
|  | return -ERANGE; | 
|  | if (!(ops->access & OP_WRITE)) | 
|  | return -EOPNOTSUPP; | 
|  | if (!keep && !(ops->access & OP_DEL)) | 
|  | return -EOPNOTSUPP; | 
|  | if (ops->packed_size > SJA1105_MAX_DYN_CMD_SIZE) | 
|  | return -ERANGE; | 
|  |  | 
|  | cmd.valident = keep; /* If false, deletes entry */ | 
|  | cmd.valid = true; /* Trigger action on table entry */ | 
|  | cmd.rdwrset = SPI_WRITE; /* Action is write */ | 
|  | cmd.index = index; | 
|  |  | 
|  | if (!ops->cmd_packing) | 
|  | return -EOPNOTSUPP; | 
|  | ops->cmd_packing(packed_buf, &cmd, PACK); | 
|  |  | 
|  | if (!ops->entry_packing) | 
|  | return -EOPNOTSUPP; | 
|  | /* Don't dereference potentially NULL pointer if just | 
|  | * deleting a table entry is what was requested. For cases | 
|  | * where 'index' field is physically part of entry structure, | 
|  | * and needed here, we deal with that in the cmd_packing callback. | 
|  | */ | 
|  | if (keep) | 
|  | ops->entry_packing(packed_buf, entry, PACK); | 
|  |  | 
|  | /* Send SPI write operation: read config table entry */ | 
|  | mutex_lock(&priv->dynamic_config_lock); | 
|  | rc = sja1105_xfer_buf(priv, SPI_WRITE, ops->addr, packed_buf, | 
|  | ops->packed_size); | 
|  | if (rc < 0) | 
|  | goto out; | 
|  |  | 
|  | rc = sja1105_dynamic_config_wait_complete(priv, ops, NULL, false, true); | 
|  | out: | 
|  | mutex_unlock(&priv->dynamic_config_lock); | 
|  |  | 
|  | return rc; | 
|  | } | 
|  |  | 
|  | static u8 sja1105_crc8_add(u8 crc, u8 byte, u8 poly) | 
|  | { | 
|  | int i; | 
|  |  | 
|  | for (i = 0; i < 8; i++) { | 
|  | if ((crc ^ byte) & (1 << 7)) { | 
|  | crc <<= 1; | 
|  | crc ^= poly; | 
|  | } else { | 
|  | crc <<= 1; | 
|  | } | 
|  | byte <<= 1; | 
|  | } | 
|  | return crc; | 
|  | } | 
|  |  | 
|  | /* CRC8 algorithm with non-reversed input, non-reversed output, | 
|  | * no input xor and no output xor. Code customized for receiving | 
|  | * the SJA1105 E/T FDB keys (vlanid, macaddr) as input. CRC polynomial | 
|  | * is also received as argument in the Koopman notation that the switch | 
|  | * hardware stores it in. | 
|  | */ | 
|  | u8 sja1105et_fdb_hash(struct sja1105_private *priv, const u8 *addr, u16 vid) | 
|  | { | 
|  | struct sja1105_l2_lookup_params_entry *l2_lookup_params = | 
|  | priv->static_config.tables[BLK_IDX_L2_LOOKUP_PARAMS].entries; | 
|  | u64 input, poly_koopman = l2_lookup_params->poly; | 
|  | /* Convert polynomial from Koopman to 'normal' notation */ | 
|  | u8 poly = (u8)(1 + (poly_koopman << 1)); | 
|  | u8 crc = 0; /* seed */ | 
|  | int i; | 
|  |  | 
|  | input = ((u64)vid << 48) | ether_addr_to_u64(addr); | 
|  |  | 
|  | /* Mask the eight bytes starting from MSB one at a time */ | 
|  | for (i = 56; i >= 0; i -= 8) { | 
|  | u8 byte = (input & (0xffull << i)) >> i; | 
|  |  | 
|  | crc = sja1105_crc8_add(crc, byte, poly); | 
|  | } | 
|  | return crc; | 
|  | } |