|  | /* SPDX-License-Identifier: GPL-2.0-only */ | 
|  | /**************************************************************************** | 
|  | * Driver for Solarflare network controllers and boards | 
|  | * Copyright 2010-2012 Solarflare Communications Inc. | 
|  | */ | 
|  | #ifndef _VFDI_H | 
|  | #define _VFDI_H | 
|  |  | 
|  | /** | 
|  | * DOC: Virtual Function Driver Interface | 
|  | * | 
|  | * This file contains software structures used to form a two way | 
|  | * communication channel between the VF driver and the PF driver, | 
|  | * named Virtual Function Driver Interface (VFDI). | 
|  | * | 
|  | * For the purposes of VFDI, a page is a memory region with size and | 
|  | * alignment of 4K.  All addresses are DMA addresses to be used within | 
|  | * the domain of the relevant VF. | 
|  | * | 
|  | * The only hardware-defined channels for a VF driver to communicate | 
|  | * with the PF driver are the event mailboxes (%FR_CZ_USR_EV | 
|  | * registers).  Writing to these registers generates an event with | 
|  | * EV_CODE = EV_CODE_USR_EV, USER_QID set to the index of the mailbox | 
|  | * and USER_EV_REG_VALUE set to the value written.  The PF driver may | 
|  | * direct or disable delivery of these events by setting | 
|  | * %FR_CZ_USR_EV_CFG. | 
|  | * | 
|  | * The PF driver can send arbitrary events to arbitrary event queues. | 
|  | * However, for consistency, VFDI events from the PF are defined to | 
|  | * follow the same form and be sent to the first event queue assigned | 
|  | * to the VF while that queue is enabled by the VF driver. | 
|  | * | 
|  | * The general form of the variable bits of VFDI events is: | 
|  | * | 
|  | *       0             16                       24   31 | 
|  | *      | DATA        | TYPE                   | SEQ   | | 
|  | * | 
|  | * SEQ is a sequence number which should be incremented by 1 (modulo | 
|  | * 256) for each event.  The sequence numbers used in each direction | 
|  | * are independent. | 
|  | * | 
|  | * The VF submits requests of type &struct vfdi_req by sending the | 
|  | * address of the request (ADDR) in a series of 4 events: | 
|  | * | 
|  | *       0             16                       24   31 | 
|  | *      | ADDR[0:15]  | VFDI_EV_TYPE_REQ_WORD0 | SEQ   | | 
|  | *      | ADDR[16:31] | VFDI_EV_TYPE_REQ_WORD1 | SEQ+1 | | 
|  | *      | ADDR[32:47] | VFDI_EV_TYPE_REQ_WORD2 | SEQ+2 | | 
|  | *      | ADDR[48:63] | VFDI_EV_TYPE_REQ_WORD3 | SEQ+3 | | 
|  | * | 
|  | * The address must be page-aligned.  After receiving such a valid | 
|  | * series of events, the PF driver will attempt to read the request | 
|  | * and write a response to the same address.  In case of an invalid | 
|  | * sequence of events or a DMA error, there will be no response. | 
|  | * | 
|  | * The VF driver may request that the PF driver writes status | 
|  | * information into its domain asynchronously.  After writing the | 
|  | * status, the PF driver will send an event of the form: | 
|  | * | 
|  | *       0             16                       24   31 | 
|  | *      | reserved    | VFDI_EV_TYPE_STATUS    | SEQ   | | 
|  | * | 
|  | * In case the VF must be reset for any reason, the PF driver will | 
|  | * send an event of the form: | 
|  | * | 
|  | *       0             16                       24   31 | 
|  | *      | reserved    | VFDI_EV_TYPE_RESET     | SEQ   | | 
|  | * | 
|  | * It is then the responsibility of the VF driver to request | 
|  | * reinitialisation of its queues. | 
|  | */ | 
|  | #define VFDI_EV_SEQ_LBN 24 | 
|  | #define VFDI_EV_SEQ_WIDTH 8 | 
|  | #define VFDI_EV_TYPE_LBN 16 | 
|  | #define VFDI_EV_TYPE_WIDTH 8 | 
|  | #define VFDI_EV_TYPE_REQ_WORD0 0 | 
|  | #define VFDI_EV_TYPE_REQ_WORD1 1 | 
|  | #define VFDI_EV_TYPE_REQ_WORD2 2 | 
|  | #define VFDI_EV_TYPE_REQ_WORD3 3 | 
|  | #define VFDI_EV_TYPE_STATUS 4 | 
|  | #define VFDI_EV_TYPE_RESET 5 | 
|  | #define VFDI_EV_DATA_LBN 0 | 
|  | #define VFDI_EV_DATA_WIDTH 16 | 
|  |  | 
|  | struct vfdi_endpoint { | 
|  | u8 mac_addr[ETH_ALEN]; | 
|  | __be16 tci; | 
|  | }; | 
|  |  | 
|  | /** | 
|  | * enum vfdi_op - VFDI operation enumeration | 
|  | * @VFDI_OP_RESPONSE: Indicates a response to the request. | 
|  | * @VFDI_OP_INIT_EVQ: Initialize SRAM entries and initialize an EVQ. | 
|  | * @VFDI_OP_INIT_RXQ: Initialize SRAM entries and initialize an RXQ. | 
|  | * @VFDI_OP_INIT_TXQ: Initialize SRAM entries and initialize a TXQ. | 
|  | * @VFDI_OP_FINI_ALL_QUEUES: Flush all queues, finalize all queues, then | 
|  | *	finalize the SRAM entries. | 
|  | * @VFDI_OP_INSERT_FILTER: Insert a MAC filter targeting the given RXQ. | 
|  | * @VFDI_OP_REMOVE_ALL_FILTERS: Remove all filters. | 
|  | * @VFDI_OP_SET_STATUS_PAGE: Set the DMA page(s) used for status updates | 
|  | *	from PF and write the initial status. | 
|  | * @VFDI_OP_CLEAR_STATUS_PAGE: Clear the DMA page(s) used for status | 
|  | *	updates from PF. | 
|  | */ | 
|  | enum vfdi_op { | 
|  | VFDI_OP_RESPONSE = 0, | 
|  | VFDI_OP_INIT_EVQ = 1, | 
|  | VFDI_OP_INIT_RXQ = 2, | 
|  | VFDI_OP_INIT_TXQ = 3, | 
|  | VFDI_OP_FINI_ALL_QUEUES = 4, | 
|  | VFDI_OP_INSERT_FILTER = 5, | 
|  | VFDI_OP_REMOVE_ALL_FILTERS = 6, | 
|  | VFDI_OP_SET_STATUS_PAGE = 7, | 
|  | VFDI_OP_CLEAR_STATUS_PAGE = 8, | 
|  | VFDI_OP_LIMIT, | 
|  | }; | 
|  |  | 
|  | /* Response codes for VFDI operations. Other values may be used in future. */ | 
|  | #define VFDI_RC_SUCCESS		0 | 
|  | #define VFDI_RC_ENOMEM		(-12) | 
|  | #define VFDI_RC_EINVAL		(-22) | 
|  | #define VFDI_RC_EOPNOTSUPP	(-95) | 
|  | #define VFDI_RC_ETIMEDOUT	(-110) | 
|  |  | 
|  | /** | 
|  | * struct vfdi_req - Request from VF driver to PF driver | 
|  | * @op: Operation code or response indicator, taken from &enum vfdi_op. | 
|  | * @rc: Response code.  Set to 0 on success or a negative error code on failure. | 
|  | * @u.init_evq.index: Index of event queue to create. | 
|  | * @u.init_evq.buf_count: Number of 4k buffers backing event queue. | 
|  | * @u.init_evq.addr: Array of length %u.init_evq.buf_count containing DMA | 
|  | *	address of each page backing the event queue. | 
|  | * @u.init_rxq.index: Index of receive queue to create. | 
|  | * @u.init_rxq.buf_count: Number of 4k buffers backing receive queue. | 
|  | * @u.init_rxq.evq: Instance of event queue to target receive events at. | 
|  | * @u.init_rxq.label: Label used in receive events. | 
|  | * @u.init_rxq.flags: Unused. | 
|  | * @u.init_rxq.addr: Array of length %u.init_rxq.buf_count containing DMA | 
|  | *	address of each page backing the receive queue. | 
|  | * @u.init_txq.index: Index of transmit queue to create. | 
|  | * @u.init_txq.buf_count: Number of 4k buffers backing transmit queue. | 
|  | * @u.init_txq.evq: Instance of event queue to target transmit completion | 
|  | *	events at. | 
|  | * @u.init_txq.label: Label used in transmit completion events. | 
|  | * @u.init_txq.flags: Checksum offload flags. | 
|  | * @u.init_txq.addr: Array of length %u.init_txq.buf_count containing DMA | 
|  | *	address of each page backing the transmit queue. | 
|  | * @u.mac_filter.rxq: Insert MAC filter at VF local address/VLAN targeting | 
|  | *	all traffic at this receive queue. | 
|  | * @u.mac_filter.flags: MAC filter flags. | 
|  | * @u.set_status_page.dma_addr: Base address for the &struct vfdi_status. | 
|  | *	This address must be page-aligned and the PF may write up to a | 
|  | *	whole page (allowing for extension of the structure). | 
|  | * @u.set_status_page.peer_page_count: Number of additional pages the VF | 
|  | *	has provided into which peer addresses may be DMAd. | 
|  | * @u.set_status_page.peer_page_addr: Array of DMA addresses of pages. | 
|  | *	If the number of peers exceeds 256, then the VF must provide | 
|  | *	additional pages in this array. The PF will then DMA up to | 
|  | *	512 vfdi_endpoint structures into each page.  These addresses | 
|  | *	must be page-aligned. | 
|  | */ | 
|  | struct vfdi_req { | 
|  | u32 op; | 
|  | u32 reserved1; | 
|  | s32 rc; | 
|  | u32 reserved2; | 
|  | union { | 
|  | struct { | 
|  | u32 index; | 
|  | u32 buf_count; | 
|  | u64 addr[]; | 
|  | } init_evq; | 
|  | struct { | 
|  | u32 index; | 
|  | u32 buf_count; | 
|  | u32 evq; | 
|  | u32 label; | 
|  | u32 flags; | 
|  | #define VFDI_RXQ_FLAG_SCATTER_EN 1 | 
|  | u32 reserved; | 
|  | u64 addr[]; | 
|  | } init_rxq; | 
|  | struct { | 
|  | u32 index; | 
|  | u32 buf_count; | 
|  | u32 evq; | 
|  | u32 label; | 
|  | u32 flags; | 
|  | #define VFDI_TXQ_FLAG_IP_CSUM_DIS 1 | 
|  | #define VFDI_TXQ_FLAG_TCPUDP_CSUM_DIS 2 | 
|  | u32 reserved; | 
|  | u64 addr[]; | 
|  | } init_txq; | 
|  | struct { | 
|  | u32 rxq; | 
|  | u32 flags; | 
|  | #define VFDI_MAC_FILTER_FLAG_RSS 1 | 
|  | #define VFDI_MAC_FILTER_FLAG_SCATTER 2 | 
|  | } mac_filter; | 
|  | struct { | 
|  | u64 dma_addr; | 
|  | u64 peer_page_count; | 
|  | u64 peer_page_addr[]; | 
|  | } set_status_page; | 
|  | } u; | 
|  | }; | 
|  |  | 
|  | /** | 
|  | * struct vfdi_status - Status provided by PF driver to VF driver | 
|  | * @generation_start: A generation count DMA'd to VF *before* the | 
|  | *	rest of the structure. | 
|  | * @generation_end: A generation count DMA'd to VF *after* the | 
|  | *	rest of the structure. | 
|  | * @version: Version of this structure; currently set to 1.  Later | 
|  | *	versions must either be layout-compatible or only be sent to VFs | 
|  | *	that specifically request them. | 
|  | * @length: Total length of this structure including embedded tables | 
|  | * @vi_scale: log2 the number of VIs available on this VF. This quantity | 
|  | *	is used by the hardware for register decoding. | 
|  | * @max_tx_channels: The maximum number of transmit queues the VF can use. | 
|  | * @rss_rxq_count: The number of receive queues present in the shared RSS | 
|  | *	indirection table. | 
|  | * @peer_count: Total number of peers in the complete peer list. If larger | 
|  | *	than ARRAY_SIZE(%peers), then the VF must provide sufficient | 
|  | *	additional pages each of which is filled with vfdi_endpoint structures. | 
|  | * @local: The MAC address and outer VLAN tag of *this* VF | 
|  | * @peers: Table of peer addresses.  The @tci fields in these structures | 
|  | *	are currently unused and must be ignored.  Additional peers are | 
|  | *	written into any additional pages provided by the VF. | 
|  | * @timer_quantum_ns: Timer quantum (nominal period between timer ticks) | 
|  | *	for interrupt moderation timers, in nanoseconds. This member is only | 
|  | *	present if @length is sufficiently large. | 
|  | */ | 
|  | struct vfdi_status { | 
|  | u32 generation_start; | 
|  | u32 generation_end; | 
|  | u32 version; | 
|  | u32 length; | 
|  | u8 vi_scale; | 
|  | u8 max_tx_channels; | 
|  | u8 rss_rxq_count; | 
|  | u8 reserved1; | 
|  | u16 peer_count; | 
|  | u16 reserved2; | 
|  | struct vfdi_endpoint local; | 
|  | struct vfdi_endpoint peers[256]; | 
|  |  | 
|  | /* Members below here extend version 1 of this structure */ | 
|  | u32 timer_quantum_ns; | 
|  | }; | 
|  |  | 
|  | #endif |