1*ef270ab1SKenneth D. Merry /*- 2*ef270ab1SKenneth D. Merry * Copyright (c) 2017 Broadcom. All rights reserved. 3*ef270ab1SKenneth D. Merry * The term "Broadcom" refers to Broadcom Limited and/or its subsidiaries. 4*ef270ab1SKenneth D. Merry * 5*ef270ab1SKenneth D. Merry * Redistribution and use in source and binary forms, with or without 6*ef270ab1SKenneth D. Merry * modification, are permitted provided that the following conditions are met: 7*ef270ab1SKenneth D. Merry * 8*ef270ab1SKenneth D. Merry * 1. Redistributions of source code must retain the above copyright notice, 9*ef270ab1SKenneth D. Merry * this list of conditions and the following disclaimer. 10*ef270ab1SKenneth D. Merry * 11*ef270ab1SKenneth D. Merry * 2. Redistributions in binary form must reproduce the above copyright notice, 12*ef270ab1SKenneth D. Merry * this list of conditions and the following disclaimer in the documentation 13*ef270ab1SKenneth D. Merry * and/or other materials provided with the distribution. 14*ef270ab1SKenneth D. Merry * 15*ef270ab1SKenneth D. Merry * 3. Neither the name of the copyright holder nor the names of its contributors 16*ef270ab1SKenneth D. Merry * may be used to endorse or promote products derived from this software 17*ef270ab1SKenneth D. Merry * without specific prior written permission. 18*ef270ab1SKenneth D. Merry * 19*ef270ab1SKenneth D. Merry * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" 20*ef270ab1SKenneth D. Merry * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE 21*ef270ab1SKenneth D. Merry * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE 22*ef270ab1SKenneth D. Merry * ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE 23*ef270ab1SKenneth D. Merry * LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR 24*ef270ab1SKenneth D. Merry * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF 25*ef270ab1SKenneth D. Merry * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS 26*ef270ab1SKenneth D. Merry * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN 27*ef270ab1SKenneth D. Merry * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) 28*ef270ab1SKenneth D. Merry * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE 29*ef270ab1SKenneth D. Merry * POSSIBILITY OF SUCH DAMAGE. 30*ef270ab1SKenneth D. Merry * 31*ef270ab1SKenneth D. Merry * $FreeBSD$ 32*ef270ab1SKenneth D. Merry */ 33*ef270ab1SKenneth D. Merry 34*ef270ab1SKenneth D. Merry /** 35*ef270ab1SKenneth D. Merry * @file 36*ef270ab1SKenneth D. Merry * OCS linux driver IO declarations 37*ef270ab1SKenneth D. Merry */ 38*ef270ab1SKenneth D. Merry 39*ef270ab1SKenneth D. Merry #if !defined(__OCS_IO_H__) 40*ef270ab1SKenneth D. Merry #define __OCS_IO_H__ 41*ef270ab1SKenneth D. Merry 42*ef270ab1SKenneth D. Merry #define io_error_log(io, fmt, ...) \ 43*ef270ab1SKenneth D. Merry do { \ 44*ef270ab1SKenneth D. Merry if (OCS_LOG_ENABLE_IO_ERRORS(io->ocs)) \ 45*ef270ab1SKenneth D. Merry ocs_log_warn(io->ocs, fmt, ##__VA_ARGS__); \ 46*ef270ab1SKenneth D. Merry } while (0) 47*ef270ab1SKenneth D. Merry 48*ef270ab1SKenneth D. Merry /** 49*ef270ab1SKenneth D. Merry * @brief FCP IO context 50*ef270ab1SKenneth D. Merry * 51*ef270ab1SKenneth D. Merry * This structure is used for transport and backend IO requests and responses. 52*ef270ab1SKenneth D. Merry */ 53*ef270ab1SKenneth D. Merry 54*ef270ab1SKenneth D. Merry #define SCSI_CMD_BUF_LENGTH 48 55*ef270ab1SKenneth D. Merry #define SCSI_RSP_BUF_LENGTH sizeof(fcp_rsp_iu_t) 56*ef270ab1SKenneth D. Merry 57*ef270ab1SKenneth D. Merry /** 58*ef270ab1SKenneth D. Merry * @brief OCS IO types 59*ef270ab1SKenneth D. Merry */ 60*ef270ab1SKenneth D. Merry typedef enum { 61*ef270ab1SKenneth D. Merry OCS_IO_TYPE_IO = 0, 62*ef270ab1SKenneth D. Merry OCS_IO_TYPE_ELS, 63*ef270ab1SKenneth D. Merry OCS_IO_TYPE_CT, 64*ef270ab1SKenneth D. Merry OCS_IO_TYPE_CT_RESP, 65*ef270ab1SKenneth D. Merry OCS_IO_TYPE_BLS_RESP, 66*ef270ab1SKenneth D. Merry OCS_IO_TYPE_ABORT, 67*ef270ab1SKenneth D. Merry 68*ef270ab1SKenneth D. Merry OCS_IO_TYPE_MAX, /**< must be last */ 69*ef270ab1SKenneth D. Merry } ocs_io_type_e; 70*ef270ab1SKenneth D. Merry 71*ef270ab1SKenneth D. Merry struct ocs_io_s { 72*ef270ab1SKenneth D. Merry 73*ef270ab1SKenneth D. Merry ocs_t *ocs; /**< pointer back to ocs */ 74*ef270ab1SKenneth D. Merry uint32_t instance_index; /**< unique instance index value */ 75*ef270ab1SKenneth D. Merry const char *display_name; /**< display name */ 76*ef270ab1SKenneth D. Merry ocs_node_t *node; /**< pointer to node */ 77*ef270ab1SKenneth D. Merry ocs_list_link_t io_alloc_link; /**< (io_pool->io_free_list) free list link */ 78*ef270ab1SKenneth D. Merry uint32_t init_task_tag; /**< initiator task tag (OX_ID) for back-end and SCSI logging */ 79*ef270ab1SKenneth D. Merry uint32_t tgt_task_tag; /**< target task tag (RX_ID) - for back-end and SCSI logging */ 80*ef270ab1SKenneth D. Merry uint32_t hw_tag; /**< HW layer unique IO id - for back-end and SCSI logging */ 81*ef270ab1SKenneth D. Merry uint32_t tag; /**< unique IO identifier */ 82*ef270ab1SKenneth D. Merry ocs_scsi_sgl_t *sgl; /**< SGL */ 83*ef270ab1SKenneth D. Merry uint32_t sgl_allocated; /**< Number of allocated SGEs */ 84*ef270ab1SKenneth D. Merry uint32_t sgl_count; /**< Number of SGEs in this SGL */ 85*ef270ab1SKenneth D. Merry ocs_scsi_ini_io_t ini_io; /**< backend initiator private IO data */ 86*ef270ab1SKenneth D. Merry ocs_scsi_tgt_io_t tgt_io; /**< backend target private IO data */ 87*ef270ab1SKenneth D. Merry uint32_t exp_xfer_len; /**< expected data transfer length, based on FC or iSCSI header */ 88*ef270ab1SKenneth D. Merry ocs_mgmt_functions_t *mgmt_functions; 89*ef270ab1SKenneth D. Merry 90*ef270ab1SKenneth D. Merry /* Declarations private to HW/SLI */ 91*ef270ab1SKenneth D. Merry void *hw_priv; /**< HW private context */ 92*ef270ab1SKenneth D. Merry 93*ef270ab1SKenneth D. Merry /* Declarations private to FC Transport */ 94*ef270ab1SKenneth D. Merry ocs_io_type_e io_type; /**< indicates what this ocs_io_t structure is used for */ 95*ef270ab1SKenneth D. Merry ocs_ref_t ref; /**< refcount object */ 96*ef270ab1SKenneth D. Merry void *dslab_item; /**< pointer back to dslab allocation object */ 97*ef270ab1SKenneth D. Merry ocs_hw_io_t *hio; /**< HW IO context */ 98*ef270ab1SKenneth D. Merry size_t transferred; /**< Number of bytes transferred so far */ 99*ef270ab1SKenneth D. Merry uint32_t auto_resp:1, /**< set if auto_trsp was set */ 100*ef270ab1SKenneth D. Merry low_latency:1, /**< set if low latency request */ 101*ef270ab1SKenneth D. Merry wq_steering:4, /**< selected WQ steering request */ 102*ef270ab1SKenneth D. Merry wq_class:4; /**< selected WQ class if steering is class */ 103*ef270ab1SKenneth D. Merry uint32_t xfer_req; /**< transfer size for current request */ 104*ef270ab1SKenneth D. Merry ocs_scsi_rsp_io_cb_t scsi_ini_cb; /**< initiator callback function */ 105*ef270ab1SKenneth D. Merry void *scsi_ini_cb_arg; /**< initiator callback function argument */ 106*ef270ab1SKenneth D. Merry ocs_scsi_io_cb_t scsi_tgt_cb; /**< target callback function */ 107*ef270ab1SKenneth D. Merry void *scsi_tgt_cb_arg; /**< target callback function argument */ 108*ef270ab1SKenneth D. Merry ocs_scsi_io_cb_t abort_cb; /**< abort callback function */ 109*ef270ab1SKenneth D. Merry void *abort_cb_arg; /**< abort callback function argument */ 110*ef270ab1SKenneth D. Merry ocs_scsi_io_cb_t bls_cb; /**< BLS callback function */ 111*ef270ab1SKenneth D. Merry void *bls_cb_arg; /**< BLS callback function argument */ 112*ef270ab1SKenneth D. Merry ocs_scsi_tmf_cmd_e tmf_cmd; /**< TMF command being processed */ 113*ef270ab1SKenneth D. Merry uint16_t abort_rx_id; /**< rx_id from the ABTS that initiated the command abort */ 114*ef270ab1SKenneth D. Merry 115*ef270ab1SKenneth D. Merry uint32_t cmd_tgt:1, /**< True if this is a Target command */ 116*ef270ab1SKenneth D. Merry send_abts:1, /**< when aborting, indicates ABTS is to be sent */ 117*ef270ab1SKenneth D. Merry cmd_ini:1, /**< True if this is an Initiator command */ 118*ef270ab1SKenneth D. Merry seq_init:1; /**< True if local node has sequence initiative */ 119*ef270ab1SKenneth D. Merry ocs_hw_io_param_t iparam; /**< iparams for hw io send call */ 120*ef270ab1SKenneth D. Merry ocs_hw_dif_info_t hw_dif; /**< HW formatted DIF parameters */ 121*ef270ab1SKenneth D. Merry ocs_scsi_dif_info_t scsi_dif_info; /**< DIF info saved for DIF error recovery */ 122*ef270ab1SKenneth D. Merry ocs_hw_io_type_e hio_type; /**< HW IO type */ 123*ef270ab1SKenneth D. Merry uint32_t wire_len; /**< wire length */ 124*ef270ab1SKenneth D. Merry void *hw_cb; /**< saved HW callback */ 125*ef270ab1SKenneth D. Merry ocs_list_link_t io_pending_link;/**< link list link pending */ 126*ef270ab1SKenneth D. Merry 127*ef270ab1SKenneth D. Merry ocs_dma_t ovfl_sgl; /**< Overflow SGL */ 128*ef270ab1SKenneth D. Merry 129*ef270ab1SKenneth D. Merry /* for ELS requests/responses */ 130*ef270ab1SKenneth D. Merry uint32_t els_pend:1, /**< True if ELS is pending */ 131*ef270ab1SKenneth D. Merry els_active:1; /**< True if ELS is active */ 132*ef270ab1SKenneth D. Merry ocs_dma_t els_req; /**< ELS request payload buffer */ 133*ef270ab1SKenneth D. Merry ocs_dma_t els_rsp; /**< ELS response payload buffer */ 134*ef270ab1SKenneth D. Merry ocs_sm_ctx_t els_sm; /**< EIO IO state machine context */ 135*ef270ab1SKenneth D. Merry uint32_t els_evtdepth; /**< current event posting nesting depth */ 136*ef270ab1SKenneth D. Merry uint32_t els_req_free:1; /**< this els is to be free'd */ 137*ef270ab1SKenneth D. Merry uint32_t els_retries_remaining; /*<< Retries remaining */ 138*ef270ab1SKenneth D. Merry void (*els_callback)(ocs_node_t *node, ocs_node_cb_t *cbdata, void *cbarg); 139*ef270ab1SKenneth D. Merry void *els_callback_arg; 140*ef270ab1SKenneth D. Merry uint32_t els_timeout_sec; /**< timeout */ 141*ef270ab1SKenneth D. Merry 142*ef270ab1SKenneth D. Merry ocs_timer_t delay_timer; /**< delay timer */ 143*ef270ab1SKenneth D. Merry 144*ef270ab1SKenneth D. Merry /* for abort handling */ 145*ef270ab1SKenneth D. Merry ocs_io_t *io_to_abort; /**< pointer to IO to abort */ 146*ef270ab1SKenneth D. Merry 147*ef270ab1SKenneth D. Merry ocs_list_link_t link; /**< linked list link */ 148*ef270ab1SKenneth D. Merry ocs_dma_t cmdbuf; /**< SCSI Command buffer, used for CDB (initiator) */ 149*ef270ab1SKenneth D. Merry ocs_dma_t rspbuf; /**< SCSI Response buffer (i+t) */ 150*ef270ab1SKenneth D. Merry uint32_t timeout; /**< Timeout value in seconds for this IO */ 151*ef270ab1SKenneth D. Merry uint8_t cs_ctl; /**< CS_CTL priority for this IO */ 152*ef270ab1SKenneth D. Merry uint8_t io_free; /**< Is io object in freelist > */ 153*ef270ab1SKenneth D. Merry uint32_t app_id; 154*ef270ab1SKenneth D. Merry }; 155*ef270ab1SKenneth D. Merry 156*ef270ab1SKenneth D. Merry /** 157*ef270ab1SKenneth D. Merry * @brief common IO callback argument 158*ef270ab1SKenneth D. Merry * 159*ef270ab1SKenneth D. Merry * Callback argument used as common I/O callback argument 160*ef270ab1SKenneth D. Merry */ 161*ef270ab1SKenneth D. Merry 162*ef270ab1SKenneth D. Merry typedef struct { 163*ef270ab1SKenneth D. Merry int32_t status; /**< completion status */ 164*ef270ab1SKenneth D. Merry int32_t ext_status; /**< extended completion status */ 165*ef270ab1SKenneth D. Merry void *app; /**< application argument */ 166*ef270ab1SKenneth D. Merry } ocs_io_cb_arg_t; 167*ef270ab1SKenneth D. Merry 168*ef270ab1SKenneth D. Merry /** 169*ef270ab1SKenneth D. Merry * @brief Test if IO object is busy 170*ef270ab1SKenneth D. Merry * 171*ef270ab1SKenneth D. Merry * Return True if IO object is busy. Busy is defined as the IO object not being on 172*ef270ab1SKenneth D. Merry * the free list 173*ef270ab1SKenneth D. Merry * 174*ef270ab1SKenneth D. Merry * @param io Pointer to IO object 175*ef270ab1SKenneth D. Merry * 176*ef270ab1SKenneth D. Merry * @return returns True if IO is busy 177*ef270ab1SKenneth D. Merry */ 178*ef270ab1SKenneth D. Merry 179*ef270ab1SKenneth D. Merry static inline int32_t 180*ef270ab1SKenneth D. Merry ocs_io_busy(ocs_io_t *io) 181*ef270ab1SKenneth D. Merry { 182*ef270ab1SKenneth D. Merry return !(io->io_free); 183*ef270ab1SKenneth D. Merry } 184*ef270ab1SKenneth D. Merry 185*ef270ab1SKenneth D. Merry typedef struct ocs_io_pool_s ocs_io_pool_t; 186*ef270ab1SKenneth D. Merry 187*ef270ab1SKenneth D. Merry extern ocs_io_pool_t *ocs_io_pool_create(ocs_t *ocs, uint32_t num_io, uint32_t num_sgl); 188*ef270ab1SKenneth D. Merry extern int32_t ocs_io_pool_free(ocs_io_pool_t *io_pool); 189*ef270ab1SKenneth D. Merry extern uint32_t ocs_io_pool_allocated(ocs_io_pool_t *io_pool); 190*ef270ab1SKenneth D. Merry 191*ef270ab1SKenneth D. Merry extern ocs_io_t *ocs_io_pool_io_alloc(ocs_io_pool_t *io_pool); 192*ef270ab1SKenneth D. Merry extern void ocs_io_pool_io_free(ocs_io_pool_t *io_pool, ocs_io_t *io); 193*ef270ab1SKenneth D. Merry extern ocs_io_t *ocs_io_find_tgt_io(ocs_t *ocs, ocs_node_t *node, uint16_t ox_id, uint16_t rx_id); 194*ef270ab1SKenneth D. Merry extern void ocs_ddump_io(ocs_textbuf_t *textbuf, ocs_io_t *io); 195*ef270ab1SKenneth D. Merry 196*ef270ab1SKenneth D. Merry #endif 197