123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416141714181419142014211422142314241425142614271428142914301431143214331434143514361437143814391440144114421443144414451446144714481449145014511452145314541455145614571458145914601461146214631464146514661467146814691470147114721473147414751476147714781479148014811482148314841485148614871488148914901491149214931494149514961497149814991500150115021503150415051506150715081509151015111512151315141515151615171518151915201521152215231524152515261527152815291530153115321533153415351536153715381539154015411542154315441545154615471548154915501551155215531554155515561557155815591560156115621563156415651566156715681569157015711572157315741575157615771578157915801581158215831584158515861587158815891590159115921593159415951596159715981599160016011602160316041605160616071608160916101611161216131614161516161617161816191620162116221623162416251626162716281629163016311632163316341635163616371638163916401641164216431644164516461647164816491650165116521653165416551656165716581659166016611662166316641665166616671668166916701671167216731674167516761677167816791680168116821683168416851686168716881689169016911692169316941695169616971698169917001701170217031704170517061707170817091710171117121713171417151716171717181719172017211722172317241725172617271728172917301731173217331734173517361737173817391740174117421743174417451746174717481749175017511752175317541755175617571758175917601761176217631764176517661767176817691770177117721773177417751776177717781779178017811782178317841785178617871788178917901791179217931794179517961797179817991800180118021803180418051806180718081809181018111812181318141815181618171818181918201821182218231824182518261827182818291830183118321833183418351836183718381839184018411842184318441845184618471848184918501851185218531854185518561857185818591860186118621863186418651866186718681869187018711872187318741875187618771878187918801881188218831884188518861887188818891890189118921893189418951896189718981899190019011902190319041905190619071908190919101911191219131914191519161917191819191920192119221923192419251926192719281929193019311932193319341935193619371938193919401941194219431944194519461947194819491950195119521953195419551956195719581959196019611962196319641965196619671968196919701971197219731974197519761977197819791980198119821983198419851986198719881989199019911992199319941995199619971998199920002001200220032004200520062007200820092010201120122013201420152016201720182019202020212022202320242025202620272028202920302031203220332034203520362037203820392040204120422043204420452046204720482049205020512052205320542055205620572058205920602061206220632064206520662067206820692070207120722073207420752076207720782079208020812082208320842085208620872088208920902091209220932094209520962097209820992100210121022103210421052106210721082109211021112112211321142115211621172118211921202121212221232124212521262127212821292130213121322133213421352136213721382139214021412142214321442145214621472148214921502151215221532154215521562157215821592160216121622163216421652166216721682169217021712172217321742175217621772178217921802181218221832184218521862187218821892190219121922193219421952196219721982199220022012202220322042205220622072208220922102211221222132214221522162217221822192220222122222223222422252226222722282229223022312232223322342235223622372238223922402241224222432244224522462247224822492250225122522253225422552256225722582259226022612262226322642265226622672268226922702271227222732274227522762277227822792280228122822283228422852286228722882289229022912292229322942295229622972298229923002301230223032304230523062307230823092310231123122313231423152316231723182319232023212322232323242325232623272328232923302331233223332334233523362337233823392340234123422343234423452346234723482349235023512352235323542355235623572358235923602361236223632364236523662367236823692370237123722373237423752376237723782379238023812382238323842385238623872388238923902391239223932394239523962397239823992400240124022403240424052406240724082409241024112412241324142415241624172418241924202421242224232424242524262427242824292430243124322433243424352436243724382439244024412442244324442445244624472448244924502451245224532454245524562457245824592460246124622463246424652466246724682469247024712472247324742475247624772478247924802481248224832484248524862487248824892490249124922493249424952496249724982499250025012502250325042505250625072508250925102511251225132514251525162517251825192520252125222523252425252526252725282529253025312532 |
- /*
- * This file is part of the Chelsio FCoE driver for Linux.
- *
- * Copyright (c) 2008-2012 Chelsio Communications, Inc. All rights reserved.
- *
- * This software is available to you under a choice of one of two
- * licenses. You may choose to be licensed under the terms of the GNU
- * General Public License (GPL) Version 2, available from the file
- * COPYING in the main directory of this source tree, or the
- * OpenIB.org BSD license below:
- *
- * Redistribution and use in source and binary forms, with or
- * without modification, are permitted provided that the following
- * conditions are met:
- *
- * - Redistributions of source code must retain the above
- * copyright notice, this list of conditions and the following
- * disclaimer.
- *
- * - Redistributions in binary form must reproduce the above
- * copyright notice, this list of conditions and the following
- * disclaimer in the documentation and/or other materials
- * provided with the distribution.
- *
- * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
- * EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
- * MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
- * NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS
- * BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN
- * ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN
- * CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
- * SOFTWARE.
- */
- #include <linux/device.h>
- #include <linux/delay.h>
- #include <linux/ctype.h>
- #include <linux/kernel.h>
- #include <linux/slab.h>
- #include <linux/string.h>
- #include <linux/compiler.h>
- #include <linux/export.h>
- #include <linux/module.h>
- #include <asm/unaligned.h>
- #include <asm/page.h>
- #include <scsi/scsi.h>
- #include <scsi/scsi_device.h>
- #include <scsi/scsi_transport_fc.h>
- #include "csio_hw.h"
- #include "csio_lnode.h"
- #include "csio_rnode.h"
- #include "csio_scsi.h"
- #include "csio_init.h"
- int csio_scsi_eqsize = 65536;
- int csio_scsi_iqlen = 128;
- int csio_scsi_ioreqs = 2048;
- uint32_t csio_max_scan_tmo;
- uint32_t csio_delta_scan_tmo = 5;
- int csio_lun_qdepth = 32;
- static int csio_ddp_descs = 128;
- static int csio_do_abrt_cls(struct csio_hw *,
- struct csio_ioreq *, bool);
- static void csio_scsis_uninit(struct csio_ioreq *, enum csio_scsi_ev);
- static void csio_scsis_io_active(struct csio_ioreq *, enum csio_scsi_ev);
- static void csio_scsis_tm_active(struct csio_ioreq *, enum csio_scsi_ev);
- static void csio_scsis_aborting(struct csio_ioreq *, enum csio_scsi_ev);
- static void csio_scsis_closing(struct csio_ioreq *, enum csio_scsi_ev);
- static void csio_scsis_shost_cmpl_await(struct csio_ioreq *, enum csio_scsi_ev);
- /*
- * csio_scsi_match_io - Match an ioreq with the given SCSI level data.
- * @ioreq: The I/O request
- * @sld: Level information
- *
- * Should be called with lock held.
- *
- */
- static bool
- csio_scsi_match_io(struct csio_ioreq *ioreq, struct csio_scsi_level_data *sld)
- {
- struct scsi_cmnd *scmnd = csio_scsi_cmnd(ioreq);
- switch (sld->level) {
- case CSIO_LEV_LUN:
- if (scmnd == NULL)
- return false;
- return ((ioreq->lnode == sld->lnode) &&
- (ioreq->rnode == sld->rnode) &&
- ((uint64_t)scmnd->device->lun == sld->oslun));
- case CSIO_LEV_RNODE:
- return ((ioreq->lnode == sld->lnode) &&
- (ioreq->rnode == sld->rnode));
- case CSIO_LEV_LNODE:
- return (ioreq->lnode == sld->lnode);
- case CSIO_LEV_ALL:
- return true;
- default:
- return false;
- }
- }
- /*
- * csio_scsi_gather_active_ios - Gather active I/Os based on level
- * @scm: SCSI module
- * @sld: Level information
- * @dest: The queue where these I/Os have to be gathered.
- *
- * Should be called with lock held.
- */
- static void
- csio_scsi_gather_active_ios(struct csio_scsim *scm,
- struct csio_scsi_level_data *sld,
- struct list_head *dest)
- {
- struct list_head *tmp, *next;
- if (list_empty(&scm->active_q))
- return;
- /* Just splice the entire active_q into dest */
- if (sld->level == CSIO_LEV_ALL) {
- list_splice_tail_init(&scm->active_q, dest);
- return;
- }
- list_for_each_safe(tmp, next, &scm->active_q) {
- if (csio_scsi_match_io((struct csio_ioreq *)tmp, sld)) {
- list_del_init(tmp);
- list_add_tail(tmp, dest);
- }
- }
- }
- static inline bool
- csio_scsi_itnexus_loss_error(uint16_t error)
- {
- switch (error) {
- case FW_ERR_LINK_DOWN:
- case FW_RDEV_NOT_READY:
- case FW_ERR_RDEV_LOST:
- case FW_ERR_RDEV_LOGO:
- case FW_ERR_RDEV_IMPL_LOGO:
- return 1;
- }
- return 0;
- }
- /*
- * csio_scsi_fcp_cmnd - Frame the SCSI FCP command paylod.
- * @req: IO req structure.
- * @addr: DMA location to place the payload.
- *
- * This routine is shared between FCP_WRITE, FCP_READ and FCP_CMD requests.
- */
- static inline void
- csio_scsi_fcp_cmnd(struct csio_ioreq *req, void *addr)
- {
- struct fcp_cmnd *fcp_cmnd = (struct fcp_cmnd *)addr;
- struct scsi_cmnd *scmnd = csio_scsi_cmnd(req);
- /* Check for Task Management */
- if (likely(scmnd->SCp.Message == 0)) {
- int_to_scsilun(scmnd->device->lun, &fcp_cmnd->fc_lun);
- fcp_cmnd->fc_tm_flags = 0;
- fcp_cmnd->fc_cmdref = 0;
- memcpy(fcp_cmnd->fc_cdb, scmnd->cmnd, 16);
- fcp_cmnd->fc_pri_ta = FCP_PTA_SIMPLE;
- fcp_cmnd->fc_dl = cpu_to_be32(scsi_bufflen(scmnd));
- if (req->nsge)
- if (req->datadir == DMA_TO_DEVICE)
- fcp_cmnd->fc_flags = FCP_CFL_WRDATA;
- else
- fcp_cmnd->fc_flags = FCP_CFL_RDDATA;
- else
- fcp_cmnd->fc_flags = 0;
- } else {
- memset(fcp_cmnd, 0, sizeof(*fcp_cmnd));
- int_to_scsilun(scmnd->device->lun, &fcp_cmnd->fc_lun);
- fcp_cmnd->fc_tm_flags = (uint8_t)scmnd->SCp.Message;
- }
- }
- /*
- * csio_scsi_init_cmd_wr - Initialize the SCSI CMD WR.
- * @req: IO req structure.
- * @addr: DMA location to place the payload.
- * @size: Size of WR (including FW WR + immed data + rsp SG entry
- *
- * Wrapper for populating fw_scsi_cmd_wr.
- */
- static inline void
- csio_scsi_init_cmd_wr(struct csio_ioreq *req, void *addr, uint32_t size)
- {
- struct csio_hw *hw = req->lnode->hwp;
- struct csio_rnode *rn = req->rnode;
- struct fw_scsi_cmd_wr *wr = (struct fw_scsi_cmd_wr *)addr;
- struct csio_dma_buf *dma_buf;
- uint8_t imm = csio_hw_to_scsim(hw)->proto_cmd_len;
- wr->op_immdlen = cpu_to_be32(FW_WR_OP_V(FW_SCSI_CMD_WR) |
- FW_SCSI_CMD_WR_IMMDLEN(imm));
- wr->flowid_len16 = cpu_to_be32(FW_WR_FLOWID_V(rn->flowid) |
- FW_WR_LEN16_V(
- DIV_ROUND_UP(size, 16)));
- wr->cookie = (uintptr_t) req;
- wr->iqid = cpu_to_be16(csio_q_physiqid(hw, req->iq_idx));
- wr->tmo_val = (uint8_t) req->tmo;
- wr->r3 = 0;
- memset(&wr->r5, 0, 8);
- /* Get RSP DMA buffer */
- dma_buf = &req->dma_buf;
- /* Prepare RSP SGL */
- wr->rsp_dmalen = cpu_to_be32(dma_buf->len);
- wr->rsp_dmaaddr = cpu_to_be64(dma_buf->paddr);
- wr->r6 = 0;
- wr->u.fcoe.ctl_pri = 0;
- wr->u.fcoe.cp_en_class = 0;
- wr->u.fcoe.r4_lo[0] = 0;
- wr->u.fcoe.r4_lo[1] = 0;
- /* Frame a FCP command */
- csio_scsi_fcp_cmnd(req, (void *)((uintptr_t)addr +
- sizeof(struct fw_scsi_cmd_wr)));
- }
- #define CSIO_SCSI_CMD_WR_SZ(_imm) \
- (sizeof(struct fw_scsi_cmd_wr) + /* WR size */ \
- ALIGN((_imm), 16)) /* Immed data */
- #define CSIO_SCSI_CMD_WR_SZ_16(_imm) \
- (ALIGN(CSIO_SCSI_CMD_WR_SZ((_imm)), 16))
- /*
- * csio_scsi_cmd - Create a SCSI CMD WR.
- * @req: IO req structure.
- *
- * Gets a WR slot in the ingress queue and initializes it with SCSI CMD WR.
- *
- */
- static inline void
- csio_scsi_cmd(struct csio_ioreq *req)
- {
- struct csio_wr_pair wrp;
- struct csio_hw *hw = req->lnode->hwp;
- struct csio_scsim *scsim = csio_hw_to_scsim(hw);
- uint32_t size = CSIO_SCSI_CMD_WR_SZ_16(scsim->proto_cmd_len);
- req->drv_status = csio_wr_get(hw, req->eq_idx, size, &wrp);
- if (unlikely(req->drv_status != 0))
- return;
- if (wrp.size1 >= size) {
- /* Initialize WR in one shot */
- csio_scsi_init_cmd_wr(req, wrp.addr1, size);
- } else {
- uint8_t *tmpwr = csio_q_eq_wrap(hw, req->eq_idx);
- /*
- * Make a temporary copy of the WR and write back
- * the copy into the WR pair.
- */
- csio_scsi_init_cmd_wr(req, (void *)tmpwr, size);
- memcpy(wrp.addr1, tmpwr, wrp.size1);
- memcpy(wrp.addr2, tmpwr + wrp.size1, size - wrp.size1);
- }
- }
- /*
- * csio_scsi_init_ulptx_dsgl - Fill in a ULP_TX_SC_DSGL
- * @hw: HW module
- * @req: IO request
- * @sgl: ULP TX SGL pointer.
- *
- */
- static inline void
- csio_scsi_init_ultptx_dsgl(struct csio_hw *hw, struct csio_ioreq *req,
- struct ulptx_sgl *sgl)
- {
- struct ulptx_sge_pair *sge_pair = NULL;
- struct scatterlist *sgel;
- uint32_t i = 0;
- uint32_t xfer_len;
- struct list_head *tmp;
- struct csio_dma_buf *dma_buf;
- struct scsi_cmnd *scmnd = csio_scsi_cmnd(req);
- sgl->cmd_nsge = htonl(ULPTX_CMD_V(ULP_TX_SC_DSGL) | ULPTX_MORE_F |
- ULPTX_NSGE_V(req->nsge));
- /* Now add the data SGLs */
- if (likely(!req->dcopy)) {
- scsi_for_each_sg(scmnd, sgel, req->nsge, i) {
- if (i == 0) {
- sgl->addr0 = cpu_to_be64(sg_dma_address(sgel));
- sgl->len0 = cpu_to_be32(sg_dma_len(sgel));
- sge_pair = (struct ulptx_sge_pair *)(sgl + 1);
- continue;
- }
- if ((i - 1) & 0x1) {
- sge_pair->addr[1] = cpu_to_be64(
- sg_dma_address(sgel));
- sge_pair->len[1] = cpu_to_be32(
- sg_dma_len(sgel));
- sge_pair++;
- } else {
- sge_pair->addr[0] = cpu_to_be64(
- sg_dma_address(sgel));
- sge_pair->len[0] = cpu_to_be32(
- sg_dma_len(sgel));
- }
- }
- } else {
- /* Program sg elements with driver's DDP buffer */
- xfer_len = scsi_bufflen(scmnd);
- list_for_each(tmp, &req->gen_list) {
- dma_buf = (struct csio_dma_buf *)tmp;
- if (i == 0) {
- sgl->addr0 = cpu_to_be64(dma_buf->paddr);
- sgl->len0 = cpu_to_be32(
- min(xfer_len, dma_buf->len));
- sge_pair = (struct ulptx_sge_pair *)(sgl + 1);
- } else if ((i - 1) & 0x1) {
- sge_pair->addr[1] = cpu_to_be64(dma_buf->paddr);
- sge_pair->len[1] = cpu_to_be32(
- min(xfer_len, dma_buf->len));
- sge_pair++;
- } else {
- sge_pair->addr[0] = cpu_to_be64(dma_buf->paddr);
- sge_pair->len[0] = cpu_to_be32(
- min(xfer_len, dma_buf->len));
- }
- xfer_len -= min(xfer_len, dma_buf->len);
- i++;
- }
- }
- }
- /*
- * csio_scsi_init_read_wr - Initialize the READ SCSI WR.
- * @req: IO req structure.
- * @wrp: DMA location to place the payload.
- * @size: Size of WR (including FW WR + immed data + rsp SG entry + data SGL
- *
- * Wrapper for populating fw_scsi_read_wr.
- */
- static inline void
- csio_scsi_init_read_wr(struct csio_ioreq *req, void *wrp, uint32_t size)
- {
- struct csio_hw *hw = req->lnode->hwp;
- struct csio_rnode *rn = req->rnode;
- struct fw_scsi_read_wr *wr = (struct fw_scsi_read_wr *)wrp;
- struct ulptx_sgl *sgl;
- struct csio_dma_buf *dma_buf;
- uint8_t imm = csio_hw_to_scsim(hw)->proto_cmd_len;
- struct scsi_cmnd *scmnd = csio_scsi_cmnd(req);
- wr->op_immdlen = cpu_to_be32(FW_WR_OP_V(FW_SCSI_READ_WR) |
- FW_SCSI_READ_WR_IMMDLEN(imm));
- wr->flowid_len16 = cpu_to_be32(FW_WR_FLOWID_V(rn->flowid) |
- FW_WR_LEN16_V(DIV_ROUND_UP(size, 16)));
- wr->cookie = (uintptr_t)req;
- wr->iqid = cpu_to_be16(csio_q_physiqid(hw, req->iq_idx));
- wr->tmo_val = (uint8_t)(req->tmo);
- wr->use_xfer_cnt = 1;
- wr->xfer_cnt = cpu_to_be32(scsi_bufflen(scmnd));
- wr->ini_xfer_cnt = cpu_to_be32(scsi_bufflen(scmnd));
- /* Get RSP DMA buffer */
- dma_buf = &req->dma_buf;
- /* Prepare RSP SGL */
- wr->rsp_dmalen = cpu_to_be32(dma_buf->len);
- wr->rsp_dmaaddr = cpu_to_be64(dma_buf->paddr);
- wr->r4 = 0;
- wr->u.fcoe.ctl_pri = 0;
- wr->u.fcoe.cp_en_class = 0;
- wr->u.fcoe.r3_lo[0] = 0;
- wr->u.fcoe.r3_lo[1] = 0;
- csio_scsi_fcp_cmnd(req, (void *)((uintptr_t)wrp +
- sizeof(struct fw_scsi_read_wr)));
- /* Move WR pointer past command and immediate data */
- sgl = (struct ulptx_sgl *)((uintptr_t)wrp +
- sizeof(struct fw_scsi_read_wr) + ALIGN(imm, 16));
- /* Fill in the DSGL */
- csio_scsi_init_ultptx_dsgl(hw, req, sgl);
- }
- /*
- * csio_scsi_init_write_wr - Initialize the WRITE SCSI WR.
- * @req: IO req structure.
- * @wrp: DMA location to place the payload.
- * @size: Size of WR (including FW WR + immed data + rsp SG entry + data SGL
- *
- * Wrapper for populating fw_scsi_write_wr.
- */
- static inline void
- csio_scsi_init_write_wr(struct csio_ioreq *req, void *wrp, uint32_t size)
- {
- struct csio_hw *hw = req->lnode->hwp;
- struct csio_rnode *rn = req->rnode;
- struct fw_scsi_write_wr *wr = (struct fw_scsi_write_wr *)wrp;
- struct ulptx_sgl *sgl;
- struct csio_dma_buf *dma_buf;
- uint8_t imm = csio_hw_to_scsim(hw)->proto_cmd_len;
- struct scsi_cmnd *scmnd = csio_scsi_cmnd(req);
- wr->op_immdlen = cpu_to_be32(FW_WR_OP_V(FW_SCSI_WRITE_WR) |
- FW_SCSI_WRITE_WR_IMMDLEN(imm));
- wr->flowid_len16 = cpu_to_be32(FW_WR_FLOWID_V(rn->flowid) |
- FW_WR_LEN16_V(DIV_ROUND_UP(size, 16)));
- wr->cookie = (uintptr_t)req;
- wr->iqid = cpu_to_be16(csio_q_physiqid(hw, req->iq_idx));
- wr->tmo_val = (uint8_t)(req->tmo);
- wr->use_xfer_cnt = 1;
- wr->xfer_cnt = cpu_to_be32(scsi_bufflen(scmnd));
- wr->ini_xfer_cnt = cpu_to_be32(scsi_bufflen(scmnd));
- /* Get RSP DMA buffer */
- dma_buf = &req->dma_buf;
- /* Prepare RSP SGL */
- wr->rsp_dmalen = cpu_to_be32(dma_buf->len);
- wr->rsp_dmaaddr = cpu_to_be64(dma_buf->paddr);
- wr->r4 = 0;
- wr->u.fcoe.ctl_pri = 0;
- wr->u.fcoe.cp_en_class = 0;
- wr->u.fcoe.r3_lo[0] = 0;
- wr->u.fcoe.r3_lo[1] = 0;
- csio_scsi_fcp_cmnd(req, (void *)((uintptr_t)wrp +
- sizeof(struct fw_scsi_write_wr)));
- /* Move WR pointer past command and immediate data */
- sgl = (struct ulptx_sgl *)((uintptr_t)wrp +
- sizeof(struct fw_scsi_write_wr) + ALIGN(imm, 16));
- /* Fill in the DSGL */
- csio_scsi_init_ultptx_dsgl(hw, req, sgl);
- }
- /* Calculate WR size needed for fw_scsi_read_wr/fw_scsi_write_wr */
- #define CSIO_SCSI_DATA_WRSZ(req, oper, sz, imm) \
- do { \
- (sz) = sizeof(struct fw_scsi_##oper##_wr) + /* WR size */ \
- ALIGN((imm), 16) + /* Immed data */ \
- sizeof(struct ulptx_sgl); /* ulptx_sgl */ \
- \
- if (unlikely((req)->nsge > 1)) \
- (sz) += (sizeof(struct ulptx_sge_pair) * \
- (ALIGN(((req)->nsge - 1), 2) / 2)); \
- /* Data SGE */ \
- } while (0)
- /*
- * csio_scsi_read - Create a SCSI READ WR.
- * @req: IO req structure.
- *
- * Gets a WR slot in the ingress queue and initializes it with
- * SCSI READ WR.
- *
- */
- static inline void
- csio_scsi_read(struct csio_ioreq *req)
- {
- struct csio_wr_pair wrp;
- uint32_t size;
- struct csio_hw *hw = req->lnode->hwp;
- struct csio_scsim *scsim = csio_hw_to_scsim(hw);
- CSIO_SCSI_DATA_WRSZ(req, read, size, scsim->proto_cmd_len);
- size = ALIGN(size, 16);
- req->drv_status = csio_wr_get(hw, req->eq_idx, size, &wrp);
- if (likely(req->drv_status == 0)) {
- if (likely(wrp.size1 >= size)) {
- /* Initialize WR in one shot */
- csio_scsi_init_read_wr(req, wrp.addr1, size);
- } else {
- uint8_t *tmpwr = csio_q_eq_wrap(hw, req->eq_idx);
- /*
- * Make a temporary copy of the WR and write back
- * the copy into the WR pair.
- */
- csio_scsi_init_read_wr(req, (void *)tmpwr, size);
- memcpy(wrp.addr1, tmpwr, wrp.size1);
- memcpy(wrp.addr2, tmpwr + wrp.size1, size - wrp.size1);
- }
- }
- }
- /*
- * csio_scsi_write - Create a SCSI WRITE WR.
- * @req: IO req structure.
- *
- * Gets a WR slot in the ingress queue and initializes it with
- * SCSI WRITE WR.
- *
- */
- static inline void
- csio_scsi_write(struct csio_ioreq *req)
- {
- struct csio_wr_pair wrp;
- uint32_t size;
- struct csio_hw *hw = req->lnode->hwp;
- struct csio_scsim *scsim = csio_hw_to_scsim(hw);
- CSIO_SCSI_DATA_WRSZ(req, write, size, scsim->proto_cmd_len);
- size = ALIGN(size, 16);
- req->drv_status = csio_wr_get(hw, req->eq_idx, size, &wrp);
- if (likely(req->drv_status == 0)) {
- if (likely(wrp.size1 >= size)) {
- /* Initialize WR in one shot */
- csio_scsi_init_write_wr(req, wrp.addr1, size);
- } else {
- uint8_t *tmpwr = csio_q_eq_wrap(hw, req->eq_idx);
- /*
- * Make a temporary copy of the WR and write back
- * the copy into the WR pair.
- */
- csio_scsi_init_write_wr(req, (void *)tmpwr, size);
- memcpy(wrp.addr1, tmpwr, wrp.size1);
- memcpy(wrp.addr2, tmpwr + wrp.size1, size - wrp.size1);
- }
- }
- }
- /*
- * csio_setup_ddp - Setup DDP buffers for Read request.
- * @req: IO req structure.
- *
- * Checks SGLs/Data buffers are virtually contiguous required for DDP.
- * If contiguous,driver posts SGLs in the WR otherwise post internal
- * buffers for such request for DDP.
- */
- static inline void
- csio_setup_ddp(struct csio_scsim *scsim, struct csio_ioreq *req)
- {
- #ifdef __CSIO_DEBUG__
- struct csio_hw *hw = req->lnode->hwp;
- #endif
- struct scatterlist *sgel = NULL;
- struct scsi_cmnd *scmnd = csio_scsi_cmnd(req);
- uint64_t sg_addr = 0;
- uint32_t ddp_pagesz = 4096;
- uint32_t buf_off;
- struct csio_dma_buf *dma_buf = NULL;
- uint32_t alloc_len = 0;
- uint32_t xfer_len = 0;
- uint32_t sg_len = 0;
- uint32_t i;
- scsi_for_each_sg(scmnd, sgel, req->nsge, i) {
- sg_addr = sg_dma_address(sgel);
- sg_len = sg_dma_len(sgel);
- buf_off = sg_addr & (ddp_pagesz - 1);
- /* Except 1st buffer,all buffer addr have to be Page aligned */
- if (i != 0 && buf_off) {
- csio_dbg(hw, "SGL addr not DDP aligned (%llx:%d)\n",
- sg_addr, sg_len);
- goto unaligned;
- }
- /* Except last buffer,all buffer must end on page boundary */
- if ((i != (req->nsge - 1)) &&
- ((buf_off + sg_len) & (ddp_pagesz - 1))) {
- csio_dbg(hw,
- "SGL addr not ending on page boundary"
- "(%llx:%d)\n", sg_addr, sg_len);
- goto unaligned;
- }
- }
- /* SGL's are virtually contiguous. HW will DDP to SGLs */
- req->dcopy = 0;
- csio_scsi_read(req);
- return;
- unaligned:
- CSIO_INC_STATS(scsim, n_unaligned);
- /*
- * For unaligned SGLs, driver will allocate internal DDP buffer.
- * Once command is completed data from DDP buffer copied to SGLs
- */
- req->dcopy = 1;
- /* Use gen_list to store the DDP buffers */
- INIT_LIST_HEAD(&req->gen_list);
- xfer_len = scsi_bufflen(scmnd);
- i = 0;
- /* Allocate ddp buffers for this request */
- while (alloc_len < xfer_len) {
- dma_buf = csio_get_scsi_ddp(scsim);
- if (dma_buf == NULL || i > scsim->max_sge) {
- req->drv_status = -EBUSY;
- break;
- }
- alloc_len += dma_buf->len;
- /* Added to IO req */
- list_add_tail(&dma_buf->list, &req->gen_list);
- i++;
- }
- if (!req->drv_status) {
- /* set number of ddp bufs used */
- req->nsge = i;
- csio_scsi_read(req);
- return;
- }
- /* release dma descs */
- if (i > 0)
- csio_put_scsi_ddp_list(scsim, &req->gen_list, i);
- }
- /*
- * csio_scsi_init_abrt_cls_wr - Initialize an ABORT/CLOSE WR.
- * @req: IO req structure.
- * @addr: DMA location to place the payload.
- * @size: Size of WR
- * @abort: abort OR close
- *
- * Wrapper for populating fw_scsi_cmd_wr.
- */
- static inline void
- csio_scsi_init_abrt_cls_wr(struct csio_ioreq *req, void *addr, uint32_t size,
- bool abort)
- {
- struct csio_hw *hw = req->lnode->hwp;
- struct csio_rnode *rn = req->rnode;
- struct fw_scsi_abrt_cls_wr *wr = (struct fw_scsi_abrt_cls_wr *)addr;
- wr->op_immdlen = cpu_to_be32(FW_WR_OP_V(FW_SCSI_ABRT_CLS_WR));
- wr->flowid_len16 = cpu_to_be32(FW_WR_FLOWID_V(rn->flowid) |
- FW_WR_LEN16_V(
- DIV_ROUND_UP(size, 16)));
- wr->cookie = (uintptr_t) req;
- wr->iqid = cpu_to_be16(csio_q_physiqid(hw, req->iq_idx));
- wr->tmo_val = (uint8_t) req->tmo;
- /* 0 for CHK_ALL_IO tells FW to look up t_cookie */
- wr->sub_opcode_to_chk_all_io =
- (FW_SCSI_ABRT_CLS_WR_SUB_OPCODE(abort) |
- FW_SCSI_ABRT_CLS_WR_CHK_ALL_IO(0));
- wr->r3[0] = 0;
- wr->r3[1] = 0;
- wr->r3[2] = 0;
- wr->r3[3] = 0;
- /* Since we re-use the same ioreq for abort as well */
- wr->t_cookie = (uintptr_t) req;
- }
- static inline void
- csio_scsi_abrt_cls(struct csio_ioreq *req, bool abort)
- {
- struct csio_wr_pair wrp;
- struct csio_hw *hw = req->lnode->hwp;
- uint32_t size = ALIGN(sizeof(struct fw_scsi_abrt_cls_wr), 16);
- req->drv_status = csio_wr_get(hw, req->eq_idx, size, &wrp);
- if (req->drv_status != 0)
- return;
- if (wrp.size1 >= size) {
- /* Initialize WR in one shot */
- csio_scsi_init_abrt_cls_wr(req, wrp.addr1, size, abort);
- } else {
- uint8_t *tmpwr = csio_q_eq_wrap(hw, req->eq_idx);
- /*
- * Make a temporary copy of the WR and write back
- * the copy into the WR pair.
- */
- csio_scsi_init_abrt_cls_wr(req, (void *)tmpwr, size, abort);
- memcpy(wrp.addr1, tmpwr, wrp.size1);
- memcpy(wrp.addr2, tmpwr + wrp.size1, size - wrp.size1);
- }
- }
- /*****************************************************************************/
- /* START: SCSI SM */
- /*****************************************************************************/
- static void
- csio_scsis_uninit(struct csio_ioreq *req, enum csio_scsi_ev evt)
- {
- struct csio_hw *hw = req->lnode->hwp;
- struct csio_scsim *scsim = csio_hw_to_scsim(hw);
- switch (evt) {
- case CSIO_SCSIE_START_IO:
- if (req->nsge) {
- if (req->datadir == DMA_TO_DEVICE) {
- req->dcopy = 0;
- csio_scsi_write(req);
- } else
- csio_setup_ddp(scsim, req);
- } else {
- csio_scsi_cmd(req);
- }
- if (likely(req->drv_status == 0)) {
- /* change state and enqueue on active_q */
- csio_set_state(&req->sm, csio_scsis_io_active);
- list_add_tail(&req->sm.sm_list, &scsim->active_q);
- csio_wr_issue(hw, req->eq_idx, false);
- CSIO_INC_STATS(scsim, n_active);
- return;
- }
- break;
- case CSIO_SCSIE_START_TM:
- csio_scsi_cmd(req);
- if (req->drv_status == 0) {
- /*
- * NOTE: We collect the affected I/Os prior to issuing
- * LUN reset, and not after it. This is to prevent
- * aborting I/Os that get issued after the LUN reset,
- * but prior to LUN reset completion (in the event that
- * the host stack has not blocked I/Os to a LUN that is
- * being reset.
- */
- csio_set_state(&req->sm, csio_scsis_tm_active);
- list_add_tail(&req->sm.sm_list, &scsim->active_q);
- csio_wr_issue(hw, req->eq_idx, false);
- CSIO_INC_STATS(scsim, n_tm_active);
- }
- return;
- case CSIO_SCSIE_ABORT:
- case CSIO_SCSIE_CLOSE:
- /*
- * NOTE:
- * We could get here due to :
- * - a window in the cleanup path of the SCSI module
- * (csio_scsi_abort_io()). Please see NOTE in this function.
- * - a window in the time we tried to issue an abort/close
- * of a request to FW, and the FW completed the request
- * itself.
- * Print a message for now, and return INVAL either way.
- */
- req->drv_status = -EINVAL;
- csio_warn(hw, "Trying to abort/close completed IO:%p!\n", req);
- break;
- default:
- csio_dbg(hw, "Unhandled event:%d sent to req:%p\n", evt, req);
- CSIO_DB_ASSERT(0);
- }
- }
- static void
- csio_scsis_io_active(struct csio_ioreq *req, enum csio_scsi_ev evt)
- {
- struct csio_hw *hw = req->lnode->hwp;
- struct csio_scsim *scm = csio_hw_to_scsim(hw);
- struct csio_rnode *rn;
- switch (evt) {
- case CSIO_SCSIE_COMPLETED:
- CSIO_DEC_STATS(scm, n_active);
- list_del_init(&req->sm.sm_list);
- csio_set_state(&req->sm, csio_scsis_uninit);
- /*
- * In MSIX mode, with multiple queues, the SCSI compeltions
- * could reach us sooner than the FW events sent to indicate
- * I-T nexus loss (link down, remote device logo etc). We
- * dont want to be returning such I/Os to the upper layer
- * immediately, since we wouldnt have reported the I-T nexus
- * loss itself. This forces us to serialize such completions
- * with the reporting of the I-T nexus loss. Therefore, we
- * internally queue up such up such completions in the rnode.
- * The reporting of I-T nexus loss to the upper layer is then
- * followed by the returning of I/Os in this internal queue.
- * Having another state alongwith another queue helps us take
- * actions for events such as ABORT received while we are
- * in this rnode queue.
- */
- if (unlikely(req->wr_status != FW_SUCCESS)) {
- rn = req->rnode;
- /*
- * FW says remote device is lost, but rnode
- * doesnt reflect it.
- */
- if (csio_scsi_itnexus_loss_error(req->wr_status) &&
- csio_is_rnode_ready(rn)) {
- csio_set_state(&req->sm,
- csio_scsis_shost_cmpl_await);
- list_add_tail(&req->sm.sm_list,
- &rn->host_cmpl_q);
- }
- }
- break;
- case CSIO_SCSIE_ABORT:
- csio_scsi_abrt_cls(req, SCSI_ABORT);
- if (req->drv_status == 0) {
- csio_wr_issue(hw, req->eq_idx, false);
- csio_set_state(&req->sm, csio_scsis_aborting);
- }
- break;
- case CSIO_SCSIE_CLOSE:
- csio_scsi_abrt_cls(req, SCSI_CLOSE);
- if (req->drv_status == 0) {
- csio_wr_issue(hw, req->eq_idx, false);
- csio_set_state(&req->sm, csio_scsis_closing);
- }
- break;
- case CSIO_SCSIE_DRVCLEANUP:
- req->wr_status = FW_HOSTERROR;
- CSIO_DEC_STATS(scm, n_active);
- csio_set_state(&req->sm, csio_scsis_uninit);
- break;
- default:
- csio_dbg(hw, "Unhandled event:%d sent to req:%p\n", evt, req);
- CSIO_DB_ASSERT(0);
- }
- }
- static void
- csio_scsis_tm_active(struct csio_ioreq *req, enum csio_scsi_ev evt)
- {
- struct csio_hw *hw = req->lnode->hwp;
- struct csio_scsim *scm = csio_hw_to_scsim(hw);
- switch (evt) {
- case CSIO_SCSIE_COMPLETED:
- CSIO_DEC_STATS(scm, n_tm_active);
- list_del_init(&req->sm.sm_list);
- csio_set_state(&req->sm, csio_scsis_uninit);
- break;
- case CSIO_SCSIE_ABORT:
- csio_scsi_abrt_cls(req, SCSI_ABORT);
- if (req->drv_status == 0) {
- csio_wr_issue(hw, req->eq_idx, false);
- csio_set_state(&req->sm, csio_scsis_aborting);
- }
- break;
- case CSIO_SCSIE_CLOSE:
- csio_scsi_abrt_cls(req, SCSI_CLOSE);
- if (req->drv_status == 0) {
- csio_wr_issue(hw, req->eq_idx, false);
- csio_set_state(&req->sm, csio_scsis_closing);
- }
- break;
- case CSIO_SCSIE_DRVCLEANUP:
- req->wr_status = FW_HOSTERROR;
- CSIO_DEC_STATS(scm, n_tm_active);
- csio_set_state(&req->sm, csio_scsis_uninit);
- break;
- default:
- csio_dbg(hw, "Unhandled event:%d sent to req:%p\n", evt, req);
- CSIO_DB_ASSERT(0);
- }
- }
- static void
- csio_scsis_aborting(struct csio_ioreq *req, enum csio_scsi_ev evt)
- {
- struct csio_hw *hw = req->lnode->hwp;
- struct csio_scsim *scm = csio_hw_to_scsim(hw);
- switch (evt) {
- case CSIO_SCSIE_COMPLETED:
- csio_dbg(hw,
- "ioreq %p recvd cmpltd (wr_status:%d) "
- "in aborting st\n", req, req->wr_status);
- /*
- * Use -ECANCELED to explicitly tell the ABORTED event that
- * the original I/O was returned to driver by FW.
- * We dont really care if the I/O was returned with success by
- * FW (because the ABORT and completion of the I/O crossed each
- * other), or any other return value. Once we are in aborting
- * state, the success or failure of the I/O is unimportant to
- * us.
- */
- req->drv_status = -ECANCELED;
- break;
- case CSIO_SCSIE_ABORT:
- CSIO_INC_STATS(scm, n_abrt_dups);
- break;
- case CSIO_SCSIE_ABORTED:
- csio_dbg(hw, "abort of %p return status:0x%x drv_status:%x\n",
- req, req->wr_status, req->drv_status);
- /*
- * Check if original I/O WR completed before the Abort
- * completion.
- */
- if (req->drv_status != -ECANCELED) {
- csio_warn(hw,
- "Abort completed before original I/O,"
- " req:%p\n", req);
- CSIO_DB_ASSERT(0);
- }
- /*
- * There are the following possible scenarios:
- * 1. The abort completed successfully, FW returned FW_SUCCESS.
- * 2. The completion of an I/O and the receipt of
- * abort for that I/O by the FW crossed each other.
- * The FW returned FW_EINVAL. The original I/O would have
- * returned with FW_SUCCESS or any other SCSI error.
- * 3. The FW couldnt sent the abort out on the wire, as there
- * was an I-T nexus loss (link down, remote device logged
- * out etc). FW sent back an appropriate IT nexus loss status
- * for the abort.
- * 4. FW sent an abort, but abort timed out (remote device
- * didnt respond). FW replied back with
- * FW_SCSI_ABORT_TIMEDOUT.
- * 5. FW couldnt genuinely abort the request for some reason,
- * and sent us an error.
- *
- * The first 3 scenarios are treated as succesful abort
- * operations by the host, while the last 2 are failed attempts
- * to abort. Manipulate the return value of the request
- * appropriately, so that host can convey these results
- * back to the upper layer.
- */
- if ((req->wr_status == FW_SUCCESS) ||
- (req->wr_status == FW_EINVAL) ||
- csio_scsi_itnexus_loss_error(req->wr_status))
- req->wr_status = FW_SCSI_ABORT_REQUESTED;
- CSIO_DEC_STATS(scm, n_active);
- list_del_init(&req->sm.sm_list);
- csio_set_state(&req->sm, csio_scsis_uninit);
- break;
- case CSIO_SCSIE_DRVCLEANUP:
- req->wr_status = FW_HOSTERROR;
- CSIO_DEC_STATS(scm, n_active);
- csio_set_state(&req->sm, csio_scsis_uninit);
- break;
- case CSIO_SCSIE_CLOSE:
- /*
- * We can receive this event from the module
- * cleanup paths, if the FW forgot to reply to the ABORT WR
- * and left this ioreq in this state. For now, just ignore
- * the event. The CLOSE event is sent to this state, as
- * the LINK may have already gone down.
- */
- break;
- default:
- csio_dbg(hw, "Unhandled event:%d sent to req:%p\n", evt, req);
- CSIO_DB_ASSERT(0);
- }
- }
- static void
- csio_scsis_closing(struct csio_ioreq *req, enum csio_scsi_ev evt)
- {
- struct csio_hw *hw = req->lnode->hwp;
- struct csio_scsim *scm = csio_hw_to_scsim(hw);
- switch (evt) {
- case CSIO_SCSIE_COMPLETED:
- csio_dbg(hw,
- "ioreq %p recvd cmpltd (wr_status:%d) "
- "in closing st\n", req, req->wr_status);
- /*
- * Use -ECANCELED to explicitly tell the CLOSED event that
- * the original I/O was returned to driver by FW.
- * We dont really care if the I/O was returned with success by
- * FW (because the CLOSE and completion of the I/O crossed each
- * other), or any other return value. Once we are in aborting
- * state, the success or failure of the I/O is unimportant to
- * us.
- */
- req->drv_status = -ECANCELED;
- break;
- case CSIO_SCSIE_CLOSED:
- /*
- * Check if original I/O WR completed before the Close
- * completion.
- */
- if (req->drv_status != -ECANCELED) {
- csio_fatal(hw,
- "Close completed before original I/O,"
- " req:%p\n", req);
- CSIO_DB_ASSERT(0);
- }
- /*
- * Either close succeeded, or we issued close to FW at the
- * same time FW compelted it to us. Either way, the I/O
- * is closed.
- */
- CSIO_DB_ASSERT((req->wr_status == FW_SUCCESS) ||
- (req->wr_status == FW_EINVAL));
- req->wr_status = FW_SCSI_CLOSE_REQUESTED;
- CSIO_DEC_STATS(scm, n_active);
- list_del_init(&req->sm.sm_list);
- csio_set_state(&req->sm, csio_scsis_uninit);
- break;
- case CSIO_SCSIE_CLOSE:
- break;
- case CSIO_SCSIE_DRVCLEANUP:
- req->wr_status = FW_HOSTERROR;
- CSIO_DEC_STATS(scm, n_active);
- csio_set_state(&req->sm, csio_scsis_uninit);
- break;
- default:
- csio_dbg(hw, "Unhandled event:%d sent to req:%p\n", evt, req);
- CSIO_DB_ASSERT(0);
- }
- }
- static void
- csio_scsis_shost_cmpl_await(struct csio_ioreq *req, enum csio_scsi_ev evt)
- {
- switch (evt) {
- case CSIO_SCSIE_ABORT:
- case CSIO_SCSIE_CLOSE:
- /*
- * Just succeed the abort request, and hope that
- * the remote device unregister path will cleanup
- * this I/O to the upper layer within a sane
- * amount of time.
- */
- /*
- * A close can come in during a LINK DOWN. The FW would have
- * returned us the I/O back, but not the remote device lost
- * FW event. In this interval, if the I/O times out at the upper
- * layer, a close can come in. Take the same action as abort:
- * return success, and hope that the remote device unregister
- * path will cleanup this I/O. If the FW still doesnt send
- * the msg, the close times out, and the upper layer resorts
- * to the next level of error recovery.
- */
- req->drv_status = 0;
- break;
- case CSIO_SCSIE_DRVCLEANUP:
- csio_set_state(&req->sm, csio_scsis_uninit);
- break;
- default:
- csio_dbg(req->lnode->hwp, "Unhandled event:%d sent to req:%p\n",
- evt, req);
- CSIO_DB_ASSERT(0);
- }
- }
- /*
- * csio_scsi_cmpl_handler - WR completion handler for SCSI.
- * @hw: HW module.
- * @wr: The completed WR from the ingress queue.
- * @len: Length of the WR.
- * @flb: Freelist buffer array.
- * @priv: Private object
- * @scsiwr: Pointer to SCSI WR.
- *
- * This is the WR completion handler called per completion from the
- * ISR. It is called with lock held. It walks past the RSS and CPL message
- * header where the actual WR is present.
- * It then gets the status, WR handle (ioreq pointer) and the len of
- * the WR, based on WR opcode. Only on a non-good status is the entire
- * WR copied into the WR cache (ioreq->fw_wr).
- * The ioreq corresponding to the WR is returned to the caller.
- * NOTE: The SCSI queue doesnt allocate a freelist today, hence
- * no freelist buffer is expected.
- */
- struct csio_ioreq *
- csio_scsi_cmpl_handler(struct csio_hw *hw, void *wr, uint32_t len,
- struct csio_fl_dma_buf *flb, void *priv, uint8_t **scsiwr)
- {
- struct csio_ioreq *ioreq = NULL;
- struct cpl_fw6_msg *cpl;
- uint8_t *tempwr;
- uint8_t status;
- struct csio_scsim *scm = csio_hw_to_scsim(hw);
- /* skip RSS header */
- cpl = (struct cpl_fw6_msg *)((uintptr_t)wr + sizeof(__be64));
- if (unlikely(cpl->opcode != CPL_FW6_MSG)) {
- csio_warn(hw, "Error: Invalid CPL msg %x recvd on SCSI q\n",
- cpl->opcode);
- CSIO_INC_STATS(scm, n_inval_cplop);
- return NULL;
- }
- tempwr = (uint8_t *)(cpl->data);
- status = csio_wr_status(tempwr);
- *scsiwr = tempwr;
- if (likely((*tempwr == FW_SCSI_READ_WR) ||
- (*tempwr == FW_SCSI_WRITE_WR) ||
- (*tempwr == FW_SCSI_CMD_WR))) {
- ioreq = (struct csio_ioreq *)((uintptr_t)
- (((struct fw_scsi_read_wr *)tempwr)->cookie));
- CSIO_DB_ASSERT(virt_addr_valid(ioreq));
- ioreq->wr_status = status;
- return ioreq;
- }
- if (*tempwr == FW_SCSI_ABRT_CLS_WR) {
- ioreq = (struct csio_ioreq *)((uintptr_t)
- (((struct fw_scsi_abrt_cls_wr *)tempwr)->cookie));
- CSIO_DB_ASSERT(virt_addr_valid(ioreq));
- ioreq->wr_status = status;
- return ioreq;
- }
- csio_warn(hw, "WR with invalid opcode in SCSI IQ: %x\n", *tempwr);
- CSIO_INC_STATS(scm, n_inval_scsiop);
- return NULL;
- }
- /*
- * csio_scsi_cleanup_io_q - Cleanup the given queue.
- * @scm: SCSI module.
- * @q: Queue to be cleaned up.
- *
- * Called with lock held. Has to exit with lock held.
- */
- void
- csio_scsi_cleanup_io_q(struct csio_scsim *scm, struct list_head *q)
- {
- struct csio_hw *hw = scm->hw;
- struct csio_ioreq *ioreq;
- struct list_head *tmp, *next;
- struct scsi_cmnd *scmnd;
- /* Call back the completion routines of the active_q */
- list_for_each_safe(tmp, next, q) {
- ioreq = (struct csio_ioreq *)tmp;
- csio_scsi_drvcleanup(ioreq);
- list_del_init(&ioreq->sm.sm_list);
- scmnd = csio_scsi_cmnd(ioreq);
- spin_unlock_irq(&hw->lock);
- /*
- * Upper layers may have cleared this command, hence this
- * check to avoid accessing stale references.
- */
- if (scmnd != NULL)
- ioreq->io_cbfn(hw, ioreq);
- spin_lock_irq(&scm->freelist_lock);
- csio_put_scsi_ioreq(scm, ioreq);
- spin_unlock_irq(&scm->freelist_lock);
- spin_lock_irq(&hw->lock);
- }
- }
- #define CSIO_SCSI_ABORT_Q_POLL_MS 2000
- static void
- csio_abrt_cls(struct csio_ioreq *ioreq, struct scsi_cmnd *scmnd)
- {
- struct csio_lnode *ln = ioreq->lnode;
- struct csio_hw *hw = ln->hwp;
- int ready = 0;
- struct csio_scsim *scsim = csio_hw_to_scsim(hw);
- int rv;
- if (csio_scsi_cmnd(ioreq) != scmnd) {
- CSIO_INC_STATS(scsim, n_abrt_race_comp);
- return;
- }
- ready = csio_is_lnode_ready(ln);
- rv = csio_do_abrt_cls(hw, ioreq, (ready ? SCSI_ABORT : SCSI_CLOSE));
- if (rv != 0) {
- if (ready)
- CSIO_INC_STATS(scsim, n_abrt_busy_error);
- else
- CSIO_INC_STATS(scsim, n_cls_busy_error);
- }
- }
- /*
- * csio_scsi_abort_io_q - Abort all I/Os on given queue
- * @scm: SCSI module.
- * @q: Queue to abort.
- * @tmo: Timeout in ms
- *
- * Attempt to abort all I/Os on given queue, and wait for a max
- * of tmo milliseconds for them to complete. Returns success
- * if all I/Os are aborted. Else returns -ETIMEDOUT.
- * Should be entered with lock held. Exits with lock held.
- * NOTE:
- * Lock has to be held across the loop that aborts I/Os, since dropping the lock
- * in between can cause the list to be corrupted. As a result, the caller
- * of this function has to ensure that the number of I/os to be aborted
- * is finite enough to not cause lock-held-for-too-long issues.
- */
- static int
- csio_scsi_abort_io_q(struct csio_scsim *scm, struct list_head *q, uint32_t tmo)
- {
- struct csio_hw *hw = scm->hw;
- struct list_head *tmp, *next;
- int count = DIV_ROUND_UP(tmo, CSIO_SCSI_ABORT_Q_POLL_MS);
- struct scsi_cmnd *scmnd;
- if (list_empty(q))
- return 0;
- csio_dbg(hw, "Aborting SCSI I/Os\n");
- /* Now abort/close I/Os in the queue passed */
- list_for_each_safe(tmp, next, q) {
- scmnd = csio_scsi_cmnd((struct csio_ioreq *)tmp);
- csio_abrt_cls((struct csio_ioreq *)tmp, scmnd);
- }
- /* Wait till all active I/Os are completed/aborted/closed */
- while (!list_empty(q) && count--) {
- spin_unlock_irq(&hw->lock);
- msleep(CSIO_SCSI_ABORT_Q_POLL_MS);
- spin_lock_irq(&hw->lock);
- }
- /* all aborts completed */
- if (list_empty(q))
- return 0;
- return -ETIMEDOUT;
- }
- /*
- * csio_scsim_cleanup_io - Cleanup all I/Os in SCSI module.
- * @scm: SCSI module.
- * @abort: abort required.
- * Called with lock held, should exit with lock held.
- * Can sleep when waiting for I/Os to complete.
- */
- int
- csio_scsim_cleanup_io(struct csio_scsim *scm, bool abort)
- {
- struct csio_hw *hw = scm->hw;
- int rv = 0;
- int count = DIV_ROUND_UP(60 * 1000, CSIO_SCSI_ABORT_Q_POLL_MS);
- /* No I/Os pending */
- if (list_empty(&scm->active_q))
- return 0;
- /* Wait until all active I/Os are completed */
- while (!list_empty(&scm->active_q) && count--) {
- spin_unlock_irq(&hw->lock);
- msleep(CSIO_SCSI_ABORT_Q_POLL_MS);
- spin_lock_irq(&hw->lock);
- }
- /* all I/Os completed */
- if (list_empty(&scm->active_q))
- return 0;
- /* Else abort */
- if (abort) {
- rv = csio_scsi_abort_io_q(scm, &scm->active_q, 30000);
- if (rv == 0)
- return rv;
- csio_dbg(hw, "Some I/O aborts timed out, cleaning up..\n");
- }
- csio_scsi_cleanup_io_q(scm, &scm->active_q);
- CSIO_DB_ASSERT(list_empty(&scm->active_q));
- return rv;
- }
- /*
- * csio_scsim_cleanup_io_lnode - Cleanup all I/Os of given lnode.
- * @scm: SCSI module.
- * @lnode: lnode
- *
- * Called with lock held, should exit with lock held.
- * Can sleep (with dropped lock) when waiting for I/Os to complete.
- */
- int
- csio_scsim_cleanup_io_lnode(struct csio_scsim *scm, struct csio_lnode *ln)
- {
- struct csio_hw *hw = scm->hw;
- struct csio_scsi_level_data sld;
- int rv;
- int count = DIV_ROUND_UP(60 * 1000, CSIO_SCSI_ABORT_Q_POLL_MS);
- csio_dbg(hw, "Gathering all SCSI I/Os on lnode %p\n", ln);
- sld.level = CSIO_LEV_LNODE;
- sld.lnode = ln;
- INIT_LIST_HEAD(&ln->cmpl_q);
- csio_scsi_gather_active_ios(scm, &sld, &ln->cmpl_q);
- /* No I/Os pending on this lnode */
- if (list_empty(&ln->cmpl_q))
- return 0;
- /* Wait until all active I/Os on this lnode are completed */
- while (!list_empty(&ln->cmpl_q) && count--) {
- spin_unlock_irq(&hw->lock);
- msleep(CSIO_SCSI_ABORT_Q_POLL_MS);
- spin_lock_irq(&hw->lock);
- }
- /* all I/Os completed */
- if (list_empty(&ln->cmpl_q))
- return 0;
- csio_dbg(hw, "Some I/Os pending on ln:%p, aborting them..\n", ln);
- /* I/Os are pending, abort them */
- rv = csio_scsi_abort_io_q(scm, &ln->cmpl_q, 30000);
- if (rv != 0) {
- csio_dbg(hw, "Some I/O aborts timed out, cleaning up..\n");
- csio_scsi_cleanup_io_q(scm, &ln->cmpl_q);
- }
- CSIO_DB_ASSERT(list_empty(&ln->cmpl_q));
- return rv;
- }
- static ssize_t
- csio_show_hw_state(struct device *dev,
- struct device_attribute *attr, char *buf)
- {
- struct csio_lnode *ln = shost_priv(class_to_shost(dev));
- struct csio_hw *hw = csio_lnode_to_hw(ln);
- if (csio_is_hw_ready(hw))
- return snprintf(buf, PAGE_SIZE, "ready\n");
- else
- return snprintf(buf, PAGE_SIZE, "not ready\n");
- }
- /* Device reset */
- static ssize_t
- csio_device_reset(struct device *dev,
- struct device_attribute *attr, const char *buf, size_t count)
- {
- struct csio_lnode *ln = shost_priv(class_to_shost(dev));
- struct csio_hw *hw = csio_lnode_to_hw(ln);
- if (*buf != '1')
- return -EINVAL;
- /* Delete NPIV lnodes */
- csio_lnodes_exit(hw, 1);
- /* Block upper IOs */
- csio_lnodes_block_request(hw);
- spin_lock_irq(&hw->lock);
- csio_hw_reset(hw);
- spin_unlock_irq(&hw->lock);
- /* Unblock upper IOs */
- csio_lnodes_unblock_request(hw);
- return count;
- }
- /* disable port */
- static ssize_t
- csio_disable_port(struct device *dev,
- struct device_attribute *attr, const char *buf, size_t count)
- {
- struct csio_lnode *ln = shost_priv(class_to_shost(dev));
- struct csio_hw *hw = csio_lnode_to_hw(ln);
- bool disable;
- if (*buf == '1' || *buf == '0')
- disable = (*buf == '1') ? true : false;
- else
- return -EINVAL;
- /* Block upper IOs */
- csio_lnodes_block_by_port(hw, ln->portid);
- spin_lock_irq(&hw->lock);
- csio_disable_lnodes(hw, ln->portid, disable);
- spin_unlock_irq(&hw->lock);
- /* Unblock upper IOs */
- csio_lnodes_unblock_by_port(hw, ln->portid);
- return count;
- }
- /* Show debug level */
- static ssize_t
- csio_show_dbg_level(struct device *dev,
- struct device_attribute *attr, char *buf)
- {
- struct csio_lnode *ln = shost_priv(class_to_shost(dev));
- return snprintf(buf, PAGE_SIZE, "%x\n", ln->params.log_level);
- }
- /* Store debug level */
- static ssize_t
- csio_store_dbg_level(struct device *dev,
- struct device_attribute *attr, const char *buf, size_t count)
- {
- struct csio_lnode *ln = shost_priv(class_to_shost(dev));
- struct csio_hw *hw = csio_lnode_to_hw(ln);
- uint32_t dbg_level = 0;
- if (!isdigit(buf[0]))
- return -EINVAL;
- if (sscanf(buf, "%i", &dbg_level))
- return -EINVAL;
- ln->params.log_level = dbg_level;
- hw->params.log_level = dbg_level;
- return 0;
- }
- static DEVICE_ATTR(hw_state, S_IRUGO, csio_show_hw_state, NULL);
- static DEVICE_ATTR(device_reset, S_IWUSR, NULL, csio_device_reset);
- static DEVICE_ATTR(disable_port, S_IWUSR, NULL, csio_disable_port);
- static DEVICE_ATTR(dbg_level, S_IRUGO | S_IWUSR, csio_show_dbg_level,
- csio_store_dbg_level);
- static struct device_attribute *csio_fcoe_lport_attrs[] = {
- &dev_attr_hw_state,
- &dev_attr_device_reset,
- &dev_attr_disable_port,
- &dev_attr_dbg_level,
- NULL,
- };
- static ssize_t
- csio_show_num_reg_rnodes(struct device *dev,
- struct device_attribute *attr, char *buf)
- {
- struct csio_lnode *ln = shost_priv(class_to_shost(dev));
- return snprintf(buf, PAGE_SIZE, "%d\n", ln->num_reg_rnodes);
- }
- static DEVICE_ATTR(num_reg_rnodes, S_IRUGO, csio_show_num_reg_rnodes, NULL);
- static struct device_attribute *csio_fcoe_vport_attrs[] = {
- &dev_attr_num_reg_rnodes,
- &dev_attr_dbg_level,
- NULL,
- };
- static inline uint32_t
- csio_scsi_copy_to_sgl(struct csio_hw *hw, struct csio_ioreq *req)
- {
- struct scsi_cmnd *scmnd = (struct scsi_cmnd *)csio_scsi_cmnd(req);
- struct scatterlist *sg;
- uint32_t bytes_left;
- uint32_t bytes_copy;
- uint32_t buf_off = 0;
- uint32_t start_off = 0;
- uint32_t sg_off = 0;
- void *sg_addr;
- void *buf_addr;
- struct csio_dma_buf *dma_buf;
- bytes_left = scsi_bufflen(scmnd);
- sg = scsi_sglist(scmnd);
- dma_buf = (struct csio_dma_buf *)csio_list_next(&req->gen_list);
- /* Copy data from driver buffer to SGs of SCSI CMD */
- while (bytes_left > 0 && sg && dma_buf) {
- if (buf_off >= dma_buf->len) {
- buf_off = 0;
- dma_buf = (struct csio_dma_buf *)
- csio_list_next(dma_buf);
- continue;
- }
- if (start_off >= sg->length) {
- start_off -= sg->length;
- sg = sg_next(sg);
- continue;
- }
- buf_addr = dma_buf->vaddr + buf_off;
- sg_off = sg->offset + start_off;
- bytes_copy = min((dma_buf->len - buf_off),
- sg->length - start_off);
- bytes_copy = min((uint32_t)(PAGE_SIZE - (sg_off & ~PAGE_MASK)),
- bytes_copy);
- sg_addr = kmap_atomic(sg_page(sg) + (sg_off >> PAGE_SHIFT));
- if (!sg_addr) {
- csio_err(hw, "failed to kmap sg:%p of ioreq:%p\n",
- sg, req);
- break;
- }
- csio_dbg(hw, "copy_to_sgl:sg_addr %p sg_off %d buf %p len %d\n",
- sg_addr, sg_off, buf_addr, bytes_copy);
- memcpy(sg_addr + (sg_off & ~PAGE_MASK), buf_addr, bytes_copy);
- kunmap_atomic(sg_addr);
- start_off += bytes_copy;
- buf_off += bytes_copy;
- bytes_left -= bytes_copy;
- }
- if (bytes_left > 0)
- return DID_ERROR;
- else
- return DID_OK;
- }
- /*
- * csio_scsi_err_handler - SCSI error handler.
- * @hw: HW module.
- * @req: IO request.
- *
- */
- static inline void
- csio_scsi_err_handler(struct csio_hw *hw, struct csio_ioreq *req)
- {
- struct scsi_cmnd *cmnd = (struct scsi_cmnd *)csio_scsi_cmnd(req);
- struct csio_scsim *scm = csio_hw_to_scsim(hw);
- struct fcp_resp_with_ext *fcp_resp;
- struct fcp_resp_rsp_info *rsp_info;
- struct csio_dma_buf *dma_buf;
- uint8_t flags, scsi_status = 0;
- uint32_t host_status = DID_OK;
- uint32_t rsp_len = 0, sns_len = 0;
- struct csio_rnode *rn = (struct csio_rnode *)(cmnd->device->hostdata);
- switch (req->wr_status) {
- case FW_HOSTERROR:
- if (unlikely(!csio_is_hw_ready(hw)))
- return;
- host_status = DID_ERROR;
- CSIO_INC_STATS(scm, n_hosterror);
- break;
- case FW_SCSI_RSP_ERR:
- dma_buf = &req->dma_buf;
- fcp_resp = (struct fcp_resp_with_ext *)dma_buf->vaddr;
- rsp_info = (struct fcp_resp_rsp_info *)(fcp_resp + 1);
- flags = fcp_resp->resp.fr_flags;
- scsi_status = fcp_resp->resp.fr_status;
- if (flags & FCP_RSP_LEN_VAL) {
- rsp_len = be32_to_cpu(fcp_resp->ext.fr_rsp_len);
- if ((rsp_len != 0 && rsp_len != 4 && rsp_len != 8) ||
- (rsp_info->rsp_code != FCP_TMF_CMPL)) {
- host_status = DID_ERROR;
- goto out;
- }
- }
- if ((flags & FCP_SNS_LEN_VAL) && fcp_resp->ext.fr_sns_len) {
- sns_len = be32_to_cpu(fcp_resp->ext.fr_sns_len);
- if (sns_len > SCSI_SENSE_BUFFERSIZE)
- sns_len = SCSI_SENSE_BUFFERSIZE;
- memcpy(cmnd->sense_buffer,
- &rsp_info->_fr_resvd[0] + rsp_len, sns_len);
- CSIO_INC_STATS(scm, n_autosense);
- }
- scsi_set_resid(cmnd, 0);
- /* Under run */
- if (flags & FCP_RESID_UNDER) {
- scsi_set_resid(cmnd,
- be32_to_cpu(fcp_resp->ext.fr_resid));
- if (!(flags & FCP_SNS_LEN_VAL) &&
- (scsi_status == SAM_STAT_GOOD) &&
- ((scsi_bufflen(cmnd) - scsi_get_resid(cmnd))
- < cmnd->underflow))
- host_status = DID_ERROR;
- } else if (flags & FCP_RESID_OVER)
- host_status = DID_ERROR;
- CSIO_INC_STATS(scm, n_rsperror);
- break;
- case FW_SCSI_OVER_FLOW_ERR:
- csio_warn(hw,
- "Over-flow error,cmnd:0x%x expected len:0x%x"
- " resid:0x%x\n", cmnd->cmnd[0],
- scsi_bufflen(cmnd), scsi_get_resid(cmnd));
- host_status = DID_ERROR;
- CSIO_INC_STATS(scm, n_ovflerror);
- break;
- case FW_SCSI_UNDER_FLOW_ERR:
- csio_warn(hw,
- "Under-flow error,cmnd:0x%x expected"
- " len:0x%x resid:0x%x lun:0x%llx ssn:0x%x\n",
- cmnd->cmnd[0], scsi_bufflen(cmnd),
- scsi_get_resid(cmnd), cmnd->device->lun,
- rn->flowid);
- host_status = DID_ERROR;
- CSIO_INC_STATS(scm, n_unflerror);
- break;
- case FW_SCSI_ABORT_REQUESTED:
- case FW_SCSI_ABORTED:
- case FW_SCSI_CLOSE_REQUESTED:
- csio_dbg(hw, "Req %p cmd:%p op:%x %s\n", req, cmnd,
- cmnd->cmnd[0],
- (req->wr_status == FW_SCSI_CLOSE_REQUESTED) ?
- "closed" : "aborted");
- /*
- * csio_eh_abort_handler checks this value to
- * succeed or fail the abort request.
- */
- host_status = DID_REQUEUE;
- if (req->wr_status == FW_SCSI_CLOSE_REQUESTED)
- CSIO_INC_STATS(scm, n_closed);
- else
- CSIO_INC_STATS(scm, n_aborted);
- break;
- case FW_SCSI_ABORT_TIMEDOUT:
- /* FW timed out the abort itself */
- csio_dbg(hw, "FW timed out abort req:%p cmnd:%p status:%x\n",
- req, cmnd, req->wr_status);
- host_status = DID_ERROR;
- CSIO_INC_STATS(scm, n_abrt_timedout);
- break;
- case FW_RDEV_NOT_READY:
- /*
- * In firmware, a RDEV can get into this state
- * temporarily, before moving into dissapeared/lost
- * state. So, the driver should complete the request equivalent
- * to device-disappeared!
- */
- CSIO_INC_STATS(scm, n_rdev_nr_error);
- host_status = DID_ERROR;
- break;
- case FW_ERR_RDEV_LOST:
- CSIO_INC_STATS(scm, n_rdev_lost_error);
- host_status = DID_ERROR;
- break;
- case FW_ERR_RDEV_LOGO:
- CSIO_INC_STATS(scm, n_rdev_logo_error);
- host_status = DID_ERROR;
- break;
- case FW_ERR_RDEV_IMPL_LOGO:
- host_status = DID_ERROR;
- break;
- case FW_ERR_LINK_DOWN:
- CSIO_INC_STATS(scm, n_link_down_error);
- host_status = DID_ERROR;
- break;
- case FW_FCOE_NO_XCHG:
- CSIO_INC_STATS(scm, n_no_xchg_error);
- host_status = DID_ERROR;
- break;
- default:
- csio_err(hw, "Unknown SCSI FW WR status:%d req:%p cmnd:%p\n",
- req->wr_status, req, cmnd);
- CSIO_DB_ASSERT(0);
- CSIO_INC_STATS(scm, n_unknown_error);
- host_status = DID_ERROR;
- break;
- }
- out:
- if (req->nsge > 0) {
- scsi_dma_unmap(cmnd);
- if (req->dcopy && (host_status == DID_OK))
- host_status = csio_scsi_copy_to_sgl(hw, req);
- }
- cmnd->result = (((host_status) << 16) | scsi_status);
- cmnd->scsi_done(cmnd);
- /* Wake up waiting threads */
- csio_scsi_cmnd(req) = NULL;
- complete(&req->cmplobj);
- }
- /*
- * csio_scsi_cbfn - SCSI callback function.
- * @hw: HW module.
- * @req: IO request.
- *
- */
- static void
- csio_scsi_cbfn(struct csio_hw *hw, struct csio_ioreq *req)
- {
- struct scsi_cmnd *cmnd = (struct scsi_cmnd *)csio_scsi_cmnd(req);
- uint8_t scsi_status = SAM_STAT_GOOD;
- uint32_t host_status = DID_OK;
- if (likely(req->wr_status == FW_SUCCESS)) {
- if (req->nsge > 0) {
- scsi_dma_unmap(cmnd);
- if (req->dcopy)
- host_status = csio_scsi_copy_to_sgl(hw, req);
- }
- cmnd->result = (((host_status) << 16) | scsi_status);
- cmnd->scsi_done(cmnd);
- csio_scsi_cmnd(req) = NULL;
- CSIO_INC_STATS(csio_hw_to_scsim(hw), n_tot_success);
- } else {
- /* Error handling */
- csio_scsi_err_handler(hw, req);
- }
- }
- /**
- * csio_queuecommand - Entry point to kickstart an I/O request.
- * @host: The scsi_host pointer.
- * @cmnd: The I/O request from ML.
- *
- * This routine does the following:
- * - Checks for HW and Rnode module readiness.
- * - Gets a free ioreq structure (which is already initialized
- * to uninit during its allocation).
- * - Maps SG elements.
- * - Initializes ioreq members.
- * - Kicks off the SCSI state machine for this IO.
- * - Returns busy status on error.
- */
- static int
- csio_queuecommand(struct Scsi_Host *host, struct scsi_cmnd *cmnd)
- {
- struct csio_lnode *ln = shost_priv(host);
- struct csio_hw *hw = csio_lnode_to_hw(ln);
- struct csio_scsim *scsim = csio_hw_to_scsim(hw);
- struct csio_rnode *rn = (struct csio_rnode *)(cmnd->device->hostdata);
- struct csio_ioreq *ioreq = NULL;
- unsigned long flags;
- int nsge = 0;
- int rv = SCSI_MLQUEUE_HOST_BUSY, nr;
- int retval;
- int cpu;
- struct csio_scsi_qset *sqset;
- struct fc_rport *rport = starget_to_rport(scsi_target(cmnd->device));
- if (!blk_rq_cpu_valid(cmnd->request))
- cpu = smp_processor_id();
- else
- cpu = cmnd->request->cpu;
- sqset = &hw->sqset[ln->portid][cpu];
- nr = fc_remote_port_chkready(rport);
- if (nr) {
- cmnd->result = nr;
- CSIO_INC_STATS(scsim, n_rn_nr_error);
- goto err_done;
- }
- if (unlikely(!csio_is_hw_ready(hw))) {
- cmnd->result = (DID_REQUEUE << 16);
- CSIO_INC_STATS(scsim, n_hw_nr_error);
- goto err_done;
- }
- /* Get req->nsge, if there are SG elements to be mapped */
- nsge = scsi_dma_map(cmnd);
- if (unlikely(nsge < 0)) {
- CSIO_INC_STATS(scsim, n_dmamap_error);
- goto err;
- }
- /* Do we support so many mappings? */
- if (unlikely(nsge > scsim->max_sge)) {
- csio_warn(hw,
- "More SGEs than can be supported."
- " SGEs: %d, Max SGEs: %d\n", nsge, scsim->max_sge);
- CSIO_INC_STATS(scsim, n_unsupp_sge_error);
- goto err_dma_unmap;
- }
- /* Get a free ioreq structure - SM is already set to uninit */
- ioreq = csio_get_scsi_ioreq_lock(hw, scsim);
- if (!ioreq) {
- csio_err(hw, "Out of I/O request elements. Active #:%d\n",
- scsim->stats.n_active);
- CSIO_INC_STATS(scsim, n_no_req_error);
- goto err_dma_unmap;
- }
- ioreq->nsge = nsge;
- ioreq->lnode = ln;
- ioreq->rnode = rn;
- ioreq->iq_idx = sqset->iq_idx;
- ioreq->eq_idx = sqset->eq_idx;
- ioreq->wr_status = 0;
- ioreq->drv_status = 0;
- csio_scsi_cmnd(ioreq) = (void *)cmnd;
- ioreq->tmo = 0;
- ioreq->datadir = cmnd->sc_data_direction;
- if (cmnd->sc_data_direction == DMA_TO_DEVICE) {
- CSIO_INC_STATS(ln, n_output_requests);
- ln->stats.n_output_bytes += scsi_bufflen(cmnd);
- } else if (cmnd->sc_data_direction == DMA_FROM_DEVICE) {
- CSIO_INC_STATS(ln, n_input_requests);
- ln->stats.n_input_bytes += scsi_bufflen(cmnd);
- } else
- CSIO_INC_STATS(ln, n_control_requests);
- /* Set cbfn */
- ioreq->io_cbfn = csio_scsi_cbfn;
- /* Needed during abort */
- cmnd->host_scribble = (unsigned char *)ioreq;
- cmnd->SCp.Message = 0;
- /* Kick off SCSI IO SM on the ioreq */
- spin_lock_irqsave(&hw->lock, flags);
- retval = csio_scsi_start_io(ioreq);
- spin_unlock_irqrestore(&hw->lock, flags);
- if (retval != 0) {
- csio_err(hw, "ioreq: %p couldnt be started, status:%d\n",
- ioreq, retval);
- CSIO_INC_STATS(scsim, n_busy_error);
- goto err_put_req;
- }
- return 0;
- err_put_req:
- csio_put_scsi_ioreq_lock(hw, scsim, ioreq);
- err_dma_unmap:
- if (nsge > 0)
- scsi_dma_unmap(cmnd);
- err:
- return rv;
- err_done:
- cmnd->scsi_done(cmnd);
- return 0;
- }
- static int
- csio_do_abrt_cls(struct csio_hw *hw, struct csio_ioreq *ioreq, bool abort)
- {
- int rv;
- int cpu = smp_processor_id();
- struct csio_lnode *ln = ioreq->lnode;
- struct csio_scsi_qset *sqset = &hw->sqset[ln->portid][cpu];
- ioreq->tmo = CSIO_SCSI_ABRT_TMO_MS;
- /*
- * Use current processor queue for posting the abort/close, but retain
- * the ingress queue ID of the original I/O being aborted/closed - we
- * need the abort/close completion to be received on the same queue
- * as the original I/O.
- */
- ioreq->eq_idx = sqset->eq_idx;
- if (abort == SCSI_ABORT)
- rv = csio_scsi_abort(ioreq);
- else
- rv = csio_scsi_close(ioreq);
- return rv;
- }
- static int
- csio_eh_abort_handler(struct scsi_cmnd *cmnd)
- {
- struct csio_ioreq *ioreq;
- struct csio_lnode *ln = shost_priv(cmnd->device->host);
- struct csio_hw *hw = csio_lnode_to_hw(ln);
- struct csio_scsim *scsim = csio_hw_to_scsim(hw);
- int ready = 0, ret;
- unsigned long tmo = 0;
- int rv;
- struct csio_rnode *rn = (struct csio_rnode *)(cmnd->device->hostdata);
- ret = fc_block_scsi_eh(cmnd);
- if (ret)
- return ret;
- ioreq = (struct csio_ioreq *)cmnd->host_scribble;
- if (!ioreq)
- return SUCCESS;
- if (!rn)
- return FAILED;
- csio_dbg(hw,
- "Request to abort ioreq:%p cmd:%p cdb:%08llx"
- " ssni:0x%x lun:%llu iq:0x%x\n",
- ioreq, cmnd, *((uint64_t *)cmnd->cmnd), rn->flowid,
- cmnd->device->lun, csio_q_physiqid(hw, ioreq->iq_idx));
- if (((struct scsi_cmnd *)csio_scsi_cmnd(ioreq)) != cmnd) {
- CSIO_INC_STATS(scsim, n_abrt_race_comp);
- return SUCCESS;
- }
- ready = csio_is_lnode_ready(ln);
- tmo = CSIO_SCSI_ABRT_TMO_MS;
- reinit_completion(&ioreq->cmplobj);
- spin_lock_irq(&hw->lock);
- rv = csio_do_abrt_cls(hw, ioreq, (ready ? SCSI_ABORT : SCSI_CLOSE));
- spin_unlock_irq(&hw->lock);
- if (rv != 0) {
- if (rv == -EINVAL) {
- /* Return success, if abort/close request issued on
- * already completed IO
- */
- return SUCCESS;
- }
- if (ready)
- CSIO_INC_STATS(scsim, n_abrt_busy_error);
- else
- CSIO_INC_STATS(scsim, n_cls_busy_error);
- goto inval_scmnd;
- }
- wait_for_completion_timeout(&ioreq->cmplobj, msecs_to_jiffies(tmo));
- /* FW didnt respond to abort within our timeout */
- if (((struct scsi_cmnd *)csio_scsi_cmnd(ioreq)) == cmnd) {
- csio_err(hw, "Abort timed out -- req: %p\n", ioreq);
- CSIO_INC_STATS(scsim, n_abrt_timedout);
- inval_scmnd:
- if (ioreq->nsge > 0)
- scsi_dma_unmap(cmnd);
- spin_lock_irq(&hw->lock);
- csio_scsi_cmnd(ioreq) = NULL;
- spin_unlock_irq(&hw->lock);
- cmnd->result = (DID_ERROR << 16);
- cmnd->scsi_done(cmnd);
- return FAILED;
- }
- /* FW successfully aborted the request */
- if (host_byte(cmnd->result) == DID_REQUEUE) {
- csio_info(hw,
- "Aborted SCSI command to (%d:%llu) serial#:0x%lx\n",
- cmnd->device->id, cmnd->device->lun,
- cmnd->serial_number);
- return SUCCESS;
- } else {
- csio_info(hw,
- "Failed to abort SCSI command, (%d:%llu) serial#:0x%lx\n",
- cmnd->device->id, cmnd->device->lun,
- cmnd->serial_number);
- return FAILED;
- }
- }
- /*
- * csio_tm_cbfn - TM callback function.
- * @hw: HW module.
- * @req: IO request.
- *
- * Cache the result in 'cmnd', since ioreq will be freed soon
- * after we return from here, and the waiting thread shouldnt trust
- * the ioreq contents.
- */
- static void
- csio_tm_cbfn(struct csio_hw *hw, struct csio_ioreq *req)
- {
- struct scsi_cmnd *cmnd = (struct scsi_cmnd *)csio_scsi_cmnd(req);
- struct csio_dma_buf *dma_buf;
- uint8_t flags = 0;
- struct fcp_resp_with_ext *fcp_resp;
- struct fcp_resp_rsp_info *rsp_info;
- csio_dbg(hw, "req: %p in csio_tm_cbfn status: %d\n",
- req, req->wr_status);
- /* Cache FW return status */
- cmnd->SCp.Status = req->wr_status;
- /* Special handling based on FCP response */
- /*
- * FW returns us this error, if flags were set. FCP4 says
- * FCP_RSP_LEN_VAL in flags shall be set for TM completions.
- * So if a target were to set this bit, we expect that the
- * rsp_code is set to FCP_TMF_CMPL for a successful TM
- * completion. Any other rsp_code means TM operation failed.
- * If a target were to just ignore setting flags, we treat
- * the TM operation as success, and FW returns FW_SUCCESS.
- */
- if (req->wr_status == FW_SCSI_RSP_ERR) {
- dma_buf = &req->dma_buf;
- fcp_resp = (struct fcp_resp_with_ext *)dma_buf->vaddr;
- rsp_info = (struct fcp_resp_rsp_info *)(fcp_resp + 1);
- flags = fcp_resp->resp.fr_flags;
- /* Modify return status if flags indicate success */
- if (flags & FCP_RSP_LEN_VAL)
- if (rsp_info->rsp_code == FCP_TMF_CMPL)
- cmnd->SCp.Status = FW_SUCCESS;
- csio_dbg(hw, "TM FCP rsp code: %d\n", rsp_info->rsp_code);
- }
- /* Wake up the TM handler thread */
- csio_scsi_cmnd(req) = NULL;
- }
- static int
- csio_eh_lun_reset_handler(struct scsi_cmnd *cmnd)
- {
- struct csio_lnode *ln = shost_priv(cmnd->device->host);
- struct csio_hw *hw = csio_lnode_to_hw(ln);
- struct csio_scsim *scsim = csio_hw_to_scsim(hw);
- struct csio_rnode *rn = (struct csio_rnode *)(cmnd->device->hostdata);
- struct csio_ioreq *ioreq = NULL;
- struct csio_scsi_qset *sqset;
- unsigned long flags;
- int retval;
- int count, ret;
- LIST_HEAD(local_q);
- struct csio_scsi_level_data sld;
- if (!rn)
- goto fail;
- csio_dbg(hw, "Request to reset LUN:%llu (ssni:0x%x tgtid:%d)\n",
- cmnd->device->lun, rn->flowid, rn->scsi_id);
- if (!csio_is_lnode_ready(ln)) {
- csio_err(hw,
- "LUN reset cannot be issued on non-ready"
- " local node vnpi:0x%x (LUN:%llu)\n",
- ln->vnp_flowid, cmnd->device->lun);
- goto fail;
- }
- /* Lnode is ready, now wait on rport node readiness */
- ret = fc_block_scsi_eh(cmnd);
- if (ret)
- return ret;
- /*
- * If we have blocked in the previous call, at this point, either the
- * remote node has come back online, or device loss timer has fired
- * and the remote node is destroyed. Allow the LUN reset only for
- * the former case, since LUN reset is a TMF I/O on the wire, and we
- * need a valid session to issue it.
- */
- if (fc_remote_port_chkready(rn->rport)) {
- csio_err(hw,
- "LUN reset cannot be issued on non-ready"
- " remote node ssni:0x%x (LUN:%llu)\n",
- rn->flowid, cmnd->device->lun);
- goto fail;
- }
- /* Get a free ioreq structure - SM is already set to uninit */
- ioreq = csio_get_scsi_ioreq_lock(hw, scsim);
- if (!ioreq) {
- csio_err(hw, "Out of IO request elements. Active # :%d\n",
- scsim->stats.n_active);
- goto fail;
- }
- sqset = &hw->sqset[ln->portid][smp_processor_id()];
- ioreq->nsge = 0;
- ioreq->lnode = ln;
- ioreq->rnode = rn;
- ioreq->iq_idx = sqset->iq_idx;
- ioreq->eq_idx = sqset->eq_idx;
- csio_scsi_cmnd(ioreq) = cmnd;
- cmnd->host_scribble = (unsigned char *)ioreq;
- cmnd->SCp.Status = 0;
- cmnd->SCp.Message = FCP_TMF_LUN_RESET;
- ioreq->tmo = CSIO_SCSI_LUNRST_TMO_MS / 1000;
- /*
- * FW times the LUN reset for ioreq->tmo, so we got to wait a little
- * longer (10s for now) than that to allow FW to return the timed
- * out command.
- */
- count = DIV_ROUND_UP((ioreq->tmo + 10) * 1000, CSIO_SCSI_TM_POLL_MS);
- /* Set cbfn */
- ioreq->io_cbfn = csio_tm_cbfn;
- /* Save of the ioreq info for later use */
- sld.level = CSIO_LEV_LUN;
- sld.lnode = ioreq->lnode;
- sld.rnode = ioreq->rnode;
- sld.oslun = cmnd->device->lun;
- spin_lock_irqsave(&hw->lock, flags);
- /* Kick off TM SM on the ioreq */
- retval = csio_scsi_start_tm(ioreq);
- spin_unlock_irqrestore(&hw->lock, flags);
- if (retval != 0) {
- csio_err(hw, "Failed to issue LUN reset, req:%p, status:%d\n",
- ioreq, retval);
- goto fail_ret_ioreq;
- }
- csio_dbg(hw, "Waiting max %d secs for LUN reset completion\n",
- count * (CSIO_SCSI_TM_POLL_MS / 1000));
- /* Wait for completion */
- while ((((struct scsi_cmnd *)csio_scsi_cmnd(ioreq)) == cmnd)
- && count--)
- msleep(CSIO_SCSI_TM_POLL_MS);
- /* LUN reset timed-out */
- if (((struct scsi_cmnd *)csio_scsi_cmnd(ioreq)) == cmnd) {
- csio_err(hw, "LUN reset (%d:%llu) timed out\n",
- cmnd->device->id, cmnd->device->lun);
- spin_lock_irq(&hw->lock);
- csio_scsi_drvcleanup(ioreq);
- list_del_init(&ioreq->sm.sm_list);
- spin_unlock_irq(&hw->lock);
- goto fail_ret_ioreq;
- }
- /* LUN reset returned, check cached status */
- if (cmnd->SCp.Status != FW_SUCCESS) {
- csio_err(hw, "LUN reset failed (%d:%llu), status: %d\n",
- cmnd->device->id, cmnd->device->lun, cmnd->SCp.Status);
- goto fail;
- }
- /* LUN reset succeeded, Start aborting affected I/Os */
- /*
- * Since the host guarantees during LUN reset that there
- * will not be any more I/Os to that LUN, until the LUN reset
- * completes, we gather pending I/Os after the LUN reset.
- */
- spin_lock_irq(&hw->lock);
- csio_scsi_gather_active_ios(scsim, &sld, &local_q);
- retval = csio_scsi_abort_io_q(scsim, &local_q, 30000);
- spin_unlock_irq(&hw->lock);
- /* Aborts may have timed out */
- if (retval != 0) {
- csio_err(hw,
- "Attempt to abort I/Os during LUN reset of %llu"
- " returned %d\n", cmnd->device->lun, retval);
- /* Return I/Os back to active_q */
- spin_lock_irq(&hw->lock);
- list_splice_tail_init(&local_q, &scsim->active_q);
- spin_unlock_irq(&hw->lock);
- goto fail;
- }
- CSIO_INC_STATS(rn, n_lun_rst);
- csio_info(hw, "LUN reset occurred (%d:%llu)\n",
- cmnd->device->id, cmnd->device->lun);
- return SUCCESS;
- fail_ret_ioreq:
- csio_put_scsi_ioreq_lock(hw, scsim, ioreq);
- fail:
- CSIO_INC_STATS(rn, n_lun_rst_fail);
- return FAILED;
- }
- static int
- csio_slave_alloc(struct scsi_device *sdev)
- {
- struct fc_rport *rport = starget_to_rport(scsi_target(sdev));
- if (!rport || fc_remote_port_chkready(rport))
- return -ENXIO;
- sdev->hostdata = *((struct csio_lnode **)(rport->dd_data));
- return 0;
- }
- static int
- csio_slave_configure(struct scsi_device *sdev)
- {
- scsi_change_queue_depth(sdev, csio_lun_qdepth);
- return 0;
- }
- static void
- csio_slave_destroy(struct scsi_device *sdev)
- {
- sdev->hostdata = NULL;
- }
- static int
- csio_scan_finished(struct Scsi_Host *shost, unsigned long time)
- {
- struct csio_lnode *ln = shost_priv(shost);
- int rv = 1;
- spin_lock_irq(shost->host_lock);
- if (!ln->hwp || csio_list_deleted(&ln->sm.sm_list))
- goto out;
- rv = csio_scan_done(ln, jiffies, time, csio_max_scan_tmo * HZ,
- csio_delta_scan_tmo * HZ);
- out:
- spin_unlock_irq(shost->host_lock);
- return rv;
- }
- struct scsi_host_template csio_fcoe_shost_template = {
- .module = THIS_MODULE,
- .name = CSIO_DRV_DESC,
- .proc_name = KBUILD_MODNAME,
- .queuecommand = csio_queuecommand,
- .eh_timed_out = fc_eh_timed_out,
- .eh_abort_handler = csio_eh_abort_handler,
- .eh_device_reset_handler = csio_eh_lun_reset_handler,
- .slave_alloc = csio_slave_alloc,
- .slave_configure = csio_slave_configure,
- .slave_destroy = csio_slave_destroy,
- .scan_finished = csio_scan_finished,
- .this_id = -1,
- .sg_tablesize = CSIO_SCSI_MAX_SGE,
- .cmd_per_lun = CSIO_MAX_CMD_PER_LUN,
- .use_clustering = ENABLE_CLUSTERING,
- .shost_attrs = csio_fcoe_lport_attrs,
- .max_sectors = CSIO_MAX_SECTOR_SIZE,
- };
- struct scsi_host_template csio_fcoe_shost_vport_template = {
- .module = THIS_MODULE,
- .name = CSIO_DRV_DESC,
- .proc_name = KBUILD_MODNAME,
- .queuecommand = csio_queuecommand,
- .eh_timed_out = fc_eh_timed_out,
- .eh_abort_handler = csio_eh_abort_handler,
- .eh_device_reset_handler = csio_eh_lun_reset_handler,
- .slave_alloc = csio_slave_alloc,
- .slave_configure = csio_slave_configure,
- .slave_destroy = csio_slave_destroy,
- .scan_finished = csio_scan_finished,
- .this_id = -1,
- .sg_tablesize = CSIO_SCSI_MAX_SGE,
- .cmd_per_lun = CSIO_MAX_CMD_PER_LUN,
- .use_clustering = ENABLE_CLUSTERING,
- .shost_attrs = csio_fcoe_vport_attrs,
- .max_sectors = CSIO_MAX_SECTOR_SIZE,
- };
- /*
- * csio_scsi_alloc_ddp_bufs - Allocate buffers for DDP of unaligned SGLs.
- * @scm: SCSI Module
- * @hw: HW device.
- * @buf_size: buffer size
- * @num_buf : Number of buffers.
- *
- * This routine allocates DMA buffers required for SCSI Data xfer, if
- * each SGL buffer for a SCSI Read request posted by SCSI midlayer are
- * not virtually contiguous.
- */
- static int
- csio_scsi_alloc_ddp_bufs(struct csio_scsim *scm, struct csio_hw *hw,
- int buf_size, int num_buf)
- {
- int n = 0;
- struct list_head *tmp;
- struct csio_dma_buf *ddp_desc = NULL;
- uint32_t unit_size = 0;
- if (!num_buf)
- return 0;
- if (!buf_size)
- return -EINVAL;
- INIT_LIST_HEAD(&scm->ddp_freelist);
- /* Align buf size to page size */
- buf_size = (buf_size + PAGE_SIZE - 1) & PAGE_MASK;
- /* Initialize dma descriptors */
- for (n = 0; n < num_buf; n++) {
- /* Set unit size to request size */
- unit_size = buf_size;
- ddp_desc = kzalloc(sizeof(struct csio_dma_buf), GFP_KERNEL);
- if (!ddp_desc) {
- csio_err(hw,
- "Failed to allocate ddp descriptors,"
- " Num allocated = %d.\n",
- scm->stats.n_free_ddp);
- goto no_mem;
- }
- /* Allocate Dma buffers for DDP */
- ddp_desc->vaddr = pci_alloc_consistent(hw->pdev, unit_size,
- &ddp_desc->paddr);
- if (!ddp_desc->vaddr) {
- csio_err(hw,
- "SCSI response DMA buffer (ddp) allocation"
- " failed!\n");
- kfree(ddp_desc);
- goto no_mem;
- }
- ddp_desc->len = unit_size;
- /* Added it to scsi ddp freelist */
- list_add_tail(&ddp_desc->list, &scm->ddp_freelist);
- CSIO_INC_STATS(scm, n_free_ddp);
- }
- return 0;
- no_mem:
- /* release dma descs back to freelist and free dma memory */
- list_for_each(tmp, &scm->ddp_freelist) {
- ddp_desc = (struct csio_dma_buf *) tmp;
- tmp = csio_list_prev(tmp);
- pci_free_consistent(hw->pdev, ddp_desc->len, ddp_desc->vaddr,
- ddp_desc->paddr);
- list_del_init(&ddp_desc->list);
- kfree(ddp_desc);
- }
- scm->stats.n_free_ddp = 0;
- return -ENOMEM;
- }
- /*
- * csio_scsi_free_ddp_bufs - free DDP buffers of unaligned SGLs.
- * @scm: SCSI Module
- * @hw: HW device.
- *
- * This routine frees ddp buffers.
- */
- static void
- csio_scsi_free_ddp_bufs(struct csio_scsim *scm, struct csio_hw *hw)
- {
- struct list_head *tmp;
- struct csio_dma_buf *ddp_desc;
- /* release dma descs back to freelist and free dma memory */
- list_for_each(tmp, &scm->ddp_freelist) {
- ddp_desc = (struct csio_dma_buf *) tmp;
- tmp = csio_list_prev(tmp);
- pci_free_consistent(hw->pdev, ddp_desc->len, ddp_desc->vaddr,
- ddp_desc->paddr);
- list_del_init(&ddp_desc->list);
- kfree(ddp_desc);
- }
- scm->stats.n_free_ddp = 0;
- }
- /**
- * csio_scsim_init - Initialize SCSI Module
- * @scm: SCSI Module
- * @hw: HW module
- *
- */
- int
- csio_scsim_init(struct csio_scsim *scm, struct csio_hw *hw)
- {
- int i;
- struct csio_ioreq *ioreq;
- struct csio_dma_buf *dma_buf;
- INIT_LIST_HEAD(&scm->active_q);
- scm->hw = hw;
- scm->proto_cmd_len = sizeof(struct fcp_cmnd);
- scm->proto_rsp_len = CSIO_SCSI_RSP_LEN;
- scm->max_sge = CSIO_SCSI_MAX_SGE;
- spin_lock_init(&scm->freelist_lock);
- /* Pre-allocate ioreqs and initialize them */
- INIT_LIST_HEAD(&scm->ioreq_freelist);
- for (i = 0; i < csio_scsi_ioreqs; i++) {
- ioreq = kzalloc(sizeof(struct csio_ioreq), GFP_KERNEL);
- if (!ioreq) {
- csio_err(hw,
- "I/O request element allocation failed, "
- " Num allocated = %d.\n",
- scm->stats.n_free_ioreq);
- goto free_ioreq;
- }
- /* Allocate Dma buffers for Response Payload */
- dma_buf = &ioreq->dma_buf;
- dma_buf->vaddr = dma_pool_alloc(hw->scsi_dma_pool, GFP_KERNEL,
- &dma_buf->paddr);
- if (!dma_buf->vaddr) {
- csio_err(hw,
- "SCSI response DMA buffer allocation"
- " failed!\n");
- kfree(ioreq);
- goto free_ioreq;
- }
- dma_buf->len = scm->proto_rsp_len;
- /* Set state to uninit */
- csio_init_state(&ioreq->sm, csio_scsis_uninit);
- INIT_LIST_HEAD(&ioreq->gen_list);
- init_completion(&ioreq->cmplobj);
- list_add_tail(&ioreq->sm.sm_list, &scm->ioreq_freelist);
- CSIO_INC_STATS(scm, n_free_ioreq);
- }
- if (csio_scsi_alloc_ddp_bufs(scm, hw, PAGE_SIZE, csio_ddp_descs))
- goto free_ioreq;
- return 0;
- free_ioreq:
- /*
- * Free up existing allocations, since an error
- * from here means we are returning for good
- */
- while (!list_empty(&scm->ioreq_freelist)) {
- struct csio_sm *tmp;
- tmp = list_first_entry(&scm->ioreq_freelist,
- struct csio_sm, sm_list);
- list_del_init(&tmp->sm_list);
- ioreq = (struct csio_ioreq *)tmp;
- dma_buf = &ioreq->dma_buf;
- dma_pool_free(hw->scsi_dma_pool, dma_buf->vaddr,
- dma_buf->paddr);
- kfree(ioreq);
- }
- scm->stats.n_free_ioreq = 0;
- return -ENOMEM;
- }
- /**
- * csio_scsim_exit: Uninitialize SCSI Module
- * @scm: SCSI Module
- *
- */
- void
- csio_scsim_exit(struct csio_scsim *scm)
- {
- struct csio_ioreq *ioreq;
- struct csio_dma_buf *dma_buf;
- while (!list_empty(&scm->ioreq_freelist)) {
- struct csio_sm *tmp;
- tmp = list_first_entry(&scm->ioreq_freelist,
- struct csio_sm, sm_list);
- list_del_init(&tmp->sm_list);
- ioreq = (struct csio_ioreq *)tmp;
- dma_buf = &ioreq->dma_buf;
- dma_pool_free(scm->hw->scsi_dma_pool, dma_buf->vaddr,
- dma_buf->paddr);
- kfree(ioreq);
- }
- scm->stats.n_free_ioreq = 0;
- csio_scsi_free_ddp_bufs(scm, scm->hw);
- }
|