1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228122912301231123212331234123512361237123812391240124112421243124412451246124712481249125012511252125312541255125612571258125912601261126212631264126512661267126812691270127112721273127412751276127712781279128012811282128312841285128612871288128912901291129212931294129512961297129812991300130113021303130413051306130713081309131013111312131313141315131613171318131913201321132213231324132513261327132813291330133113321333133413351336133713381339134013411342134313441345134613471348134913501351135213531354135513561357135813591360136113621363136413651366136713681369137013711372137313741375137613771378137913801381138213831384138513861387138813891390139113921393139413951396139713981399140014011402140314041405140614071408140914101411141214131414141514161417141814191420142114221423142414251426142714281429143014311432143314341435143614371438143914401441144214431444144514461447144814491450145114521453145414551456145714581459146014611462146314641465146614671468146914701471147214731474147514761477147814791480148114821483148414851486148714881489149014911492149314941495149614971498149915001501150215031504150515061507150815091510151115121513151415151516151715181519152015211522152315241525152615271528152915301531153215331534153515361537153815391540154115421543154415451546154715481549155015511552155315541555155615571558155915601561156215631564156515661567156815691570157115721573157415751576157715781579158015811582158315841585158615871588158915901591159215931594159515961597159815991600160116021603160416051606160716081609161016111612161316141615161616171618161916201621162216231624162516261627162816291630163116321633163416351636163716381639164016411642164316441645164616471648164916501651165216531654165516561657165816591660166116621663166416651666166716681669167016711672167316741675167616771678167916801681168216831684168516861687168816891690169116921693169416951696169716981699170017011702170317041705170617071708170917101711171217131714171517161717171817191720172117221723172417251726172717281729173017311732173317341735173617371738173917401741174217431744174517461747174817491750175117521753175417551756175717581759176017611762176317641765176617671768176917701771177217731774177517761777177817791780178117821783178417851786178717881789179017911792179317941795179617971798179918001801180218031804180518061807180818091810181118121813181418151816181718181819182018211822182318241825182618271828182918301831183218331834183518361837183818391840184118421843184418451846184718481849185018511852185318541855185618571858185918601861186218631864186518661867186818691870187118721873187418751876187718781879188018811882188318841885188618871888188918901891189218931894189518961897189818991900190119021903190419051906190719081909191019111912191319141915191619171918191919201921192219231924192519261927192819291930193119321933193419351936193719381939194019411942194319441945194619471948194919501951195219531954195519561957195819591960196119621963196419651966196719681969197019711972197319741975197619771978197919801981198219831984198519861987198819891990199119921993199419951996199719981999200020012002200320042005200620072008200920102011201220132014201520162017201820192020202120222023202420252026202720282029203020312032203320342035203620372038203920402041204220432044204520462047204820492050205120522053205420552056205720582059206020612062206320642065206620672068206920702071207220732074207520762077207820792080208120822083208420852086208720882089209020912092209320942095209620972098209921002101210221032104210521062107210821092110211121122113211421152116211721182119212021212122212321242125212621272128212921302131213221332134213521362137213821392140214121422143214421452146214721482149215021512152 |
- /*
- * Linux network driver for Brocade Converged Network Adapter.
- *
- * This program is free software; you can redistribute it and/or modify it
- * under the terms of the GNU General Public License (GPL) Version 2 as
- * published by the Free Software Foundation
- *
- * This program is distributed in the hope that it will be useful, but
- * WITHOUT ANY WARRANTY; without even the implied warranty of
- * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
- * General Public License for more details.
- */
- /*
- * Copyright (c) 2005-2011 Brocade Communications Systems, Inc.
- * All rights reserved
- * www.brocade.com
- */
- #include "bna.h"
- static inline int
- ethport_can_be_up(struct bna_ethport *ethport)
- {
- int ready = 0;
- if (ethport->bna->enet.type == BNA_ENET_T_REGULAR)
- ready = ((ethport->flags & BNA_ETHPORT_F_ADMIN_UP) &&
- (ethport->flags & BNA_ETHPORT_F_RX_STARTED) &&
- (ethport->flags & BNA_ETHPORT_F_PORT_ENABLED));
- else
- ready = ((ethport->flags & BNA_ETHPORT_F_ADMIN_UP) &&
- (ethport->flags & BNA_ETHPORT_F_RX_STARTED) &&
- !(ethport->flags & BNA_ETHPORT_F_PORT_ENABLED));
- return ready;
- }
- #define ethport_is_up ethport_can_be_up
- enum bna_ethport_event {
- ETHPORT_E_START = 1,
- ETHPORT_E_STOP = 2,
- ETHPORT_E_FAIL = 3,
- ETHPORT_E_UP = 4,
- ETHPORT_E_DOWN = 5,
- ETHPORT_E_FWRESP_UP_OK = 6,
- ETHPORT_E_FWRESP_DOWN = 7,
- ETHPORT_E_FWRESP_UP_FAIL = 8,
- };
- enum bna_enet_event {
- ENET_E_START = 1,
- ENET_E_STOP = 2,
- ENET_E_FAIL = 3,
- ENET_E_PAUSE_CFG = 4,
- ENET_E_MTU_CFG = 5,
- ENET_E_FWRESP_PAUSE = 6,
- ENET_E_CHLD_STOPPED = 7,
- };
- enum bna_ioceth_event {
- IOCETH_E_ENABLE = 1,
- IOCETH_E_DISABLE = 2,
- IOCETH_E_IOC_RESET = 3,
- IOCETH_E_IOC_FAILED = 4,
- IOCETH_E_IOC_READY = 5,
- IOCETH_E_ENET_ATTR_RESP = 6,
- IOCETH_E_ENET_STOPPED = 7,
- IOCETH_E_IOC_DISABLED = 8,
- };
- #define bna_stats_copy(_name, _type) \
- do { \
- count = sizeof(struct bfi_enet_stats_ ## _type) / sizeof(u64); \
- stats_src = (u64 *)&bna->stats.hw_stats_kva->_name ## _stats; \
- stats_dst = (u64 *)&bna->stats.hw_stats._name ## _stats; \
- for (i = 0; i < count; i++) \
- stats_dst[i] = be64_to_cpu(stats_src[i]); \
- } while (0) \
- /*
- * FW response handlers
- */
- static void
- bna_bfi_ethport_enable_aen(struct bna_ethport *ethport,
- struct bfi_msgq_mhdr *msghdr)
- {
- ethport->flags |= BNA_ETHPORT_F_PORT_ENABLED;
- if (ethport_can_be_up(ethport))
- bfa_fsm_send_event(ethport, ETHPORT_E_UP);
- }
- static void
- bna_bfi_ethport_disable_aen(struct bna_ethport *ethport,
- struct bfi_msgq_mhdr *msghdr)
- {
- int ethport_up = ethport_is_up(ethport);
- ethport->flags &= ~BNA_ETHPORT_F_PORT_ENABLED;
- if (ethport_up)
- bfa_fsm_send_event(ethport, ETHPORT_E_DOWN);
- }
- static void
- bna_bfi_ethport_admin_rsp(struct bna_ethport *ethport,
- struct bfi_msgq_mhdr *msghdr)
- {
- struct bfi_enet_enable_req *admin_req =
- ðport->bfi_enet_cmd.admin_req;
- struct bfi_enet_rsp *rsp = (struct bfi_enet_rsp *)msghdr;
- switch (admin_req->enable) {
- case BNA_STATUS_T_ENABLED:
- if (rsp->error == BFI_ENET_CMD_OK)
- bfa_fsm_send_event(ethport, ETHPORT_E_FWRESP_UP_OK);
- else {
- ethport->flags &= ~BNA_ETHPORT_F_PORT_ENABLED;
- bfa_fsm_send_event(ethport, ETHPORT_E_FWRESP_UP_FAIL);
- }
- break;
- case BNA_STATUS_T_DISABLED:
- bfa_fsm_send_event(ethport, ETHPORT_E_FWRESP_DOWN);
- ethport->link_status = BNA_LINK_DOWN;
- ethport->link_cbfn(ethport->bna->bnad, BNA_LINK_DOWN);
- break;
- }
- }
- static void
- bna_bfi_ethport_lpbk_rsp(struct bna_ethport *ethport,
- struct bfi_msgq_mhdr *msghdr)
- {
- struct bfi_enet_diag_lb_req *diag_lb_req =
- ðport->bfi_enet_cmd.lpbk_req;
- struct bfi_enet_rsp *rsp = (struct bfi_enet_rsp *)msghdr;
- switch (diag_lb_req->enable) {
- case BNA_STATUS_T_ENABLED:
- if (rsp->error == BFI_ENET_CMD_OK)
- bfa_fsm_send_event(ethport, ETHPORT_E_FWRESP_UP_OK);
- else {
- ethport->flags &= ~BNA_ETHPORT_F_ADMIN_UP;
- bfa_fsm_send_event(ethport, ETHPORT_E_FWRESP_UP_FAIL);
- }
- break;
- case BNA_STATUS_T_DISABLED:
- bfa_fsm_send_event(ethport, ETHPORT_E_FWRESP_DOWN);
- break;
- }
- }
- static void
- bna_bfi_pause_set_rsp(struct bna_enet *enet, struct bfi_msgq_mhdr *msghdr)
- {
- bfa_fsm_send_event(enet, ENET_E_FWRESP_PAUSE);
- }
- static void
- bna_bfi_attr_get_rsp(struct bna_ioceth *ioceth,
- struct bfi_msgq_mhdr *msghdr)
- {
- struct bfi_enet_attr_rsp *rsp = (struct bfi_enet_attr_rsp *)msghdr;
- /**
- * Store only if not set earlier, since BNAD can override the HW
- * attributes
- */
- if (!ioceth->attr.fw_query_complete) {
- ioceth->attr.num_txq = ntohl(rsp->max_cfg);
- ioceth->attr.num_rxp = ntohl(rsp->max_cfg);
- ioceth->attr.num_ucmac = ntohl(rsp->max_ucmac);
- ioceth->attr.num_mcmac = BFI_ENET_MAX_MCAM;
- ioceth->attr.max_rit_size = ntohl(rsp->rit_size);
- ioceth->attr.fw_query_complete = true;
- }
- bfa_fsm_send_event(ioceth, IOCETH_E_ENET_ATTR_RESP);
- }
- static void
- bna_bfi_stats_get_rsp(struct bna *bna, struct bfi_msgq_mhdr *msghdr)
- {
- struct bfi_enet_stats_req *stats_req = &bna->stats_mod.stats_get;
- u64 *stats_src;
- u64 *stats_dst;
- u32 tx_enet_mask = ntohl(stats_req->tx_enet_mask);
- u32 rx_enet_mask = ntohl(stats_req->rx_enet_mask);
- int count;
- int i;
- bna_stats_copy(mac, mac);
- bna_stats_copy(bpc, bpc);
- bna_stats_copy(rad, rad);
- bna_stats_copy(rlb, rad);
- bna_stats_copy(fc_rx, fc_rx);
- bna_stats_copy(fc_tx, fc_tx);
- stats_src = (u64 *)&(bna->stats.hw_stats_kva->rxf_stats[0]);
- /* Copy Rxf stats to SW area, scatter them while copying */
- for (i = 0; i < BFI_ENET_CFG_MAX; i++) {
- stats_dst = (u64 *)&(bna->stats.hw_stats.rxf_stats[i]);
- memset(stats_dst, 0, sizeof(struct bfi_enet_stats_rxf));
- if (rx_enet_mask & ((u32)(1 << i))) {
- int k;
- count = sizeof(struct bfi_enet_stats_rxf) /
- sizeof(u64);
- for (k = 0; k < count; k++) {
- stats_dst[k] = be64_to_cpu(*stats_src);
- stats_src++;
- }
- }
- }
- /* Copy Txf stats to SW area, scatter them while copying */
- for (i = 0; i < BFI_ENET_CFG_MAX; i++) {
- stats_dst = (u64 *)&(bna->stats.hw_stats.txf_stats[i]);
- memset(stats_dst, 0, sizeof(struct bfi_enet_stats_txf));
- if (tx_enet_mask & ((u32)(1 << i))) {
- int k;
- count = sizeof(struct bfi_enet_stats_txf) /
- sizeof(u64);
- for (k = 0; k < count; k++) {
- stats_dst[k] = be64_to_cpu(*stats_src);
- stats_src++;
- }
- }
- }
- bna->stats_mod.stats_get_busy = false;
- bnad_cb_stats_get(bna->bnad, BNA_CB_SUCCESS, &bna->stats);
- }
- static void
- bna_bfi_ethport_linkup_aen(struct bna_ethport *ethport,
- struct bfi_msgq_mhdr *msghdr)
- {
- ethport->link_status = BNA_LINK_UP;
- /* Dispatch events */
- ethport->link_cbfn(ethport->bna->bnad, ethport->link_status);
- }
- static void
- bna_bfi_ethport_linkdown_aen(struct bna_ethport *ethport,
- struct bfi_msgq_mhdr *msghdr)
- {
- ethport->link_status = BNA_LINK_DOWN;
- /* Dispatch events */
- ethport->link_cbfn(ethport->bna->bnad, BNA_LINK_DOWN);
- }
- static void
- bna_err_handler(struct bna *bna, u32 intr_status)
- {
- if (BNA_IS_HALT_INTR(bna, intr_status))
- bna_halt_clear(bna);
- bfa_nw_ioc_error_isr(&bna->ioceth.ioc);
- }
- void
- bna_mbox_handler(struct bna *bna, u32 intr_status)
- {
- if (BNA_IS_ERR_INTR(bna, intr_status)) {
- bna_err_handler(bna, intr_status);
- return;
- }
- if (BNA_IS_MBOX_INTR(bna, intr_status))
- bfa_nw_ioc_mbox_isr(&bna->ioceth.ioc);
- }
- static void
- bna_msgq_rsp_handler(void *arg, struct bfi_msgq_mhdr *msghdr)
- {
- struct bna *bna = (struct bna *)arg;
- struct bna_tx *tx;
- struct bna_rx *rx;
- switch (msghdr->msg_id) {
- case BFI_ENET_I2H_RX_CFG_SET_RSP:
- bna_rx_from_rid(bna, msghdr->enet_id, rx);
- if (rx)
- bna_bfi_rx_enet_start_rsp(rx, msghdr);
- break;
- case BFI_ENET_I2H_RX_CFG_CLR_RSP:
- bna_rx_from_rid(bna, msghdr->enet_id, rx);
- if (rx)
- bna_bfi_rx_enet_stop_rsp(rx, msghdr);
- break;
- case BFI_ENET_I2H_RIT_CFG_RSP:
- case BFI_ENET_I2H_RSS_CFG_RSP:
- case BFI_ENET_I2H_RSS_ENABLE_RSP:
- case BFI_ENET_I2H_RX_PROMISCUOUS_RSP:
- case BFI_ENET_I2H_RX_DEFAULT_RSP:
- case BFI_ENET_I2H_MAC_UCAST_SET_RSP:
- case BFI_ENET_I2H_MAC_UCAST_CLR_RSP:
- case BFI_ENET_I2H_MAC_UCAST_ADD_RSP:
- case BFI_ENET_I2H_MAC_UCAST_DEL_RSP:
- case BFI_ENET_I2H_MAC_MCAST_DEL_RSP:
- case BFI_ENET_I2H_MAC_MCAST_FILTER_RSP:
- case BFI_ENET_I2H_RX_VLAN_SET_RSP:
- case BFI_ENET_I2H_RX_VLAN_STRIP_ENABLE_RSP:
- bna_rx_from_rid(bna, msghdr->enet_id, rx);
- if (rx)
- bna_bfi_rxf_cfg_rsp(&rx->rxf, msghdr);
- break;
- case BFI_ENET_I2H_MAC_MCAST_ADD_RSP:
- bna_rx_from_rid(bna, msghdr->enet_id, rx);
- if (rx)
- bna_bfi_rxf_mcast_add_rsp(&rx->rxf, msghdr);
- break;
- case BFI_ENET_I2H_TX_CFG_SET_RSP:
- bna_tx_from_rid(bna, msghdr->enet_id, tx);
- if (tx)
- bna_bfi_tx_enet_start_rsp(tx, msghdr);
- break;
- case BFI_ENET_I2H_TX_CFG_CLR_RSP:
- bna_tx_from_rid(bna, msghdr->enet_id, tx);
- if (tx)
- bna_bfi_tx_enet_stop_rsp(tx, msghdr);
- break;
- case BFI_ENET_I2H_PORT_ADMIN_RSP:
- bna_bfi_ethport_admin_rsp(&bna->ethport, msghdr);
- break;
- case BFI_ENET_I2H_DIAG_LOOPBACK_RSP:
- bna_bfi_ethport_lpbk_rsp(&bna->ethport, msghdr);
- break;
- case BFI_ENET_I2H_SET_PAUSE_RSP:
- bna_bfi_pause_set_rsp(&bna->enet, msghdr);
- break;
- case BFI_ENET_I2H_GET_ATTR_RSP:
- bna_bfi_attr_get_rsp(&bna->ioceth, msghdr);
- break;
- case BFI_ENET_I2H_STATS_GET_RSP:
- bna_bfi_stats_get_rsp(bna, msghdr);
- break;
- case BFI_ENET_I2H_STATS_CLR_RSP:
- /* No-op */
- break;
- case BFI_ENET_I2H_LINK_UP_AEN:
- bna_bfi_ethport_linkup_aen(&bna->ethport, msghdr);
- break;
- case BFI_ENET_I2H_LINK_DOWN_AEN:
- bna_bfi_ethport_linkdown_aen(&bna->ethport, msghdr);
- break;
- case BFI_ENET_I2H_PORT_ENABLE_AEN:
- bna_bfi_ethport_enable_aen(&bna->ethport, msghdr);
- break;
- case BFI_ENET_I2H_PORT_DISABLE_AEN:
- bna_bfi_ethport_disable_aen(&bna->ethport, msghdr);
- break;
- case BFI_ENET_I2H_BW_UPDATE_AEN:
- bna_bfi_bw_update_aen(&bna->tx_mod);
- break;
- default:
- break;
- }
- }
- /**
- * ETHPORT
- */
- #define call_ethport_stop_cbfn(_ethport) \
- do { \
- if ((_ethport)->stop_cbfn) { \
- void (*cbfn)(struct bna_enet *); \
- cbfn = (_ethport)->stop_cbfn; \
- (_ethport)->stop_cbfn = NULL; \
- cbfn(&(_ethport)->bna->enet); \
- } \
- } while (0)
- #define call_ethport_adminup_cbfn(ethport, status) \
- do { \
- if ((ethport)->adminup_cbfn) { \
- void (*cbfn)(struct bnad *, enum bna_cb_status); \
- cbfn = (ethport)->adminup_cbfn; \
- (ethport)->adminup_cbfn = NULL; \
- cbfn((ethport)->bna->bnad, status); \
- } \
- } while (0)
- static void
- bna_bfi_ethport_admin_up(struct bna_ethport *ethport)
- {
- struct bfi_enet_enable_req *admin_up_req =
- ðport->bfi_enet_cmd.admin_req;
- bfi_msgq_mhdr_set(admin_up_req->mh, BFI_MC_ENET,
- BFI_ENET_H2I_PORT_ADMIN_UP_REQ, 0, 0);
- admin_up_req->mh.num_entries = htons(
- bfi_msgq_num_cmd_entries(sizeof(struct bfi_enet_enable_req)));
- admin_up_req->enable = BNA_STATUS_T_ENABLED;
- bfa_msgq_cmd_set(ðport->msgq_cmd, NULL, NULL,
- sizeof(struct bfi_enet_enable_req), &admin_up_req->mh);
- bfa_msgq_cmd_post(ðport->bna->msgq, ðport->msgq_cmd);
- }
- static void
- bna_bfi_ethport_admin_down(struct bna_ethport *ethport)
- {
- struct bfi_enet_enable_req *admin_down_req =
- ðport->bfi_enet_cmd.admin_req;
- bfi_msgq_mhdr_set(admin_down_req->mh, BFI_MC_ENET,
- BFI_ENET_H2I_PORT_ADMIN_UP_REQ, 0, 0);
- admin_down_req->mh.num_entries = htons(
- bfi_msgq_num_cmd_entries(sizeof(struct bfi_enet_enable_req)));
- admin_down_req->enable = BNA_STATUS_T_DISABLED;
- bfa_msgq_cmd_set(ðport->msgq_cmd, NULL, NULL,
- sizeof(struct bfi_enet_enable_req), &admin_down_req->mh);
- bfa_msgq_cmd_post(ðport->bna->msgq, ðport->msgq_cmd);
- }
- static void
- bna_bfi_ethport_lpbk_up(struct bna_ethport *ethport)
- {
- struct bfi_enet_diag_lb_req *lpbk_up_req =
- ðport->bfi_enet_cmd.lpbk_req;
- bfi_msgq_mhdr_set(lpbk_up_req->mh, BFI_MC_ENET,
- BFI_ENET_H2I_DIAG_LOOPBACK_REQ, 0, 0);
- lpbk_up_req->mh.num_entries = htons(
- bfi_msgq_num_cmd_entries(sizeof(struct bfi_enet_diag_lb_req)));
- lpbk_up_req->mode = (ethport->bna->enet.type ==
- BNA_ENET_T_LOOPBACK_INTERNAL) ?
- BFI_ENET_DIAG_LB_OPMODE_EXT :
- BFI_ENET_DIAG_LB_OPMODE_CBL;
- lpbk_up_req->enable = BNA_STATUS_T_ENABLED;
- bfa_msgq_cmd_set(ðport->msgq_cmd, NULL, NULL,
- sizeof(struct bfi_enet_diag_lb_req), &lpbk_up_req->mh);
- bfa_msgq_cmd_post(ðport->bna->msgq, ðport->msgq_cmd);
- }
- static void
- bna_bfi_ethport_lpbk_down(struct bna_ethport *ethport)
- {
- struct bfi_enet_diag_lb_req *lpbk_down_req =
- ðport->bfi_enet_cmd.lpbk_req;
- bfi_msgq_mhdr_set(lpbk_down_req->mh, BFI_MC_ENET,
- BFI_ENET_H2I_DIAG_LOOPBACK_REQ, 0, 0);
- lpbk_down_req->mh.num_entries = htons(
- bfi_msgq_num_cmd_entries(sizeof(struct bfi_enet_diag_lb_req)));
- lpbk_down_req->enable = BNA_STATUS_T_DISABLED;
- bfa_msgq_cmd_set(ðport->msgq_cmd, NULL, NULL,
- sizeof(struct bfi_enet_diag_lb_req), &lpbk_down_req->mh);
- bfa_msgq_cmd_post(ðport->bna->msgq, ðport->msgq_cmd);
- }
- static void
- bna_bfi_ethport_up(struct bna_ethport *ethport)
- {
- if (ethport->bna->enet.type == BNA_ENET_T_REGULAR)
- bna_bfi_ethport_admin_up(ethport);
- else
- bna_bfi_ethport_lpbk_up(ethport);
- }
- static void
- bna_bfi_ethport_down(struct bna_ethport *ethport)
- {
- if (ethport->bna->enet.type == BNA_ENET_T_REGULAR)
- bna_bfi_ethport_admin_down(ethport);
- else
- bna_bfi_ethport_lpbk_down(ethport);
- }
- bfa_fsm_state_decl(bna_ethport, stopped, struct bna_ethport,
- enum bna_ethport_event);
- bfa_fsm_state_decl(bna_ethport, down, struct bna_ethport,
- enum bna_ethport_event);
- bfa_fsm_state_decl(bna_ethport, up_resp_wait, struct bna_ethport,
- enum bna_ethport_event);
- bfa_fsm_state_decl(bna_ethport, down_resp_wait, struct bna_ethport,
- enum bna_ethport_event);
- bfa_fsm_state_decl(bna_ethport, up, struct bna_ethport,
- enum bna_ethport_event);
- bfa_fsm_state_decl(bna_ethport, last_resp_wait, struct bna_ethport,
- enum bna_ethport_event);
- static void
- bna_ethport_sm_stopped_entry(struct bna_ethport *ethport)
- {
- call_ethport_stop_cbfn(ethport);
- }
- static void
- bna_ethport_sm_stopped(struct bna_ethport *ethport,
- enum bna_ethport_event event)
- {
- switch (event) {
- case ETHPORT_E_START:
- bfa_fsm_set_state(ethport, bna_ethport_sm_down);
- break;
- case ETHPORT_E_STOP:
- call_ethport_stop_cbfn(ethport);
- break;
- case ETHPORT_E_FAIL:
- /* No-op */
- break;
- case ETHPORT_E_DOWN:
- /* This event is received due to Rx objects failing */
- /* No-op */
- break;
- default:
- bfa_sm_fault(event);
- }
- }
- static void
- bna_ethport_sm_down_entry(struct bna_ethport *ethport)
- {
- }
- static void
- bna_ethport_sm_down(struct bna_ethport *ethport,
- enum bna_ethport_event event)
- {
- switch (event) {
- case ETHPORT_E_STOP:
- bfa_fsm_set_state(ethport, bna_ethport_sm_stopped);
- break;
- case ETHPORT_E_FAIL:
- bfa_fsm_set_state(ethport, bna_ethport_sm_stopped);
- break;
- case ETHPORT_E_UP:
- bfa_fsm_set_state(ethport, bna_ethport_sm_up_resp_wait);
- bna_bfi_ethport_up(ethport);
- break;
- default:
- bfa_sm_fault(event);
- }
- }
- static void
- bna_ethport_sm_up_resp_wait_entry(struct bna_ethport *ethport)
- {
- }
- static void
- bna_ethport_sm_up_resp_wait(struct bna_ethport *ethport,
- enum bna_ethport_event event)
- {
- switch (event) {
- case ETHPORT_E_STOP:
- bfa_fsm_set_state(ethport, bna_ethport_sm_last_resp_wait);
- break;
- case ETHPORT_E_FAIL:
- call_ethport_adminup_cbfn(ethport, BNA_CB_FAIL);
- bfa_fsm_set_state(ethport, bna_ethport_sm_stopped);
- break;
- case ETHPORT_E_DOWN:
- call_ethport_adminup_cbfn(ethport, BNA_CB_INTERRUPT);
- bfa_fsm_set_state(ethport, bna_ethport_sm_down_resp_wait);
- break;
- case ETHPORT_E_FWRESP_UP_OK:
- call_ethport_adminup_cbfn(ethport, BNA_CB_SUCCESS);
- bfa_fsm_set_state(ethport, bna_ethport_sm_up);
- break;
- case ETHPORT_E_FWRESP_UP_FAIL:
- call_ethport_adminup_cbfn(ethport, BNA_CB_FAIL);
- bfa_fsm_set_state(ethport, bna_ethport_sm_down);
- break;
- case ETHPORT_E_FWRESP_DOWN:
- /* down_resp_wait -> up_resp_wait transition on ETHPORT_E_UP */
- bna_bfi_ethport_up(ethport);
- break;
- default:
- bfa_sm_fault(event);
- }
- }
- static void
- bna_ethport_sm_down_resp_wait_entry(struct bna_ethport *ethport)
- {
- /**
- * NOTE: Do not call bna_bfi_ethport_down() here. That will over step
- * mbox due to up_resp_wait -> down_resp_wait transition on event
- * ETHPORT_E_DOWN
- */
- }
- static void
- bna_ethport_sm_down_resp_wait(struct bna_ethport *ethport,
- enum bna_ethport_event event)
- {
- switch (event) {
- case ETHPORT_E_STOP:
- bfa_fsm_set_state(ethport, bna_ethport_sm_last_resp_wait);
- break;
- case ETHPORT_E_FAIL:
- bfa_fsm_set_state(ethport, bna_ethport_sm_stopped);
- break;
- case ETHPORT_E_UP:
- bfa_fsm_set_state(ethport, bna_ethport_sm_up_resp_wait);
- break;
- case ETHPORT_E_FWRESP_UP_OK:
- /* up_resp_wait->down_resp_wait transition on ETHPORT_E_DOWN */
- bna_bfi_ethport_down(ethport);
- break;
- case ETHPORT_E_FWRESP_UP_FAIL:
- case ETHPORT_E_FWRESP_DOWN:
- bfa_fsm_set_state(ethport, bna_ethport_sm_down);
- break;
- default:
- bfa_sm_fault(event);
- }
- }
- static void
- bna_ethport_sm_up_entry(struct bna_ethport *ethport)
- {
- }
- static void
- bna_ethport_sm_up(struct bna_ethport *ethport,
- enum bna_ethport_event event)
- {
- switch (event) {
- case ETHPORT_E_STOP:
- bfa_fsm_set_state(ethport, bna_ethport_sm_last_resp_wait);
- bna_bfi_ethport_down(ethport);
- break;
- case ETHPORT_E_FAIL:
- bfa_fsm_set_state(ethport, bna_ethport_sm_stopped);
- break;
- case ETHPORT_E_DOWN:
- bfa_fsm_set_state(ethport, bna_ethport_sm_down_resp_wait);
- bna_bfi_ethport_down(ethport);
- break;
- default:
- bfa_sm_fault(event);
- }
- }
- static void
- bna_ethport_sm_last_resp_wait_entry(struct bna_ethport *ethport)
- {
- }
- static void
- bna_ethport_sm_last_resp_wait(struct bna_ethport *ethport,
- enum bna_ethport_event event)
- {
- switch (event) {
- case ETHPORT_E_FAIL:
- bfa_fsm_set_state(ethport, bna_ethport_sm_stopped);
- break;
- case ETHPORT_E_DOWN:
- /**
- * This event is received due to Rx objects stopping in
- * parallel to ethport
- */
- /* No-op */
- break;
- case ETHPORT_E_FWRESP_UP_OK:
- /* up_resp_wait->last_resp_wait transition on ETHPORT_T_STOP */
- bna_bfi_ethport_down(ethport);
- break;
- case ETHPORT_E_FWRESP_UP_FAIL:
- case ETHPORT_E_FWRESP_DOWN:
- bfa_fsm_set_state(ethport, bna_ethport_sm_stopped);
- break;
- default:
- bfa_sm_fault(event);
- }
- }
- static void
- bna_ethport_init(struct bna_ethport *ethport, struct bna *bna)
- {
- ethport->flags |= (BNA_ETHPORT_F_ADMIN_UP | BNA_ETHPORT_F_PORT_ENABLED);
- ethport->bna = bna;
- ethport->link_status = BNA_LINK_DOWN;
- ethport->link_cbfn = bnad_cb_ethport_link_status;
- ethport->rx_started_count = 0;
- ethport->stop_cbfn = NULL;
- ethport->adminup_cbfn = NULL;
- bfa_fsm_set_state(ethport, bna_ethport_sm_stopped);
- }
- static void
- bna_ethport_uninit(struct bna_ethport *ethport)
- {
- ethport->flags &= ~BNA_ETHPORT_F_ADMIN_UP;
- ethport->flags &= ~BNA_ETHPORT_F_PORT_ENABLED;
- ethport->bna = NULL;
- }
- static void
- bna_ethport_start(struct bna_ethport *ethport)
- {
- bfa_fsm_send_event(ethport, ETHPORT_E_START);
- }
- static void
- bna_enet_cb_ethport_stopped(struct bna_enet *enet)
- {
- bfa_wc_down(&enet->chld_stop_wc);
- }
- static void
- bna_ethport_stop(struct bna_ethport *ethport)
- {
- ethport->stop_cbfn = bna_enet_cb_ethport_stopped;
- bfa_fsm_send_event(ethport, ETHPORT_E_STOP);
- }
- static void
- bna_ethport_fail(struct bna_ethport *ethport)
- {
- /* Reset the physical port status to enabled */
- ethport->flags |= BNA_ETHPORT_F_PORT_ENABLED;
- if (ethport->link_status != BNA_LINK_DOWN) {
- ethport->link_status = BNA_LINK_DOWN;
- ethport->link_cbfn(ethport->bna->bnad, BNA_LINK_DOWN);
- }
- bfa_fsm_send_event(ethport, ETHPORT_E_FAIL);
- }
- /* Should be called only when ethport is disabled */
- void
- bna_ethport_cb_rx_started(struct bna_ethport *ethport)
- {
- ethport->rx_started_count++;
- if (ethport->rx_started_count == 1) {
- ethport->flags |= BNA_ETHPORT_F_RX_STARTED;
- if (ethport_can_be_up(ethport))
- bfa_fsm_send_event(ethport, ETHPORT_E_UP);
- }
- }
- void
- bna_ethport_cb_rx_stopped(struct bna_ethport *ethport)
- {
- int ethport_up = ethport_is_up(ethport);
- ethport->rx_started_count--;
- if (ethport->rx_started_count == 0) {
- ethport->flags &= ~BNA_ETHPORT_F_RX_STARTED;
- if (ethport_up)
- bfa_fsm_send_event(ethport, ETHPORT_E_DOWN);
- }
- }
- /**
- * ENET
- */
- #define bna_enet_chld_start(enet) \
- do { \
- enum bna_tx_type tx_type = \
- ((enet)->type == BNA_ENET_T_REGULAR) ? \
- BNA_TX_T_REGULAR : BNA_TX_T_LOOPBACK; \
- enum bna_rx_type rx_type = \
- ((enet)->type == BNA_ENET_T_REGULAR) ? \
- BNA_RX_T_REGULAR : BNA_RX_T_LOOPBACK; \
- bna_ethport_start(&(enet)->bna->ethport); \
- bna_tx_mod_start(&(enet)->bna->tx_mod, tx_type); \
- bna_rx_mod_start(&(enet)->bna->rx_mod, rx_type); \
- } while (0)
- #define bna_enet_chld_stop(enet) \
- do { \
- enum bna_tx_type tx_type = \
- ((enet)->type == BNA_ENET_T_REGULAR) ? \
- BNA_TX_T_REGULAR : BNA_TX_T_LOOPBACK; \
- enum bna_rx_type rx_type = \
- ((enet)->type == BNA_ENET_T_REGULAR) ? \
- BNA_RX_T_REGULAR : BNA_RX_T_LOOPBACK; \
- bfa_wc_init(&(enet)->chld_stop_wc, bna_enet_cb_chld_stopped, (enet));\
- bfa_wc_up(&(enet)->chld_stop_wc); \
- bna_ethport_stop(&(enet)->bna->ethport); \
- bfa_wc_up(&(enet)->chld_stop_wc); \
- bna_tx_mod_stop(&(enet)->bna->tx_mod, tx_type); \
- bfa_wc_up(&(enet)->chld_stop_wc); \
- bna_rx_mod_stop(&(enet)->bna->rx_mod, rx_type); \
- bfa_wc_wait(&(enet)->chld_stop_wc); \
- } while (0)
- #define bna_enet_chld_fail(enet) \
- do { \
- bna_ethport_fail(&(enet)->bna->ethport); \
- bna_tx_mod_fail(&(enet)->bna->tx_mod); \
- bna_rx_mod_fail(&(enet)->bna->rx_mod); \
- } while (0)
- #define bna_enet_rx_start(enet) \
- do { \
- enum bna_rx_type rx_type = \
- ((enet)->type == BNA_ENET_T_REGULAR) ? \
- BNA_RX_T_REGULAR : BNA_RX_T_LOOPBACK; \
- bna_rx_mod_start(&(enet)->bna->rx_mod, rx_type); \
- } while (0)
- #define bna_enet_rx_stop(enet) \
- do { \
- enum bna_rx_type rx_type = \
- ((enet)->type == BNA_ENET_T_REGULAR) ? \
- BNA_RX_T_REGULAR : BNA_RX_T_LOOPBACK; \
- bfa_wc_init(&(enet)->chld_stop_wc, bna_enet_cb_chld_stopped, (enet));\
- bfa_wc_up(&(enet)->chld_stop_wc); \
- bna_rx_mod_stop(&(enet)->bna->rx_mod, rx_type); \
- bfa_wc_wait(&(enet)->chld_stop_wc); \
- } while (0)
- #define call_enet_stop_cbfn(enet) \
- do { \
- if ((enet)->stop_cbfn) { \
- void (*cbfn)(void *); \
- void *cbarg; \
- cbfn = (enet)->stop_cbfn; \
- cbarg = (enet)->stop_cbarg; \
- (enet)->stop_cbfn = NULL; \
- (enet)->stop_cbarg = NULL; \
- cbfn(cbarg); \
- } \
- } while (0)
- #define call_enet_pause_cbfn(enet) \
- do { \
- if ((enet)->pause_cbfn) { \
- void (*cbfn)(struct bnad *); \
- cbfn = (enet)->pause_cbfn; \
- (enet)->pause_cbfn = NULL; \
- cbfn((enet)->bna->bnad); \
- } \
- } while (0)
- #define call_enet_mtu_cbfn(enet) \
- do { \
- if ((enet)->mtu_cbfn) { \
- void (*cbfn)(struct bnad *); \
- cbfn = (enet)->mtu_cbfn; \
- (enet)->mtu_cbfn = NULL; \
- cbfn((enet)->bna->bnad); \
- } \
- } while (0)
- static void bna_enet_cb_chld_stopped(void *arg);
- static void bna_bfi_pause_set(struct bna_enet *enet);
- bfa_fsm_state_decl(bna_enet, stopped, struct bna_enet,
- enum bna_enet_event);
- bfa_fsm_state_decl(bna_enet, pause_init_wait, struct bna_enet,
- enum bna_enet_event);
- bfa_fsm_state_decl(bna_enet, last_resp_wait, struct bna_enet,
- enum bna_enet_event);
- bfa_fsm_state_decl(bna_enet, started, struct bna_enet,
- enum bna_enet_event);
- bfa_fsm_state_decl(bna_enet, cfg_wait, struct bna_enet,
- enum bna_enet_event);
- bfa_fsm_state_decl(bna_enet, cfg_stop_wait, struct bna_enet,
- enum bna_enet_event);
- bfa_fsm_state_decl(bna_enet, chld_stop_wait, struct bna_enet,
- enum bna_enet_event);
- static void
- bna_enet_sm_stopped_entry(struct bna_enet *enet)
- {
- call_enet_pause_cbfn(enet);
- call_enet_mtu_cbfn(enet);
- call_enet_stop_cbfn(enet);
- }
- static void
- bna_enet_sm_stopped(struct bna_enet *enet, enum bna_enet_event event)
- {
- switch (event) {
- case ENET_E_START:
- bfa_fsm_set_state(enet, bna_enet_sm_pause_init_wait);
- break;
- case ENET_E_STOP:
- call_enet_stop_cbfn(enet);
- break;
- case ENET_E_FAIL:
- /* No-op */
- break;
- case ENET_E_PAUSE_CFG:
- call_enet_pause_cbfn(enet);
- break;
- case ENET_E_MTU_CFG:
- call_enet_mtu_cbfn(enet);
- break;
- case ENET_E_CHLD_STOPPED:
- /**
- * This event is received due to Ethport, Tx and Rx objects
- * failing
- */
- /* No-op */
- break;
- default:
- bfa_sm_fault(event);
- }
- }
- static void
- bna_enet_sm_pause_init_wait_entry(struct bna_enet *enet)
- {
- bna_bfi_pause_set(enet);
- }
- static void
- bna_enet_sm_pause_init_wait(struct bna_enet *enet,
- enum bna_enet_event event)
- {
- switch (event) {
- case ENET_E_STOP:
- enet->flags &= ~BNA_ENET_F_PAUSE_CHANGED;
- bfa_fsm_set_state(enet, bna_enet_sm_last_resp_wait);
- break;
- case ENET_E_FAIL:
- enet->flags &= ~BNA_ENET_F_PAUSE_CHANGED;
- bfa_fsm_set_state(enet, bna_enet_sm_stopped);
- break;
- case ENET_E_PAUSE_CFG:
- enet->flags |= BNA_ENET_F_PAUSE_CHANGED;
- break;
- case ENET_E_MTU_CFG:
- /* No-op */
- break;
- case ENET_E_FWRESP_PAUSE:
- if (enet->flags & BNA_ENET_F_PAUSE_CHANGED) {
- enet->flags &= ~BNA_ENET_F_PAUSE_CHANGED;
- bna_bfi_pause_set(enet);
- } else {
- bfa_fsm_set_state(enet, bna_enet_sm_started);
- bna_enet_chld_start(enet);
- }
- break;
- default:
- bfa_sm_fault(event);
- }
- }
- static void
- bna_enet_sm_last_resp_wait_entry(struct bna_enet *enet)
- {
- enet->flags &= ~BNA_ENET_F_PAUSE_CHANGED;
- }
- static void
- bna_enet_sm_last_resp_wait(struct bna_enet *enet,
- enum bna_enet_event event)
- {
- switch (event) {
- case ENET_E_FAIL:
- case ENET_E_FWRESP_PAUSE:
- bfa_fsm_set_state(enet, bna_enet_sm_stopped);
- break;
- default:
- bfa_sm_fault(event);
- }
- }
- static void
- bna_enet_sm_started_entry(struct bna_enet *enet)
- {
- /**
- * NOTE: Do not call bna_enet_chld_start() here, since it will be
- * inadvertently called during cfg_wait->started transition as well
- */
- call_enet_pause_cbfn(enet);
- call_enet_mtu_cbfn(enet);
- }
- static void
- bna_enet_sm_started(struct bna_enet *enet,
- enum bna_enet_event event)
- {
- switch (event) {
- case ENET_E_STOP:
- bfa_fsm_set_state(enet, bna_enet_sm_chld_stop_wait);
- break;
- case ENET_E_FAIL:
- bfa_fsm_set_state(enet, bna_enet_sm_stopped);
- bna_enet_chld_fail(enet);
- break;
- case ENET_E_PAUSE_CFG:
- bfa_fsm_set_state(enet, bna_enet_sm_cfg_wait);
- bna_bfi_pause_set(enet);
- break;
- case ENET_E_MTU_CFG:
- bfa_fsm_set_state(enet, bna_enet_sm_cfg_wait);
- bna_enet_rx_stop(enet);
- break;
- default:
- bfa_sm_fault(event);
- }
- }
- static void
- bna_enet_sm_cfg_wait_entry(struct bna_enet *enet)
- {
- }
- static void
- bna_enet_sm_cfg_wait(struct bna_enet *enet,
- enum bna_enet_event event)
- {
- switch (event) {
- case ENET_E_STOP:
- enet->flags &= ~BNA_ENET_F_PAUSE_CHANGED;
- enet->flags &= ~BNA_ENET_F_MTU_CHANGED;
- bfa_fsm_set_state(enet, bna_enet_sm_cfg_stop_wait);
- break;
- case ENET_E_FAIL:
- enet->flags &= ~BNA_ENET_F_PAUSE_CHANGED;
- enet->flags &= ~BNA_ENET_F_MTU_CHANGED;
- bfa_fsm_set_state(enet, bna_enet_sm_stopped);
- bna_enet_chld_fail(enet);
- break;
- case ENET_E_PAUSE_CFG:
- enet->flags |= BNA_ENET_F_PAUSE_CHANGED;
- break;
- case ENET_E_MTU_CFG:
- enet->flags |= BNA_ENET_F_MTU_CHANGED;
- break;
- case ENET_E_CHLD_STOPPED:
- bna_enet_rx_start(enet);
- /* Fall through */
- case ENET_E_FWRESP_PAUSE:
- if (enet->flags & BNA_ENET_F_PAUSE_CHANGED) {
- enet->flags &= ~BNA_ENET_F_PAUSE_CHANGED;
- bna_bfi_pause_set(enet);
- } else if (enet->flags & BNA_ENET_F_MTU_CHANGED) {
- enet->flags &= ~BNA_ENET_F_MTU_CHANGED;
- bna_enet_rx_stop(enet);
- } else {
- bfa_fsm_set_state(enet, bna_enet_sm_started);
- }
- break;
- default:
- bfa_sm_fault(event);
- }
- }
- static void
- bna_enet_sm_cfg_stop_wait_entry(struct bna_enet *enet)
- {
- enet->flags &= ~BNA_ENET_F_PAUSE_CHANGED;
- enet->flags &= ~BNA_ENET_F_MTU_CHANGED;
- }
- static void
- bna_enet_sm_cfg_stop_wait(struct bna_enet *enet,
- enum bna_enet_event event)
- {
- switch (event) {
- case ENET_E_FAIL:
- bfa_fsm_set_state(enet, bna_enet_sm_stopped);
- bna_enet_chld_fail(enet);
- break;
- case ENET_E_FWRESP_PAUSE:
- case ENET_E_CHLD_STOPPED:
- bfa_fsm_set_state(enet, bna_enet_sm_chld_stop_wait);
- break;
- default:
- bfa_sm_fault(event);
- }
- }
- static void
- bna_enet_sm_chld_stop_wait_entry(struct bna_enet *enet)
- {
- bna_enet_chld_stop(enet);
- }
- static void
- bna_enet_sm_chld_stop_wait(struct bna_enet *enet,
- enum bna_enet_event event)
- {
- switch (event) {
- case ENET_E_FAIL:
- bfa_fsm_set_state(enet, bna_enet_sm_stopped);
- bna_enet_chld_fail(enet);
- break;
- case ENET_E_CHLD_STOPPED:
- bfa_fsm_set_state(enet, bna_enet_sm_stopped);
- break;
- default:
- bfa_sm_fault(event);
- }
- }
- static void
- bna_bfi_pause_set(struct bna_enet *enet)
- {
- struct bfi_enet_set_pause_req *pause_req = &enet->pause_req;
- bfi_msgq_mhdr_set(pause_req->mh, BFI_MC_ENET,
- BFI_ENET_H2I_SET_PAUSE_REQ, 0, 0);
- pause_req->mh.num_entries = htons(
- bfi_msgq_num_cmd_entries(sizeof(struct bfi_enet_set_pause_req)));
- pause_req->tx_pause = enet->pause_config.tx_pause;
- pause_req->rx_pause = enet->pause_config.rx_pause;
- bfa_msgq_cmd_set(&enet->msgq_cmd, NULL, NULL,
- sizeof(struct bfi_enet_set_pause_req), &pause_req->mh);
- bfa_msgq_cmd_post(&enet->bna->msgq, &enet->msgq_cmd);
- }
- static void
- bna_enet_cb_chld_stopped(void *arg)
- {
- struct bna_enet *enet = (struct bna_enet *)arg;
- bfa_fsm_send_event(enet, ENET_E_CHLD_STOPPED);
- }
- static void
- bna_enet_init(struct bna_enet *enet, struct bna *bna)
- {
- enet->bna = bna;
- enet->flags = 0;
- enet->mtu = 0;
- enet->type = BNA_ENET_T_REGULAR;
- enet->stop_cbfn = NULL;
- enet->stop_cbarg = NULL;
- enet->pause_cbfn = NULL;
- enet->mtu_cbfn = NULL;
- bfa_fsm_set_state(enet, bna_enet_sm_stopped);
- }
- static void
- bna_enet_uninit(struct bna_enet *enet)
- {
- enet->flags = 0;
- enet->bna = NULL;
- }
- static void
- bna_enet_start(struct bna_enet *enet)
- {
- enet->flags |= BNA_ENET_F_IOCETH_READY;
- if (enet->flags & BNA_ENET_F_ENABLED)
- bfa_fsm_send_event(enet, ENET_E_START);
- }
- static void
- bna_ioceth_cb_enet_stopped(void *arg)
- {
- struct bna_ioceth *ioceth = (struct bna_ioceth *)arg;
- bfa_fsm_send_event(ioceth, IOCETH_E_ENET_STOPPED);
- }
- static void
- bna_enet_stop(struct bna_enet *enet)
- {
- enet->stop_cbfn = bna_ioceth_cb_enet_stopped;
- enet->stop_cbarg = &enet->bna->ioceth;
- enet->flags &= ~BNA_ENET_F_IOCETH_READY;
- bfa_fsm_send_event(enet, ENET_E_STOP);
- }
- static void
- bna_enet_fail(struct bna_enet *enet)
- {
- enet->flags &= ~BNA_ENET_F_IOCETH_READY;
- bfa_fsm_send_event(enet, ENET_E_FAIL);
- }
- void
- bna_enet_cb_tx_stopped(struct bna_enet *enet)
- {
- bfa_wc_down(&enet->chld_stop_wc);
- }
- void
- bna_enet_cb_rx_stopped(struct bna_enet *enet)
- {
- bfa_wc_down(&enet->chld_stop_wc);
- }
- int
- bna_enet_mtu_get(struct bna_enet *enet)
- {
- return enet->mtu;
- }
- void
- bna_enet_enable(struct bna_enet *enet)
- {
- if (enet->fsm != (bfa_sm_t)bna_enet_sm_stopped)
- return;
- enet->flags |= BNA_ENET_F_ENABLED;
- if (enet->flags & BNA_ENET_F_IOCETH_READY)
- bfa_fsm_send_event(enet, ENET_E_START);
- }
- void
- bna_enet_disable(struct bna_enet *enet, enum bna_cleanup_type type,
- void (*cbfn)(void *))
- {
- if (type == BNA_SOFT_CLEANUP) {
- (*cbfn)(enet->bna->bnad);
- return;
- }
- enet->stop_cbfn = cbfn;
- enet->stop_cbarg = enet->bna->bnad;
- enet->flags &= ~BNA_ENET_F_ENABLED;
- bfa_fsm_send_event(enet, ENET_E_STOP);
- }
- void
- bna_enet_pause_config(struct bna_enet *enet,
- struct bna_pause_config *pause_config,
- void (*cbfn)(struct bnad *))
- {
- enet->pause_config = *pause_config;
- enet->pause_cbfn = cbfn;
- bfa_fsm_send_event(enet, ENET_E_PAUSE_CFG);
- }
- void
- bna_enet_mtu_set(struct bna_enet *enet, int mtu,
- void (*cbfn)(struct bnad *))
- {
- enet->mtu = mtu;
- enet->mtu_cbfn = cbfn;
- bfa_fsm_send_event(enet, ENET_E_MTU_CFG);
- }
- void
- bna_enet_perm_mac_get(struct bna_enet *enet, mac_t *mac)
- {
- *mac = bfa_nw_ioc_get_mac(&enet->bna->ioceth.ioc);
- }
- /**
- * IOCETH
- */
- #define enable_mbox_intr(_ioceth) \
- do { \
- u32 intr_status; \
- bna_intr_status_get((_ioceth)->bna, intr_status); \
- bnad_cb_mbox_intr_enable((_ioceth)->bna->bnad); \
- bna_mbox_intr_enable((_ioceth)->bna); \
- } while (0)
- #define disable_mbox_intr(_ioceth) \
- do { \
- bna_mbox_intr_disable((_ioceth)->bna); \
- bnad_cb_mbox_intr_disable((_ioceth)->bna->bnad); \
- } while (0)
- #define call_ioceth_stop_cbfn(_ioceth) \
- do { \
- if ((_ioceth)->stop_cbfn) { \
- void (*cbfn)(struct bnad *); \
- struct bnad *cbarg; \
- cbfn = (_ioceth)->stop_cbfn; \
- cbarg = (_ioceth)->stop_cbarg; \
- (_ioceth)->stop_cbfn = NULL; \
- (_ioceth)->stop_cbarg = NULL; \
- cbfn(cbarg); \
- } \
- } while (0)
- #define bna_stats_mod_uninit(_stats_mod) \
- do { \
- } while (0)
- #define bna_stats_mod_start(_stats_mod) \
- do { \
- (_stats_mod)->ioc_ready = true; \
- } while (0)
- #define bna_stats_mod_stop(_stats_mod) \
- do { \
- (_stats_mod)->ioc_ready = false; \
- } while (0)
- #define bna_stats_mod_fail(_stats_mod) \
- do { \
- (_stats_mod)->ioc_ready = false; \
- (_stats_mod)->stats_get_busy = false; \
- (_stats_mod)->stats_clr_busy = false; \
- } while (0)
- static void bna_bfi_attr_get(struct bna_ioceth *ioceth);
- bfa_fsm_state_decl(bna_ioceth, stopped, struct bna_ioceth,
- enum bna_ioceth_event);
- bfa_fsm_state_decl(bna_ioceth, ioc_ready_wait, struct bna_ioceth,
- enum bna_ioceth_event);
- bfa_fsm_state_decl(bna_ioceth, enet_attr_wait, struct bna_ioceth,
- enum bna_ioceth_event);
- bfa_fsm_state_decl(bna_ioceth, ready, struct bna_ioceth,
- enum bna_ioceth_event);
- bfa_fsm_state_decl(bna_ioceth, last_resp_wait, struct bna_ioceth,
- enum bna_ioceth_event);
- bfa_fsm_state_decl(bna_ioceth, enet_stop_wait, struct bna_ioceth,
- enum bna_ioceth_event);
- bfa_fsm_state_decl(bna_ioceth, ioc_disable_wait, struct bna_ioceth,
- enum bna_ioceth_event);
- bfa_fsm_state_decl(bna_ioceth, failed, struct bna_ioceth,
- enum bna_ioceth_event);
- static void
- bna_ioceth_sm_stopped_entry(struct bna_ioceth *ioceth)
- {
- call_ioceth_stop_cbfn(ioceth);
- }
- static void
- bna_ioceth_sm_stopped(struct bna_ioceth *ioceth,
- enum bna_ioceth_event event)
- {
- switch (event) {
- case IOCETH_E_ENABLE:
- bfa_fsm_set_state(ioceth, bna_ioceth_sm_ioc_ready_wait);
- bfa_nw_ioc_enable(&ioceth->ioc);
- break;
- case IOCETH_E_DISABLE:
- bfa_fsm_set_state(ioceth, bna_ioceth_sm_stopped);
- break;
- case IOCETH_E_IOC_RESET:
- enable_mbox_intr(ioceth);
- break;
- case IOCETH_E_IOC_FAILED:
- disable_mbox_intr(ioceth);
- bfa_fsm_set_state(ioceth, bna_ioceth_sm_failed);
- break;
- default:
- bfa_sm_fault(event);
- }
- }
- static void
- bna_ioceth_sm_ioc_ready_wait_entry(struct bna_ioceth *ioceth)
- {
- /**
- * Do not call bfa_nw_ioc_enable() here. It must be called in the
- * previous state due to failed -> ioc_ready_wait transition.
- */
- }
- static void
- bna_ioceth_sm_ioc_ready_wait(struct bna_ioceth *ioceth,
- enum bna_ioceth_event event)
- {
- switch (event) {
- case IOCETH_E_DISABLE:
- bfa_fsm_set_state(ioceth, bna_ioceth_sm_ioc_disable_wait);
- bfa_nw_ioc_disable(&ioceth->ioc);
- break;
- case IOCETH_E_IOC_RESET:
- enable_mbox_intr(ioceth);
- break;
- case IOCETH_E_IOC_FAILED:
- disable_mbox_intr(ioceth);
- bfa_fsm_set_state(ioceth, bna_ioceth_sm_failed);
- break;
- case IOCETH_E_IOC_READY:
- bfa_fsm_set_state(ioceth, bna_ioceth_sm_enet_attr_wait);
- break;
- default:
- bfa_sm_fault(event);
- }
- }
- static void
- bna_ioceth_sm_enet_attr_wait_entry(struct bna_ioceth *ioceth)
- {
- bna_bfi_attr_get(ioceth);
- }
- static void
- bna_ioceth_sm_enet_attr_wait(struct bna_ioceth *ioceth,
- enum bna_ioceth_event event)
- {
- switch (event) {
- case IOCETH_E_DISABLE:
- bfa_fsm_set_state(ioceth, bna_ioceth_sm_last_resp_wait);
- break;
- case IOCETH_E_IOC_FAILED:
- disable_mbox_intr(ioceth);
- bfa_fsm_set_state(ioceth, bna_ioceth_sm_failed);
- break;
- case IOCETH_E_ENET_ATTR_RESP:
- bfa_fsm_set_state(ioceth, bna_ioceth_sm_ready);
- break;
- default:
- bfa_sm_fault(event);
- }
- }
- static void
- bna_ioceth_sm_ready_entry(struct bna_ioceth *ioceth)
- {
- bna_enet_start(&ioceth->bna->enet);
- bna_stats_mod_start(&ioceth->bna->stats_mod);
- bnad_cb_ioceth_ready(ioceth->bna->bnad);
- }
- static void
- bna_ioceth_sm_ready(struct bna_ioceth *ioceth, enum bna_ioceth_event event)
- {
- switch (event) {
- case IOCETH_E_DISABLE:
- bfa_fsm_set_state(ioceth, bna_ioceth_sm_enet_stop_wait);
- break;
- case IOCETH_E_IOC_FAILED:
- disable_mbox_intr(ioceth);
- bna_enet_fail(&ioceth->bna->enet);
- bna_stats_mod_fail(&ioceth->bna->stats_mod);
- bfa_fsm_set_state(ioceth, bna_ioceth_sm_failed);
- break;
- default:
- bfa_sm_fault(event);
- }
- }
- static void
- bna_ioceth_sm_last_resp_wait_entry(struct bna_ioceth *ioceth)
- {
- }
- static void
- bna_ioceth_sm_last_resp_wait(struct bna_ioceth *ioceth,
- enum bna_ioceth_event event)
- {
- switch (event) {
- case IOCETH_E_IOC_FAILED:
- bfa_fsm_set_state(ioceth, bna_ioceth_sm_ioc_disable_wait);
- disable_mbox_intr(ioceth);
- bfa_nw_ioc_disable(&ioceth->ioc);
- break;
- case IOCETH_E_ENET_ATTR_RESP:
- bfa_fsm_set_state(ioceth, bna_ioceth_sm_ioc_disable_wait);
- bfa_nw_ioc_disable(&ioceth->ioc);
- break;
- default:
- bfa_sm_fault(event);
- }
- }
- static void
- bna_ioceth_sm_enet_stop_wait_entry(struct bna_ioceth *ioceth)
- {
- bna_stats_mod_stop(&ioceth->bna->stats_mod);
- bna_enet_stop(&ioceth->bna->enet);
- }
- static void
- bna_ioceth_sm_enet_stop_wait(struct bna_ioceth *ioceth,
- enum bna_ioceth_event event)
- {
- switch (event) {
- case IOCETH_E_IOC_FAILED:
- bfa_fsm_set_state(ioceth, bna_ioceth_sm_ioc_disable_wait);
- disable_mbox_intr(ioceth);
- bna_enet_fail(&ioceth->bna->enet);
- bna_stats_mod_fail(&ioceth->bna->stats_mod);
- bfa_nw_ioc_disable(&ioceth->ioc);
- break;
- case IOCETH_E_ENET_STOPPED:
- bfa_fsm_set_state(ioceth, bna_ioceth_sm_ioc_disable_wait);
- bfa_nw_ioc_disable(&ioceth->ioc);
- break;
- default:
- bfa_sm_fault(event);
- }
- }
- static void
- bna_ioceth_sm_ioc_disable_wait_entry(struct bna_ioceth *ioceth)
- {
- }
- static void
- bna_ioceth_sm_ioc_disable_wait(struct bna_ioceth *ioceth,
- enum bna_ioceth_event event)
- {
- switch (event) {
- case IOCETH_E_IOC_DISABLED:
- disable_mbox_intr(ioceth);
- bfa_fsm_set_state(ioceth, bna_ioceth_sm_stopped);
- break;
- case IOCETH_E_ENET_STOPPED:
- /* This event is received due to enet failing */
- /* No-op */
- break;
- default:
- bfa_sm_fault(event);
- }
- }
- static void
- bna_ioceth_sm_failed_entry(struct bna_ioceth *ioceth)
- {
- bnad_cb_ioceth_failed(ioceth->bna->bnad);
- }
- static void
- bna_ioceth_sm_failed(struct bna_ioceth *ioceth,
- enum bna_ioceth_event event)
- {
- switch (event) {
- case IOCETH_E_DISABLE:
- bfa_fsm_set_state(ioceth, bna_ioceth_sm_ioc_disable_wait);
- bfa_nw_ioc_disable(&ioceth->ioc);
- break;
- case IOCETH_E_IOC_RESET:
- enable_mbox_intr(ioceth);
- bfa_fsm_set_state(ioceth, bna_ioceth_sm_ioc_ready_wait);
- break;
- case IOCETH_E_IOC_FAILED:
- break;
- default:
- bfa_sm_fault(event);
- }
- }
- static void
- bna_bfi_attr_get(struct bna_ioceth *ioceth)
- {
- struct bfi_enet_attr_req *attr_req = &ioceth->attr_req;
- bfi_msgq_mhdr_set(attr_req->mh, BFI_MC_ENET,
- BFI_ENET_H2I_GET_ATTR_REQ, 0, 0);
- attr_req->mh.num_entries = htons(
- bfi_msgq_num_cmd_entries(sizeof(struct bfi_enet_attr_req)));
- bfa_msgq_cmd_set(&ioceth->msgq_cmd, NULL, NULL,
- sizeof(struct bfi_enet_attr_req), &attr_req->mh);
- bfa_msgq_cmd_post(&ioceth->bna->msgq, &ioceth->msgq_cmd);
- }
- /* IOC callback functions */
- static void
- bna_cb_ioceth_enable(void *arg, enum bfa_status error)
- {
- struct bna_ioceth *ioceth = (struct bna_ioceth *)arg;
- if (error)
- bfa_fsm_send_event(ioceth, IOCETH_E_IOC_FAILED);
- else
- bfa_fsm_send_event(ioceth, IOCETH_E_IOC_READY);
- }
- static void
- bna_cb_ioceth_disable(void *arg)
- {
- struct bna_ioceth *ioceth = (struct bna_ioceth *)arg;
- bfa_fsm_send_event(ioceth, IOCETH_E_IOC_DISABLED);
- }
- static void
- bna_cb_ioceth_hbfail(void *arg)
- {
- struct bna_ioceth *ioceth = (struct bna_ioceth *)arg;
- bfa_fsm_send_event(ioceth, IOCETH_E_IOC_FAILED);
- }
- static void
- bna_cb_ioceth_reset(void *arg)
- {
- struct bna_ioceth *ioceth = (struct bna_ioceth *)arg;
- bfa_fsm_send_event(ioceth, IOCETH_E_IOC_RESET);
- }
- static struct bfa_ioc_cbfn bna_ioceth_cbfn = {
- bna_cb_ioceth_enable,
- bna_cb_ioceth_disable,
- bna_cb_ioceth_hbfail,
- bna_cb_ioceth_reset
- };
- static void bna_attr_init(struct bna_ioceth *ioceth)
- {
- ioceth->attr.num_txq = BFI_ENET_DEF_TXQ;
- ioceth->attr.num_rxp = BFI_ENET_DEF_RXP;
- ioceth->attr.num_ucmac = BFI_ENET_DEF_UCAM;
- ioceth->attr.num_mcmac = BFI_ENET_MAX_MCAM;
- ioceth->attr.max_rit_size = BFI_ENET_DEF_RITSZ;
- ioceth->attr.fw_query_complete = false;
- }
- static void
- bna_ioceth_init(struct bna_ioceth *ioceth, struct bna *bna,
- struct bna_res_info *res_info)
- {
- u64 dma;
- u8 *kva;
- ioceth->bna = bna;
- /**
- * Attach IOC and claim:
- * 1. DMA memory for IOC attributes
- * 2. Kernel memory for FW trace
- */
- bfa_nw_ioc_attach(&ioceth->ioc, ioceth, &bna_ioceth_cbfn);
- bfa_nw_ioc_pci_init(&ioceth->ioc, &bna->pcidev, BFI_PCIFN_CLASS_ETH);
- BNA_GET_DMA_ADDR(
- &res_info[BNA_RES_MEM_T_ATTR].res_u.mem_info.mdl[0].dma, dma);
- kva = res_info[BNA_RES_MEM_T_ATTR].res_u.mem_info.mdl[0].kva;
- bfa_nw_ioc_mem_claim(&ioceth->ioc, kva, dma);
- kva = res_info[BNA_RES_MEM_T_FWTRC].res_u.mem_info.mdl[0].kva;
- bfa_nw_ioc_debug_memclaim(&ioceth->ioc, kva);
- /**
- * Attach common modules (Diag, SFP, CEE, Port) and claim respective
- * DMA memory.
- */
- BNA_GET_DMA_ADDR(
- &res_info[BNA_RES_MEM_T_COM].res_u.mem_info.mdl[0].dma, dma);
- kva = res_info[BNA_RES_MEM_T_COM].res_u.mem_info.mdl[0].kva;
- bfa_nw_cee_attach(&bna->cee, &ioceth->ioc, bna);
- bfa_nw_cee_mem_claim(&bna->cee, kva, dma);
- kva += bfa_nw_cee_meminfo();
- dma += bfa_nw_cee_meminfo();
- bfa_nw_flash_attach(&bna->flash, &ioceth->ioc, bna);
- bfa_nw_flash_memclaim(&bna->flash, kva, dma);
- kva += bfa_nw_flash_meminfo();
- dma += bfa_nw_flash_meminfo();
- bfa_msgq_attach(&bna->msgq, &ioceth->ioc);
- bfa_msgq_memclaim(&bna->msgq, kva, dma);
- bfa_msgq_regisr(&bna->msgq, BFI_MC_ENET, bna_msgq_rsp_handler, bna);
- kva += bfa_msgq_meminfo();
- dma += bfa_msgq_meminfo();
- ioceth->stop_cbfn = NULL;
- ioceth->stop_cbarg = NULL;
- bna_attr_init(ioceth);
- bfa_fsm_set_state(ioceth, bna_ioceth_sm_stopped);
- }
- static void
- bna_ioceth_uninit(struct bna_ioceth *ioceth)
- {
- bfa_nw_ioc_detach(&ioceth->ioc);
- ioceth->bna = NULL;
- }
- void
- bna_ioceth_enable(struct bna_ioceth *ioceth)
- {
- if (ioceth->fsm == (bfa_fsm_t)bna_ioceth_sm_ready) {
- bnad_cb_ioceth_ready(ioceth->bna->bnad);
- return;
- }
- if (ioceth->fsm == (bfa_fsm_t)bna_ioceth_sm_stopped)
- bfa_fsm_send_event(ioceth, IOCETH_E_ENABLE);
- }
- void
- bna_ioceth_disable(struct bna_ioceth *ioceth, enum bna_cleanup_type type)
- {
- if (type == BNA_SOFT_CLEANUP) {
- bnad_cb_ioceth_disabled(ioceth->bna->bnad);
- return;
- }
- ioceth->stop_cbfn = bnad_cb_ioceth_disabled;
- ioceth->stop_cbarg = ioceth->bna->bnad;
- bfa_fsm_send_event(ioceth, IOCETH_E_DISABLE);
- }
- static void
- bna_ucam_mod_init(struct bna_ucam_mod *ucam_mod, struct bna *bna,
- struct bna_res_info *res_info)
- {
- int i;
- ucam_mod->ucmac = (struct bna_mac *)
- res_info[BNA_MOD_RES_MEM_T_UCMAC_ARRAY].res_u.mem_info.mdl[0].kva;
- INIT_LIST_HEAD(&ucam_mod->free_q);
- for (i = 0; i < bna->ioceth.attr.num_ucmac; i++) {
- bfa_q_qe_init(&ucam_mod->ucmac[i].qe);
- list_add_tail(&ucam_mod->ucmac[i].qe, &ucam_mod->free_q);
- }
- ucam_mod->bna = bna;
- }
- static void
- bna_ucam_mod_uninit(struct bna_ucam_mod *ucam_mod)
- {
- struct list_head *qe;
- int i = 0;
- list_for_each(qe, &ucam_mod->free_q)
- i++;
- ucam_mod->bna = NULL;
- }
- static void
- bna_mcam_mod_init(struct bna_mcam_mod *mcam_mod, struct bna *bna,
- struct bna_res_info *res_info)
- {
- int i;
- mcam_mod->mcmac = (struct bna_mac *)
- res_info[BNA_MOD_RES_MEM_T_MCMAC_ARRAY].res_u.mem_info.mdl[0].kva;
- INIT_LIST_HEAD(&mcam_mod->free_q);
- for (i = 0; i < bna->ioceth.attr.num_mcmac; i++) {
- bfa_q_qe_init(&mcam_mod->mcmac[i].qe);
- list_add_tail(&mcam_mod->mcmac[i].qe, &mcam_mod->free_q);
- }
- mcam_mod->mchandle = (struct bna_mcam_handle *)
- res_info[BNA_MOD_RES_MEM_T_MCHANDLE_ARRAY].res_u.mem_info.mdl[0].kva;
- INIT_LIST_HEAD(&mcam_mod->free_handle_q);
- for (i = 0; i < bna->ioceth.attr.num_mcmac; i++) {
- bfa_q_qe_init(&mcam_mod->mchandle[i].qe);
- list_add_tail(&mcam_mod->mchandle[i].qe,
- &mcam_mod->free_handle_q);
- }
- mcam_mod->bna = bna;
- }
- static void
- bna_mcam_mod_uninit(struct bna_mcam_mod *mcam_mod)
- {
- struct list_head *qe;
- int i;
- i = 0;
- list_for_each(qe, &mcam_mod->free_q) i++;
- i = 0;
- list_for_each(qe, &mcam_mod->free_handle_q) i++;
- mcam_mod->bna = NULL;
- }
- static void
- bna_bfi_stats_get(struct bna *bna)
- {
- struct bfi_enet_stats_req *stats_req = &bna->stats_mod.stats_get;
- bna->stats_mod.stats_get_busy = true;
- bfi_msgq_mhdr_set(stats_req->mh, BFI_MC_ENET,
- BFI_ENET_H2I_STATS_GET_REQ, 0, 0);
- stats_req->mh.num_entries = htons(
- bfi_msgq_num_cmd_entries(sizeof(struct bfi_enet_stats_req)));
- stats_req->stats_mask = htons(BFI_ENET_STATS_ALL);
- stats_req->tx_enet_mask = htonl(bna->tx_mod.rid_mask);
- stats_req->rx_enet_mask = htonl(bna->rx_mod.rid_mask);
- stats_req->host_buffer.a32.addr_hi = bna->stats.hw_stats_dma.msb;
- stats_req->host_buffer.a32.addr_lo = bna->stats.hw_stats_dma.lsb;
- bfa_msgq_cmd_set(&bna->stats_mod.stats_get_cmd, NULL, NULL,
- sizeof(struct bfi_enet_stats_req), &stats_req->mh);
- bfa_msgq_cmd_post(&bna->msgq, &bna->stats_mod.stats_get_cmd);
- }
- void
- bna_res_req(struct bna_res_info *res_info)
- {
- /* DMA memory for COMMON_MODULE */
- res_info[BNA_RES_MEM_T_COM].res_type = BNA_RES_T_MEM;
- res_info[BNA_RES_MEM_T_COM].res_u.mem_info.mem_type = BNA_MEM_T_DMA;
- res_info[BNA_RES_MEM_T_COM].res_u.mem_info.num = 1;
- res_info[BNA_RES_MEM_T_COM].res_u.mem_info.len = ALIGN(
- (bfa_nw_cee_meminfo() +
- bfa_nw_flash_meminfo() +
- bfa_msgq_meminfo()), PAGE_SIZE);
- /* DMA memory for retrieving IOC attributes */
- res_info[BNA_RES_MEM_T_ATTR].res_type = BNA_RES_T_MEM;
- res_info[BNA_RES_MEM_T_ATTR].res_u.mem_info.mem_type = BNA_MEM_T_DMA;
- res_info[BNA_RES_MEM_T_ATTR].res_u.mem_info.num = 1;
- res_info[BNA_RES_MEM_T_ATTR].res_u.mem_info.len =
- ALIGN(bfa_nw_ioc_meminfo(), PAGE_SIZE);
- /* Virtual memory for retreiving fw_trc */
- res_info[BNA_RES_MEM_T_FWTRC].res_type = BNA_RES_T_MEM;
- res_info[BNA_RES_MEM_T_FWTRC].res_u.mem_info.mem_type = BNA_MEM_T_KVA;
- res_info[BNA_RES_MEM_T_FWTRC].res_u.mem_info.num = 1;
- res_info[BNA_RES_MEM_T_FWTRC].res_u.mem_info.len = BNA_DBG_FWTRC_LEN;
- /* DMA memory for retreiving stats */
- res_info[BNA_RES_MEM_T_STATS].res_type = BNA_RES_T_MEM;
- res_info[BNA_RES_MEM_T_STATS].res_u.mem_info.mem_type = BNA_MEM_T_DMA;
- res_info[BNA_RES_MEM_T_STATS].res_u.mem_info.num = 1;
- res_info[BNA_RES_MEM_T_STATS].res_u.mem_info.len =
- ALIGN(sizeof(struct bfi_enet_stats),
- PAGE_SIZE);
- }
- void
- bna_mod_res_req(struct bna *bna, struct bna_res_info *res_info)
- {
- struct bna_attr *attr = &bna->ioceth.attr;
- /* Virtual memory for Tx objects - stored by Tx module */
- res_info[BNA_MOD_RES_MEM_T_TX_ARRAY].res_type = BNA_RES_T_MEM;
- res_info[BNA_MOD_RES_MEM_T_TX_ARRAY].res_u.mem_info.mem_type =
- BNA_MEM_T_KVA;
- res_info[BNA_MOD_RES_MEM_T_TX_ARRAY].res_u.mem_info.num = 1;
- res_info[BNA_MOD_RES_MEM_T_TX_ARRAY].res_u.mem_info.len =
- attr->num_txq * sizeof(struct bna_tx);
- /* Virtual memory for TxQ - stored by Tx module */
- res_info[BNA_MOD_RES_MEM_T_TXQ_ARRAY].res_type = BNA_RES_T_MEM;
- res_info[BNA_MOD_RES_MEM_T_TXQ_ARRAY].res_u.mem_info.mem_type =
- BNA_MEM_T_KVA;
- res_info[BNA_MOD_RES_MEM_T_TXQ_ARRAY].res_u.mem_info.num = 1;
- res_info[BNA_MOD_RES_MEM_T_TXQ_ARRAY].res_u.mem_info.len =
- attr->num_txq * sizeof(struct bna_txq);
- /* Virtual memory for Rx objects - stored by Rx module */
- res_info[BNA_MOD_RES_MEM_T_RX_ARRAY].res_type = BNA_RES_T_MEM;
- res_info[BNA_MOD_RES_MEM_T_RX_ARRAY].res_u.mem_info.mem_type =
- BNA_MEM_T_KVA;
- res_info[BNA_MOD_RES_MEM_T_RX_ARRAY].res_u.mem_info.num = 1;
- res_info[BNA_MOD_RES_MEM_T_RX_ARRAY].res_u.mem_info.len =
- attr->num_rxp * sizeof(struct bna_rx);
- /* Virtual memory for RxPath - stored by Rx module */
- res_info[BNA_MOD_RES_MEM_T_RXP_ARRAY].res_type = BNA_RES_T_MEM;
- res_info[BNA_MOD_RES_MEM_T_RXP_ARRAY].res_u.mem_info.mem_type =
- BNA_MEM_T_KVA;
- res_info[BNA_MOD_RES_MEM_T_RXP_ARRAY].res_u.mem_info.num = 1;
- res_info[BNA_MOD_RES_MEM_T_RXP_ARRAY].res_u.mem_info.len =
- attr->num_rxp * sizeof(struct bna_rxp);
- /* Virtual memory for RxQ - stored by Rx module */
- res_info[BNA_MOD_RES_MEM_T_RXQ_ARRAY].res_type = BNA_RES_T_MEM;
- res_info[BNA_MOD_RES_MEM_T_RXQ_ARRAY].res_u.mem_info.mem_type =
- BNA_MEM_T_KVA;
- res_info[BNA_MOD_RES_MEM_T_RXQ_ARRAY].res_u.mem_info.num = 1;
- res_info[BNA_MOD_RES_MEM_T_RXQ_ARRAY].res_u.mem_info.len =
- (attr->num_rxp * 2) * sizeof(struct bna_rxq);
- /* Virtual memory for Unicast MAC address - stored by ucam module */
- res_info[BNA_MOD_RES_MEM_T_UCMAC_ARRAY].res_type = BNA_RES_T_MEM;
- res_info[BNA_MOD_RES_MEM_T_UCMAC_ARRAY].res_u.mem_info.mem_type =
- BNA_MEM_T_KVA;
- res_info[BNA_MOD_RES_MEM_T_UCMAC_ARRAY].res_u.mem_info.num = 1;
- res_info[BNA_MOD_RES_MEM_T_UCMAC_ARRAY].res_u.mem_info.len =
- attr->num_ucmac * sizeof(struct bna_mac);
- /* Virtual memory for Multicast MAC address - stored by mcam module */
- res_info[BNA_MOD_RES_MEM_T_MCMAC_ARRAY].res_type = BNA_RES_T_MEM;
- res_info[BNA_MOD_RES_MEM_T_MCMAC_ARRAY].res_u.mem_info.mem_type =
- BNA_MEM_T_KVA;
- res_info[BNA_MOD_RES_MEM_T_MCMAC_ARRAY].res_u.mem_info.num = 1;
- res_info[BNA_MOD_RES_MEM_T_MCMAC_ARRAY].res_u.mem_info.len =
- attr->num_mcmac * sizeof(struct bna_mac);
- /* Virtual memory for Multicast handle - stored by mcam module */
- res_info[BNA_MOD_RES_MEM_T_MCHANDLE_ARRAY].res_type = BNA_RES_T_MEM;
- res_info[BNA_MOD_RES_MEM_T_MCHANDLE_ARRAY].res_u.mem_info.mem_type =
- BNA_MEM_T_KVA;
- res_info[BNA_MOD_RES_MEM_T_MCHANDLE_ARRAY].res_u.mem_info.num = 1;
- res_info[BNA_MOD_RES_MEM_T_MCHANDLE_ARRAY].res_u.mem_info.len =
- attr->num_mcmac * sizeof(struct bna_mcam_handle);
- }
- void
- bna_init(struct bna *bna, struct bnad *bnad,
- struct bfa_pcidev *pcidev, struct bna_res_info *res_info)
- {
- bna->bnad = bnad;
- bna->pcidev = *pcidev;
- bna->stats.hw_stats_kva = (struct bfi_enet_stats *)
- res_info[BNA_RES_MEM_T_STATS].res_u.mem_info.mdl[0].kva;
- bna->stats.hw_stats_dma.msb =
- res_info[BNA_RES_MEM_T_STATS].res_u.mem_info.mdl[0].dma.msb;
- bna->stats.hw_stats_dma.lsb =
- res_info[BNA_RES_MEM_T_STATS].res_u.mem_info.mdl[0].dma.lsb;
- bna_reg_addr_init(bna, &bna->pcidev);
- /* Also initializes diag, cee, sfp, phy_port, msgq */
- bna_ioceth_init(&bna->ioceth, bna, res_info);
- bna_enet_init(&bna->enet, bna);
- bna_ethport_init(&bna->ethport, bna);
- }
- void
- bna_mod_init(struct bna *bna, struct bna_res_info *res_info)
- {
- bna_tx_mod_init(&bna->tx_mod, bna, res_info);
- bna_rx_mod_init(&bna->rx_mod, bna, res_info);
- bna_ucam_mod_init(&bna->ucam_mod, bna, res_info);
- bna_mcam_mod_init(&bna->mcam_mod, bna, res_info);
- bna->default_mode_rid = BFI_INVALID_RID;
- bna->promisc_rid = BFI_INVALID_RID;
- bna->mod_flags |= BNA_MOD_F_INIT_DONE;
- }
- void
- bna_uninit(struct bna *bna)
- {
- if (bna->mod_flags & BNA_MOD_F_INIT_DONE) {
- bna_mcam_mod_uninit(&bna->mcam_mod);
- bna_ucam_mod_uninit(&bna->ucam_mod);
- bna_rx_mod_uninit(&bna->rx_mod);
- bna_tx_mod_uninit(&bna->tx_mod);
- bna->mod_flags &= ~BNA_MOD_F_INIT_DONE;
- }
- bna_stats_mod_uninit(&bna->stats_mod);
- bna_ethport_uninit(&bna->ethport);
- bna_enet_uninit(&bna->enet);
- bna_ioceth_uninit(&bna->ioceth);
- bna->bnad = NULL;
- }
- int
- bna_num_txq_set(struct bna *bna, int num_txq)
- {
- if (bna->ioceth.attr.fw_query_complete &&
- (num_txq <= bna->ioceth.attr.num_txq)) {
- bna->ioceth.attr.num_txq = num_txq;
- return BNA_CB_SUCCESS;
- }
- return BNA_CB_FAIL;
- }
- int
- bna_num_rxp_set(struct bna *bna, int num_rxp)
- {
- if (bna->ioceth.attr.fw_query_complete &&
- (num_rxp <= bna->ioceth.attr.num_rxp)) {
- bna->ioceth.attr.num_rxp = num_rxp;
- return BNA_CB_SUCCESS;
- }
- return BNA_CB_FAIL;
- }
- struct bna_mac *
- bna_ucam_mod_mac_get(struct bna_ucam_mod *ucam_mod)
- {
- struct list_head *qe;
- if (list_empty(&ucam_mod->free_q))
- return NULL;
- bfa_q_deq(&ucam_mod->free_q, &qe);
- return (struct bna_mac *)qe;
- }
- void
- bna_ucam_mod_mac_put(struct bna_ucam_mod *ucam_mod, struct bna_mac *mac)
- {
- list_add_tail(&mac->qe, &ucam_mod->free_q);
- }
- struct bna_mac *
- bna_mcam_mod_mac_get(struct bna_mcam_mod *mcam_mod)
- {
- struct list_head *qe;
- if (list_empty(&mcam_mod->free_q))
- return NULL;
- bfa_q_deq(&mcam_mod->free_q, &qe);
- return (struct bna_mac *)qe;
- }
- void
- bna_mcam_mod_mac_put(struct bna_mcam_mod *mcam_mod, struct bna_mac *mac)
- {
- list_add_tail(&mac->qe, &mcam_mod->free_q);
- }
- struct bna_mcam_handle *
- bna_mcam_mod_handle_get(struct bna_mcam_mod *mcam_mod)
- {
- struct list_head *qe;
- if (list_empty(&mcam_mod->free_handle_q))
- return NULL;
- bfa_q_deq(&mcam_mod->free_handle_q, &qe);
- return (struct bna_mcam_handle *)qe;
- }
- void
- bna_mcam_mod_handle_put(struct bna_mcam_mod *mcam_mod,
- struct bna_mcam_handle *handle)
- {
- list_add_tail(&handle->qe, &mcam_mod->free_handle_q);
- }
- void
- bna_hw_stats_get(struct bna *bna)
- {
- if (!bna->stats_mod.ioc_ready) {
- bnad_cb_stats_get(bna->bnad, BNA_CB_FAIL, &bna->stats);
- return;
- }
- if (bna->stats_mod.stats_get_busy) {
- bnad_cb_stats_get(bna->bnad, BNA_CB_BUSY, &bna->stats);
- return;
- }
- bna_bfi_stats_get(bna);
- }
|