cxgb4_dcb.c 29 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149
  1. /*
  2. * Copyright (C) 2013-2014 Chelsio Communications. All rights reserved.
  3. *
  4. * Written by Anish Bhatt (anish@chelsio.com)
  5. * Casey Leedom (leedom@chelsio.com)
  6. *
  7. * This program is free software; you can redistribute it and/or modify it
  8. * under the terms and conditions of the GNU General Public License,
  9. * version 2, as published by the Free Software Foundation.
  10. *
  11. * This program is distributed in the hope it will be useful, but WITHOUT
  12. * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
  13. * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for
  14. * more details.
  15. *
  16. * The full GNU General Public License is included in this distribution in
  17. * the file called "COPYING".
  18. *
  19. */
  20. #include "cxgb4.h"
  21. /* DCBx version control
  22. */
  23. char *dcb_ver_array[] = {
  24. "Unknown",
  25. "DCBx-CIN",
  26. "DCBx-CEE 1.01",
  27. "DCBx-IEEE",
  28. "", "", "",
  29. "Auto Negotiated"
  30. };
  31. /* Initialize a port's Data Center Bridging state. Typically used after a
  32. * Link Down event.
  33. */
  34. void cxgb4_dcb_state_init(struct net_device *dev)
  35. {
  36. struct port_info *pi = netdev2pinfo(dev);
  37. struct port_dcb_info *dcb = &pi->dcb;
  38. int version_temp = dcb->dcb_version;
  39. memset(dcb, 0, sizeof(struct port_dcb_info));
  40. dcb->state = CXGB4_DCB_STATE_START;
  41. if (version_temp)
  42. dcb->dcb_version = version_temp;
  43. netdev_dbg(dev, "%s: Initializing DCB state for port[%d]\n",
  44. __func__, pi->port_id);
  45. }
  46. void cxgb4_dcb_version_init(struct net_device *dev)
  47. {
  48. struct port_info *pi = netdev2pinfo(dev);
  49. struct port_dcb_info *dcb = &pi->dcb;
  50. /* Any writes here are only done on kernels that exlicitly need
  51. * a specific version, say < 2.6.38 which only support CEE
  52. */
  53. dcb->dcb_version = FW_PORT_DCB_VER_AUTO;
  54. }
  55. static void cxgb4_dcb_cleanup_apps(struct net_device *dev)
  56. {
  57. struct port_info *pi = netdev2pinfo(dev);
  58. struct adapter *adap = pi->adapter;
  59. struct port_dcb_info *dcb = &pi->dcb;
  60. struct dcb_app app;
  61. int i, err;
  62. /* zero priority implies remove */
  63. app.priority = 0;
  64. for (i = 0; i < CXGB4_MAX_DCBX_APP_SUPPORTED; i++) {
  65. /* Check if app list is exhausted */
  66. if (!dcb->app_priority[i].protocolid)
  67. break;
  68. app.protocol = dcb->app_priority[i].protocolid;
  69. if (dcb->dcb_version == FW_PORT_DCB_VER_IEEE) {
  70. app.priority = dcb->app_priority[i].user_prio_map;
  71. app.selector = dcb->app_priority[i].sel_field + 1;
  72. err = dcb_ieee_delapp(dev, &app);
  73. } else {
  74. app.selector = !!(dcb->app_priority[i].sel_field);
  75. err = dcb_setapp(dev, &app);
  76. }
  77. if (err) {
  78. dev_err(adap->pdev_dev,
  79. "Failed DCB Clear %s Application Priority: sel=%d, prot=%d, , err=%d\n",
  80. dcb_ver_array[dcb->dcb_version], app.selector,
  81. app.protocol, -err);
  82. break;
  83. }
  84. }
  85. }
  86. /* Finite State machine for Data Center Bridging.
  87. */
  88. void cxgb4_dcb_state_fsm(struct net_device *dev,
  89. enum cxgb4_dcb_state_input transition_to)
  90. {
  91. struct port_info *pi = netdev2pinfo(dev);
  92. struct port_dcb_info *dcb = &pi->dcb;
  93. struct adapter *adap = pi->adapter;
  94. enum cxgb4_dcb_state current_state = dcb->state;
  95. netdev_dbg(dev, "%s: State change from %d to %d for %s\n",
  96. __func__, dcb->state, transition_to, dev->name);
  97. switch (current_state) {
  98. case CXGB4_DCB_STATE_START: {
  99. switch (transition_to) {
  100. case CXGB4_DCB_INPUT_FW_DISABLED: {
  101. /* we're going to use Host DCB */
  102. dcb->state = CXGB4_DCB_STATE_HOST;
  103. dcb->supported = CXGB4_DCBX_HOST_SUPPORT;
  104. break;
  105. }
  106. case CXGB4_DCB_INPUT_FW_ENABLED: {
  107. /* we're going to use Firmware DCB */
  108. dcb->state = CXGB4_DCB_STATE_FW_INCOMPLETE;
  109. dcb->supported = DCB_CAP_DCBX_LLD_MANAGED;
  110. if (dcb->dcb_version == FW_PORT_DCB_VER_IEEE)
  111. dcb->supported |= DCB_CAP_DCBX_VER_IEEE;
  112. else
  113. dcb->supported |= DCB_CAP_DCBX_VER_CEE;
  114. break;
  115. }
  116. case CXGB4_DCB_INPUT_FW_INCOMPLETE: {
  117. /* expected transition */
  118. break;
  119. }
  120. case CXGB4_DCB_INPUT_FW_ALLSYNCED: {
  121. dcb->state = CXGB4_DCB_STATE_FW_ALLSYNCED;
  122. break;
  123. }
  124. default:
  125. goto bad_state_input;
  126. }
  127. break;
  128. }
  129. case CXGB4_DCB_STATE_FW_INCOMPLETE: {
  130. switch (transition_to) {
  131. case CXGB4_DCB_INPUT_FW_ENABLED: {
  132. /* we're alreaady in firmware DCB mode */
  133. break;
  134. }
  135. case CXGB4_DCB_INPUT_FW_INCOMPLETE: {
  136. /* we're already incomplete */
  137. break;
  138. }
  139. case CXGB4_DCB_INPUT_FW_ALLSYNCED: {
  140. dcb->state = CXGB4_DCB_STATE_FW_ALLSYNCED;
  141. dcb->enabled = 1;
  142. linkwatch_fire_event(dev);
  143. break;
  144. }
  145. default:
  146. goto bad_state_input;
  147. }
  148. break;
  149. }
  150. case CXGB4_DCB_STATE_FW_ALLSYNCED: {
  151. switch (transition_to) {
  152. case CXGB4_DCB_INPUT_FW_ENABLED: {
  153. /* we're alreaady in firmware DCB mode */
  154. break;
  155. }
  156. case CXGB4_DCB_INPUT_FW_INCOMPLETE: {
  157. /* We were successfully running with firmware DCB but
  158. * now it's telling us that it's in an "incomplete
  159. * state. We need to reset back to a ground state
  160. * of incomplete.
  161. */
  162. cxgb4_dcb_cleanup_apps(dev);
  163. cxgb4_dcb_state_init(dev);
  164. dcb->state = CXGB4_DCB_STATE_FW_INCOMPLETE;
  165. dcb->supported = CXGB4_DCBX_FW_SUPPORT;
  166. linkwatch_fire_event(dev);
  167. break;
  168. }
  169. case CXGB4_DCB_INPUT_FW_ALLSYNCED: {
  170. /* we're already all sync'ed
  171. * this is only applicable for IEEE or
  172. * when another VI already completed negotiaton
  173. */
  174. dcb->enabled = 1;
  175. linkwatch_fire_event(dev);
  176. break;
  177. }
  178. default:
  179. goto bad_state_input;
  180. }
  181. break;
  182. }
  183. case CXGB4_DCB_STATE_HOST: {
  184. switch (transition_to) {
  185. case CXGB4_DCB_INPUT_FW_DISABLED: {
  186. /* we're alreaady in Host DCB mode */
  187. break;
  188. }
  189. default:
  190. goto bad_state_input;
  191. }
  192. break;
  193. }
  194. default:
  195. goto bad_state_transition;
  196. }
  197. return;
  198. bad_state_input:
  199. dev_err(adap->pdev_dev, "cxgb4_dcb_state_fsm: illegal input symbol %d\n",
  200. transition_to);
  201. return;
  202. bad_state_transition:
  203. dev_err(adap->pdev_dev, "cxgb4_dcb_state_fsm: bad state transition, state = %d, input = %d\n",
  204. current_state, transition_to);
  205. }
  206. /* Handle a DCB/DCBX update message from the firmware.
  207. */
  208. void cxgb4_dcb_handle_fw_update(struct adapter *adap,
  209. const struct fw_port_cmd *pcmd)
  210. {
  211. const union fw_port_dcb *fwdcb = &pcmd->u.dcb;
  212. int port = FW_PORT_CMD_PORTID_GET(be32_to_cpu(pcmd->op_to_portid));
  213. struct net_device *dev = adap->port[port];
  214. struct port_info *pi = netdev_priv(dev);
  215. struct port_dcb_info *dcb = &pi->dcb;
  216. int dcb_type = pcmd->u.dcb.pgid.type;
  217. int dcb_running_version;
  218. /* Handle Firmware DCB Control messages separately since they drive
  219. * our state machine.
  220. */
  221. if (dcb_type == FW_PORT_DCB_TYPE_CONTROL) {
  222. enum cxgb4_dcb_state_input input =
  223. ((pcmd->u.dcb.control.all_syncd_pkd &
  224. FW_PORT_CMD_ALL_SYNCD)
  225. ? CXGB4_DCB_STATE_FW_ALLSYNCED
  226. : CXGB4_DCB_STATE_FW_INCOMPLETE);
  227. if (dcb->dcb_version != FW_PORT_DCB_VER_UNKNOWN) {
  228. dcb_running_version = FW_PORT_CMD_DCB_VERSION_GET(
  229. be16_to_cpu(
  230. pcmd->u.dcb.control.dcb_version_to_app_state));
  231. if (dcb_running_version == FW_PORT_DCB_VER_CEE1D01 ||
  232. dcb_running_version == FW_PORT_DCB_VER_IEEE) {
  233. dcb->dcb_version = dcb_running_version;
  234. dev_warn(adap->pdev_dev, "Interface %s is running %s\n",
  235. dev->name,
  236. dcb_ver_array[dcb->dcb_version]);
  237. } else {
  238. dev_warn(adap->pdev_dev,
  239. "Something screwed up, requested firmware for %s, but firmware returned %s instead\n",
  240. dcb_ver_array[dcb->dcb_version],
  241. dcb_ver_array[dcb_running_version]);
  242. dcb->dcb_version = FW_PORT_DCB_VER_UNKNOWN;
  243. }
  244. }
  245. cxgb4_dcb_state_fsm(dev, input);
  246. return;
  247. }
  248. /* It's weird, and almost certainly an error, to get Firmware DCB
  249. * messages when we either haven't been told whether we're going to be
  250. * doing Host or Firmware DCB; and even worse when we've been told
  251. * that we're doing Host DCB!
  252. */
  253. if (dcb->state == CXGB4_DCB_STATE_START ||
  254. dcb->state == CXGB4_DCB_STATE_HOST) {
  255. dev_err(adap->pdev_dev, "Receiving Firmware DCB messages in State %d\n",
  256. dcb->state);
  257. return;
  258. }
  259. /* Now handle the general Firmware DCB update messages ...
  260. */
  261. switch (dcb_type) {
  262. case FW_PORT_DCB_TYPE_PGID:
  263. dcb->pgid = be32_to_cpu(fwdcb->pgid.pgid);
  264. dcb->msgs |= CXGB4_DCB_FW_PGID;
  265. break;
  266. case FW_PORT_DCB_TYPE_PGRATE:
  267. dcb->pg_num_tcs_supported = fwdcb->pgrate.num_tcs_supported;
  268. memcpy(dcb->pgrate, &fwdcb->pgrate.pgrate,
  269. sizeof(dcb->pgrate));
  270. memcpy(dcb->tsa, &fwdcb->pgrate.tsa,
  271. sizeof(dcb->tsa));
  272. dcb->msgs |= CXGB4_DCB_FW_PGRATE;
  273. if (dcb->msgs & CXGB4_DCB_FW_PGID)
  274. IEEE_FAUX_SYNC(dev, dcb);
  275. break;
  276. case FW_PORT_DCB_TYPE_PRIORATE:
  277. memcpy(dcb->priorate, &fwdcb->priorate.strict_priorate,
  278. sizeof(dcb->priorate));
  279. dcb->msgs |= CXGB4_DCB_FW_PRIORATE;
  280. break;
  281. case FW_PORT_DCB_TYPE_PFC:
  282. dcb->pfcen = fwdcb->pfc.pfcen;
  283. dcb->pfc_num_tcs_supported = fwdcb->pfc.max_pfc_tcs;
  284. dcb->msgs |= CXGB4_DCB_FW_PFC;
  285. IEEE_FAUX_SYNC(dev, dcb);
  286. break;
  287. case FW_PORT_DCB_TYPE_APP_ID: {
  288. const struct fw_port_app_priority *fwap = &fwdcb->app_priority;
  289. int idx = fwap->idx;
  290. struct app_priority *ap = &dcb->app_priority[idx];
  291. struct dcb_app app = {
  292. .protocol = be16_to_cpu(fwap->protocolid),
  293. };
  294. int err;
  295. /* Convert from firmware format to relevant format
  296. * when using app selector
  297. */
  298. if (dcb->dcb_version == FW_PORT_DCB_VER_IEEE) {
  299. app.selector = (fwap->sel_field + 1);
  300. app.priority = ffs(fwap->user_prio_map) - 1;
  301. err = dcb_ieee_setapp(dev, &app);
  302. IEEE_FAUX_SYNC(dev, dcb);
  303. } else {
  304. /* Default is CEE */
  305. app.selector = !!(fwap->sel_field);
  306. app.priority = fwap->user_prio_map;
  307. err = dcb_setapp(dev, &app);
  308. }
  309. if (err)
  310. dev_err(adap->pdev_dev,
  311. "Failed DCB Set Application Priority: sel=%d, prot=%d, prio=%d, err=%d\n",
  312. app.selector, app.protocol, app.priority, -err);
  313. ap->user_prio_map = fwap->user_prio_map;
  314. ap->sel_field = fwap->sel_field;
  315. ap->protocolid = be16_to_cpu(fwap->protocolid);
  316. dcb->msgs |= CXGB4_DCB_FW_APP_ID;
  317. break;
  318. }
  319. default:
  320. dev_err(adap->pdev_dev, "Unknown DCB update type received %x\n",
  321. dcb_type);
  322. break;
  323. }
  324. }
  325. /* Data Center Bridging netlink operations.
  326. */
  327. /* Get current DCB enabled/disabled state.
  328. */
  329. static u8 cxgb4_getstate(struct net_device *dev)
  330. {
  331. struct port_info *pi = netdev2pinfo(dev);
  332. return pi->dcb.enabled;
  333. }
  334. /* Set DCB enabled/disabled.
  335. */
  336. static u8 cxgb4_setstate(struct net_device *dev, u8 enabled)
  337. {
  338. struct port_info *pi = netdev2pinfo(dev);
  339. /* If DCBx is host-managed, dcb is enabled by outside lldp agents */
  340. if (pi->dcb.state == CXGB4_DCB_STATE_HOST) {
  341. pi->dcb.enabled = enabled;
  342. return 0;
  343. }
  344. /* Firmware doesn't provide any mechanism to control the DCB state.
  345. */
  346. if (enabled != (pi->dcb.state == CXGB4_DCB_STATE_FW_ALLSYNCED))
  347. return 1;
  348. return 0;
  349. }
  350. static void cxgb4_getpgtccfg(struct net_device *dev, int tc,
  351. u8 *prio_type, u8 *pgid, u8 *bw_per,
  352. u8 *up_tc_map, int local)
  353. {
  354. struct fw_port_cmd pcmd;
  355. struct port_info *pi = netdev2pinfo(dev);
  356. struct adapter *adap = pi->adapter;
  357. int err;
  358. *prio_type = *pgid = *bw_per = *up_tc_map = 0;
  359. if (local)
  360. INIT_PORT_DCB_READ_LOCAL_CMD(pcmd, pi->port_id);
  361. else
  362. INIT_PORT_DCB_READ_PEER_CMD(pcmd, pi->port_id);
  363. pcmd.u.dcb.pgid.type = FW_PORT_DCB_TYPE_PGID;
  364. err = t4_wr_mbox(adap, adap->mbox, &pcmd, sizeof(pcmd), &pcmd);
  365. if (err != FW_PORT_DCB_CFG_SUCCESS) {
  366. dev_err(adap->pdev_dev, "DCB read PGID failed with %d\n", -err);
  367. return;
  368. }
  369. *pgid = (be32_to_cpu(pcmd.u.dcb.pgid.pgid) >> (tc * 4)) & 0xf;
  370. INIT_PORT_DCB_READ_PEER_CMD(pcmd, pi->port_id);
  371. pcmd.u.dcb.pgrate.type = FW_PORT_DCB_TYPE_PGRATE;
  372. err = t4_wr_mbox(adap, adap->mbox, &pcmd, sizeof(pcmd), &pcmd);
  373. if (err != FW_PORT_DCB_CFG_SUCCESS) {
  374. dev_err(adap->pdev_dev, "DCB read PGRATE failed with %d\n",
  375. -err);
  376. return;
  377. }
  378. *bw_per = pcmd.u.dcb.pgrate.pgrate[*pgid];
  379. *up_tc_map = (1 << tc);
  380. /* prio_type is link strict */
  381. if (*pgid != 0xF)
  382. *prio_type = 0x2;
  383. }
  384. static void cxgb4_getpgtccfg_tx(struct net_device *dev, int tc,
  385. u8 *prio_type, u8 *pgid, u8 *bw_per,
  386. u8 *up_tc_map)
  387. {
  388. /* tc 0 is written at MSB position */
  389. return cxgb4_getpgtccfg(dev, (7 - tc), prio_type, pgid, bw_per,
  390. up_tc_map, 1);
  391. }
  392. static void cxgb4_getpgtccfg_rx(struct net_device *dev, int tc,
  393. u8 *prio_type, u8 *pgid, u8 *bw_per,
  394. u8 *up_tc_map)
  395. {
  396. /* tc 0 is written at MSB position */
  397. return cxgb4_getpgtccfg(dev, (7 - tc), prio_type, pgid, bw_per,
  398. up_tc_map, 0);
  399. }
  400. static void cxgb4_setpgtccfg_tx(struct net_device *dev, int tc,
  401. u8 prio_type, u8 pgid, u8 bw_per,
  402. u8 up_tc_map)
  403. {
  404. struct fw_port_cmd pcmd;
  405. struct port_info *pi = netdev2pinfo(dev);
  406. struct adapter *adap = pi->adapter;
  407. int fw_tc = 7 - tc;
  408. u32 _pgid;
  409. int err;
  410. if (pgid == DCB_ATTR_VALUE_UNDEFINED)
  411. return;
  412. if (bw_per == DCB_ATTR_VALUE_UNDEFINED)
  413. return;
  414. INIT_PORT_DCB_READ_LOCAL_CMD(pcmd, pi->port_id);
  415. pcmd.u.dcb.pgid.type = FW_PORT_DCB_TYPE_PGID;
  416. err = t4_wr_mbox(adap, adap->mbox, &pcmd, sizeof(pcmd), &pcmd);
  417. if (err != FW_PORT_DCB_CFG_SUCCESS) {
  418. dev_err(adap->pdev_dev, "DCB read PGID failed with %d\n", -err);
  419. return;
  420. }
  421. _pgid = be32_to_cpu(pcmd.u.dcb.pgid.pgid);
  422. _pgid &= ~(0xF << (fw_tc * 4));
  423. _pgid |= pgid << (fw_tc * 4);
  424. pcmd.u.dcb.pgid.pgid = cpu_to_be32(_pgid);
  425. INIT_PORT_DCB_WRITE_CMD(pcmd, pi->port_id);
  426. err = t4_wr_mbox(adap, adap->mbox, &pcmd, sizeof(pcmd), &pcmd);
  427. if (err != FW_PORT_DCB_CFG_SUCCESS) {
  428. dev_err(adap->pdev_dev, "DCB write PGID failed with %d\n",
  429. -err);
  430. return;
  431. }
  432. memset(&pcmd, 0, sizeof(struct fw_port_cmd));
  433. INIT_PORT_DCB_READ_LOCAL_CMD(pcmd, pi->port_id);
  434. pcmd.u.dcb.pgrate.type = FW_PORT_DCB_TYPE_PGRATE;
  435. err = t4_wr_mbox(adap, adap->mbox, &pcmd, sizeof(pcmd), &pcmd);
  436. if (err != FW_PORT_DCB_CFG_SUCCESS) {
  437. dev_err(adap->pdev_dev, "DCB read PGRATE failed with %d\n",
  438. -err);
  439. return;
  440. }
  441. pcmd.u.dcb.pgrate.pgrate[pgid] = bw_per;
  442. INIT_PORT_DCB_WRITE_CMD(pcmd, pi->port_id);
  443. if (pi->dcb.state == CXGB4_DCB_STATE_HOST)
  444. pcmd.op_to_portid |= cpu_to_be32(FW_PORT_CMD_APPLY);
  445. err = t4_wr_mbox(adap, adap->mbox, &pcmd, sizeof(pcmd), &pcmd);
  446. if (err != FW_PORT_DCB_CFG_SUCCESS)
  447. dev_err(adap->pdev_dev, "DCB write PGRATE failed with %d\n",
  448. -err);
  449. }
  450. static void cxgb4_getpgbwgcfg(struct net_device *dev, int pgid, u8 *bw_per,
  451. int local)
  452. {
  453. struct fw_port_cmd pcmd;
  454. struct port_info *pi = netdev2pinfo(dev);
  455. struct adapter *adap = pi->adapter;
  456. int err;
  457. if (local)
  458. INIT_PORT_DCB_READ_LOCAL_CMD(pcmd, pi->port_id);
  459. else
  460. INIT_PORT_DCB_READ_PEER_CMD(pcmd, pi->port_id);
  461. pcmd.u.dcb.pgrate.type = FW_PORT_DCB_TYPE_PGRATE;
  462. err = t4_wr_mbox(adap, adap->mbox, &pcmd, sizeof(pcmd), &pcmd);
  463. if (err != FW_PORT_DCB_CFG_SUCCESS) {
  464. dev_err(adap->pdev_dev, "DCB read PGRATE failed with %d\n",
  465. -err);
  466. return;
  467. }
  468. *bw_per = pcmd.u.dcb.pgrate.pgrate[pgid];
  469. }
  470. static void cxgb4_getpgbwgcfg_tx(struct net_device *dev, int pgid, u8 *bw_per)
  471. {
  472. return cxgb4_getpgbwgcfg(dev, pgid, bw_per, 1);
  473. }
  474. static void cxgb4_getpgbwgcfg_rx(struct net_device *dev, int pgid, u8 *bw_per)
  475. {
  476. return cxgb4_getpgbwgcfg(dev, pgid, bw_per, 0);
  477. }
  478. static void cxgb4_setpgbwgcfg_tx(struct net_device *dev, int pgid,
  479. u8 bw_per)
  480. {
  481. struct fw_port_cmd pcmd;
  482. struct port_info *pi = netdev2pinfo(dev);
  483. struct adapter *adap = pi->adapter;
  484. int err;
  485. INIT_PORT_DCB_READ_LOCAL_CMD(pcmd, pi->port_id);
  486. pcmd.u.dcb.pgrate.type = FW_PORT_DCB_TYPE_PGRATE;
  487. err = t4_wr_mbox(adap, adap->mbox, &pcmd, sizeof(pcmd), &pcmd);
  488. if (err != FW_PORT_DCB_CFG_SUCCESS) {
  489. dev_err(adap->pdev_dev, "DCB read PGRATE failed with %d\n",
  490. -err);
  491. return;
  492. }
  493. pcmd.u.dcb.pgrate.pgrate[pgid] = bw_per;
  494. INIT_PORT_DCB_WRITE_CMD(pcmd, pi->port_id);
  495. if (pi->dcb.state == CXGB4_DCB_STATE_HOST)
  496. pcmd.op_to_portid |= cpu_to_be32(FW_PORT_CMD_APPLY);
  497. err = t4_wr_mbox(adap, adap->mbox, &pcmd, sizeof(pcmd), &pcmd);
  498. if (err != FW_PORT_DCB_CFG_SUCCESS)
  499. dev_err(adap->pdev_dev, "DCB write PGRATE failed with %d\n",
  500. -err);
  501. }
  502. /* Return whether the specified Traffic Class Priority has Priority Pause
  503. * Frames enabled.
  504. */
  505. static void cxgb4_getpfccfg(struct net_device *dev, int priority, u8 *pfccfg)
  506. {
  507. struct port_info *pi = netdev2pinfo(dev);
  508. struct port_dcb_info *dcb = &pi->dcb;
  509. if (dcb->state != CXGB4_DCB_STATE_FW_ALLSYNCED ||
  510. priority >= CXGB4_MAX_PRIORITY)
  511. *pfccfg = 0;
  512. else
  513. *pfccfg = (pi->dcb.pfcen >> (7 - priority)) & 1;
  514. }
  515. /* Enable/disable Priority Pause Frames for the specified Traffic Class
  516. * Priority.
  517. */
  518. static void cxgb4_setpfccfg(struct net_device *dev, int priority, u8 pfccfg)
  519. {
  520. struct fw_port_cmd pcmd;
  521. struct port_info *pi = netdev2pinfo(dev);
  522. struct adapter *adap = pi->adapter;
  523. int err;
  524. if (pi->dcb.state != CXGB4_DCB_STATE_FW_ALLSYNCED ||
  525. priority >= CXGB4_MAX_PRIORITY)
  526. return;
  527. INIT_PORT_DCB_WRITE_CMD(pcmd, pi->port_id);
  528. if (pi->dcb.state == CXGB4_DCB_STATE_HOST)
  529. pcmd.op_to_portid |= cpu_to_be32(FW_PORT_CMD_APPLY);
  530. pcmd.u.dcb.pfc.type = FW_PORT_DCB_TYPE_PFC;
  531. pcmd.u.dcb.pfc.pfcen = pi->dcb.pfcen;
  532. if (pfccfg)
  533. pcmd.u.dcb.pfc.pfcen |= (1 << (7 - priority));
  534. else
  535. pcmd.u.dcb.pfc.pfcen &= (~(1 << (7 - priority)));
  536. err = t4_wr_mbox(adap, adap->mbox, &pcmd, sizeof(pcmd), &pcmd);
  537. if (err != FW_PORT_DCB_CFG_SUCCESS) {
  538. dev_err(adap->pdev_dev, "DCB PFC write failed with %d\n", -err);
  539. return;
  540. }
  541. pi->dcb.pfcen = pcmd.u.dcb.pfc.pfcen;
  542. }
  543. static u8 cxgb4_setall(struct net_device *dev)
  544. {
  545. return 0;
  546. }
  547. /* Return DCB capabilities.
  548. */
  549. static u8 cxgb4_getcap(struct net_device *dev, int cap_id, u8 *caps)
  550. {
  551. struct port_info *pi = netdev2pinfo(dev);
  552. switch (cap_id) {
  553. case DCB_CAP_ATTR_PG:
  554. case DCB_CAP_ATTR_PFC:
  555. *caps = true;
  556. break;
  557. case DCB_CAP_ATTR_PG_TCS:
  558. /* 8 priorities for PG represented by bitmap */
  559. *caps = 0x80;
  560. break;
  561. case DCB_CAP_ATTR_PFC_TCS:
  562. /* 8 priorities for PFC represented by bitmap */
  563. *caps = 0x80;
  564. break;
  565. case DCB_CAP_ATTR_GSP:
  566. *caps = true;
  567. break;
  568. case DCB_CAP_ATTR_UP2TC:
  569. case DCB_CAP_ATTR_BCN:
  570. *caps = false;
  571. break;
  572. case DCB_CAP_ATTR_DCBX:
  573. *caps = pi->dcb.supported;
  574. break;
  575. default:
  576. *caps = false;
  577. }
  578. return 0;
  579. }
  580. /* Return the number of Traffic Classes for the indicated Traffic Class ID.
  581. */
  582. static int cxgb4_getnumtcs(struct net_device *dev, int tcs_id, u8 *num)
  583. {
  584. struct port_info *pi = netdev2pinfo(dev);
  585. switch (tcs_id) {
  586. case DCB_NUMTCS_ATTR_PG:
  587. if (pi->dcb.msgs & CXGB4_DCB_FW_PGRATE)
  588. *num = pi->dcb.pg_num_tcs_supported;
  589. else
  590. *num = 0x8;
  591. break;
  592. case DCB_NUMTCS_ATTR_PFC:
  593. *num = 0x8;
  594. break;
  595. default:
  596. return -EINVAL;
  597. }
  598. return 0;
  599. }
  600. /* Set the number of Traffic Classes supported for the indicated Traffic Class
  601. * ID.
  602. */
  603. static int cxgb4_setnumtcs(struct net_device *dev, int tcs_id, u8 num)
  604. {
  605. /* Setting the number of Traffic Classes isn't supported.
  606. */
  607. return -ENOSYS;
  608. }
  609. /* Return whether Priority Flow Control is enabled. */
  610. static u8 cxgb4_getpfcstate(struct net_device *dev)
  611. {
  612. struct port_info *pi = netdev2pinfo(dev);
  613. if (pi->dcb.state != CXGB4_DCB_STATE_FW_ALLSYNCED)
  614. return false;
  615. return pi->dcb.pfcen != 0;
  616. }
  617. /* Enable/disable Priority Flow Control. */
  618. static void cxgb4_setpfcstate(struct net_device *dev, u8 state)
  619. {
  620. /* We can't enable/disable Priority Flow Control but we also can't
  621. * return an error ...
  622. */
  623. }
  624. /* Return the Application User Priority Map associated with the specified
  625. * Application ID.
  626. */
  627. static int __cxgb4_getapp(struct net_device *dev, u8 app_idtype, u16 app_id,
  628. int peer)
  629. {
  630. struct port_info *pi = netdev2pinfo(dev);
  631. struct adapter *adap = pi->adapter;
  632. int i;
  633. if (pi->dcb.state != CXGB4_DCB_STATE_FW_ALLSYNCED)
  634. return 0;
  635. for (i = 0; i < CXGB4_MAX_DCBX_APP_SUPPORTED; i++) {
  636. struct fw_port_cmd pcmd;
  637. int err;
  638. if (peer)
  639. INIT_PORT_DCB_READ_PEER_CMD(pcmd, pi->port_id);
  640. else
  641. INIT_PORT_DCB_READ_LOCAL_CMD(pcmd, pi->port_id);
  642. pcmd.u.dcb.app_priority.type = FW_PORT_DCB_TYPE_APP_ID;
  643. pcmd.u.dcb.app_priority.idx = i;
  644. err = t4_wr_mbox(adap, adap->mbox, &pcmd, sizeof(pcmd), &pcmd);
  645. if (err != FW_PORT_DCB_CFG_SUCCESS) {
  646. dev_err(adap->pdev_dev, "DCB APP read failed with %d\n",
  647. -err);
  648. return err;
  649. }
  650. if (be16_to_cpu(pcmd.u.dcb.app_priority.protocolid) == app_id)
  651. if (pcmd.u.dcb.app_priority.sel_field == app_idtype)
  652. return pcmd.u.dcb.app_priority.user_prio_map;
  653. /* exhausted app list */
  654. if (!pcmd.u.dcb.app_priority.protocolid)
  655. break;
  656. }
  657. return -EEXIST;
  658. }
  659. /* Return the Application User Priority Map associated with the specified
  660. * Application ID.
  661. */
  662. static int cxgb4_getapp(struct net_device *dev, u8 app_idtype, u16 app_id)
  663. {
  664. return __cxgb4_getapp(dev, app_idtype, app_id, 0);
  665. }
  666. /* Write a new Application User Priority Map for the specified Application ID
  667. */
  668. static int __cxgb4_setapp(struct net_device *dev, u8 app_idtype, u16 app_id,
  669. u8 app_prio)
  670. {
  671. struct fw_port_cmd pcmd;
  672. struct port_info *pi = netdev2pinfo(dev);
  673. struct adapter *adap = pi->adapter;
  674. int i, err;
  675. if (pi->dcb.state != CXGB4_DCB_STATE_FW_ALLSYNCED)
  676. return -EINVAL;
  677. /* DCB info gets thrown away on link up */
  678. if (!netif_carrier_ok(dev))
  679. return -ENOLINK;
  680. for (i = 0; i < CXGB4_MAX_DCBX_APP_SUPPORTED; i++) {
  681. INIT_PORT_DCB_READ_LOCAL_CMD(pcmd, pi->port_id);
  682. pcmd.u.dcb.app_priority.type = FW_PORT_DCB_TYPE_APP_ID;
  683. pcmd.u.dcb.app_priority.idx = i;
  684. err = t4_wr_mbox(adap, adap->mbox, &pcmd, sizeof(pcmd), &pcmd);
  685. if (err != FW_PORT_DCB_CFG_SUCCESS) {
  686. dev_err(adap->pdev_dev, "DCB app table read failed with %d\n",
  687. -err);
  688. return err;
  689. }
  690. if (be16_to_cpu(pcmd.u.dcb.app_priority.protocolid) == app_id) {
  691. /* overwrite existing app table */
  692. pcmd.u.dcb.app_priority.protocolid = 0;
  693. break;
  694. }
  695. /* find first empty slot */
  696. if (!pcmd.u.dcb.app_priority.protocolid)
  697. break;
  698. }
  699. if (i == CXGB4_MAX_DCBX_APP_SUPPORTED) {
  700. /* no empty slots available */
  701. dev_err(adap->pdev_dev, "DCB app table full\n");
  702. return -EBUSY;
  703. }
  704. /* write out new app table entry */
  705. INIT_PORT_DCB_WRITE_CMD(pcmd, pi->port_id);
  706. if (pi->dcb.state == CXGB4_DCB_STATE_HOST)
  707. pcmd.op_to_portid |= cpu_to_be32(FW_PORT_CMD_APPLY);
  708. pcmd.u.dcb.app_priority.type = FW_PORT_DCB_TYPE_APP_ID;
  709. pcmd.u.dcb.app_priority.protocolid = cpu_to_be16(app_id);
  710. pcmd.u.dcb.app_priority.sel_field = app_idtype;
  711. pcmd.u.dcb.app_priority.user_prio_map = app_prio;
  712. pcmd.u.dcb.app_priority.idx = i;
  713. err = t4_wr_mbox(adap, adap->mbox, &pcmd, sizeof(pcmd), &pcmd);
  714. if (err != FW_PORT_DCB_CFG_SUCCESS) {
  715. dev_err(adap->pdev_dev, "DCB app table write failed with %d\n",
  716. -err);
  717. return err;
  718. }
  719. return 0;
  720. }
  721. /* Priority for CEE inside dcb_app is bitmask, with 0 being an invalid value */
  722. static int cxgb4_setapp(struct net_device *dev, u8 app_idtype, u16 app_id,
  723. u8 app_prio)
  724. {
  725. int ret;
  726. struct dcb_app app = {
  727. .selector = app_idtype,
  728. .protocol = app_id,
  729. .priority = app_prio,
  730. };
  731. if (app_idtype != DCB_APP_IDTYPE_ETHTYPE &&
  732. app_idtype != DCB_APP_IDTYPE_PORTNUM)
  733. return -EINVAL;
  734. /* Convert app_idtype to a format that firmware understands */
  735. ret = __cxgb4_setapp(dev, app_idtype == DCB_APP_IDTYPE_ETHTYPE ?
  736. app_idtype : 3, app_id, app_prio);
  737. if (ret)
  738. return ret;
  739. return dcb_setapp(dev, &app);
  740. }
  741. /* Return whether IEEE Data Center Bridging has been negotiated.
  742. */
  743. static inline int
  744. cxgb4_ieee_negotiation_complete(struct net_device *dev,
  745. enum cxgb4_dcb_fw_msgs dcb_subtype)
  746. {
  747. struct port_info *pi = netdev2pinfo(dev);
  748. struct port_dcb_info *dcb = &pi->dcb;
  749. if (dcb_subtype && !(dcb->msgs & dcb_subtype))
  750. return 0;
  751. return (dcb->state == CXGB4_DCB_STATE_FW_ALLSYNCED &&
  752. (dcb->supported & DCB_CAP_DCBX_VER_IEEE));
  753. }
  754. /* Fill in the Application User Priority Map associated with the
  755. * specified Application.
  756. * Priority for IEEE dcb_app is an integer, with 0 being a valid value
  757. */
  758. static int cxgb4_ieee_getapp(struct net_device *dev, struct dcb_app *app)
  759. {
  760. int prio;
  761. if (!cxgb4_ieee_negotiation_complete(dev, CXGB4_DCB_FW_APP_ID))
  762. return -EINVAL;
  763. if (!(app->selector && app->protocol))
  764. return -EINVAL;
  765. /* Try querying firmware first, use firmware format */
  766. prio = __cxgb4_getapp(dev, app->selector - 1, app->protocol, 0);
  767. if (prio < 0)
  768. prio = dcb_ieee_getapp_mask(dev, app);
  769. app->priority = ffs(prio) - 1;
  770. return 0;
  771. }
  772. /* Write a new Application User Priority Map for the specified Application ID.
  773. * Priority for IEEE dcb_app is an integer, with 0 being a valid value
  774. */
  775. static int cxgb4_ieee_setapp(struct net_device *dev, struct dcb_app *app)
  776. {
  777. int ret;
  778. if (!cxgb4_ieee_negotiation_complete(dev, CXGB4_DCB_FW_APP_ID))
  779. return -EINVAL;
  780. if (!(app->selector && app->protocol))
  781. return -EINVAL;
  782. if (!(app->selector > IEEE_8021QAZ_APP_SEL_ETHERTYPE &&
  783. app->selector < IEEE_8021QAZ_APP_SEL_ANY))
  784. return -EINVAL;
  785. /* change selector to a format that firmware understands */
  786. ret = __cxgb4_setapp(dev, app->selector - 1, app->protocol,
  787. (1 << app->priority));
  788. if (ret)
  789. return ret;
  790. return dcb_ieee_setapp(dev, app);
  791. }
  792. /* Return our DCBX parameters.
  793. */
  794. static u8 cxgb4_getdcbx(struct net_device *dev)
  795. {
  796. struct port_info *pi = netdev2pinfo(dev);
  797. /* This is already set by cxgb4_set_dcb_caps, so just return it */
  798. return pi->dcb.supported;
  799. }
  800. /* Set our DCBX parameters.
  801. */
  802. static u8 cxgb4_setdcbx(struct net_device *dev, u8 dcb_request)
  803. {
  804. struct port_info *pi = netdev2pinfo(dev);
  805. /* Filter out requests which exceed our capabilities.
  806. */
  807. if ((dcb_request & (CXGB4_DCBX_FW_SUPPORT | CXGB4_DCBX_HOST_SUPPORT))
  808. != dcb_request)
  809. return 1;
  810. /* Can't enable DCB if we haven't successfully negotiated it.
  811. */
  812. if (pi->dcb.state != CXGB4_DCB_STATE_FW_ALLSYNCED)
  813. return 1;
  814. /* There's currently no mechanism to allow for the firmware DCBX
  815. * negotiation to be changed from the Host Driver. If the caller
  816. * requests exactly the same parameters that we already have then
  817. * we'll allow them to be successfully "set" ...
  818. */
  819. if (dcb_request != pi->dcb.supported)
  820. return 1;
  821. pi->dcb.supported = dcb_request;
  822. return 0;
  823. }
  824. static int cxgb4_getpeer_app(struct net_device *dev,
  825. struct dcb_peer_app_info *info, u16 *app_count)
  826. {
  827. struct fw_port_cmd pcmd;
  828. struct port_info *pi = netdev2pinfo(dev);
  829. struct adapter *adap = pi->adapter;
  830. int i, err = 0;
  831. if (pi->dcb.state != CXGB4_DCB_STATE_FW_ALLSYNCED)
  832. return 1;
  833. info->willing = 0;
  834. info->error = 0;
  835. *app_count = 0;
  836. for (i = 0; i < CXGB4_MAX_DCBX_APP_SUPPORTED; i++) {
  837. INIT_PORT_DCB_READ_PEER_CMD(pcmd, pi->port_id);
  838. pcmd.u.dcb.app_priority.type = FW_PORT_DCB_TYPE_APP_ID;
  839. pcmd.u.dcb.app_priority.idx = *app_count;
  840. err = t4_wr_mbox(adap, adap->mbox, &pcmd, sizeof(pcmd), &pcmd);
  841. if (err != FW_PORT_DCB_CFG_SUCCESS) {
  842. dev_err(adap->pdev_dev, "DCB app table read failed with %d\n",
  843. -err);
  844. return err;
  845. }
  846. /* find first empty slot */
  847. if (!pcmd.u.dcb.app_priority.protocolid)
  848. break;
  849. }
  850. *app_count = i;
  851. return err;
  852. }
  853. static int cxgb4_getpeerapp_tbl(struct net_device *dev, struct dcb_app *table)
  854. {
  855. struct fw_port_cmd pcmd;
  856. struct port_info *pi = netdev2pinfo(dev);
  857. struct adapter *adap = pi->adapter;
  858. int i, err = 0;
  859. if (pi->dcb.state != CXGB4_DCB_STATE_FW_ALLSYNCED)
  860. return 1;
  861. for (i = 0; i < CXGB4_MAX_DCBX_APP_SUPPORTED; i++) {
  862. INIT_PORT_DCB_READ_PEER_CMD(pcmd, pi->port_id);
  863. pcmd.u.dcb.app_priority.type = FW_PORT_DCB_TYPE_APP_ID;
  864. pcmd.u.dcb.app_priority.idx = i;
  865. err = t4_wr_mbox(adap, adap->mbox, &pcmd, sizeof(pcmd), &pcmd);
  866. if (err != FW_PORT_DCB_CFG_SUCCESS) {
  867. dev_err(adap->pdev_dev, "DCB app table read failed with %d\n",
  868. -err);
  869. return err;
  870. }
  871. /* find first empty slot */
  872. if (!pcmd.u.dcb.app_priority.protocolid)
  873. break;
  874. table[i].selector = pcmd.u.dcb.app_priority.sel_field;
  875. table[i].protocol =
  876. be16_to_cpu(pcmd.u.dcb.app_priority.protocolid);
  877. table[i].priority =
  878. ffs(pcmd.u.dcb.app_priority.user_prio_map) - 1;
  879. }
  880. return err;
  881. }
  882. /* Return Priority Group information.
  883. */
  884. static int cxgb4_cee_peer_getpg(struct net_device *dev, struct cee_pg *pg)
  885. {
  886. struct fw_port_cmd pcmd;
  887. struct port_info *pi = netdev2pinfo(dev);
  888. struct adapter *adap = pi->adapter;
  889. u32 pgid;
  890. int i, err;
  891. /* We're always "willing" -- the Switch Fabric always dictates the
  892. * DCBX parameters to us.
  893. */
  894. pg->willing = true;
  895. INIT_PORT_DCB_READ_PEER_CMD(pcmd, pi->port_id);
  896. pcmd.u.dcb.pgid.type = FW_PORT_DCB_TYPE_PGID;
  897. err = t4_wr_mbox(adap, adap->mbox, &pcmd, sizeof(pcmd), &pcmd);
  898. if (err != FW_PORT_DCB_CFG_SUCCESS) {
  899. dev_err(adap->pdev_dev, "DCB read PGID failed with %d\n", -err);
  900. return err;
  901. }
  902. pgid = be32_to_cpu(pcmd.u.dcb.pgid.pgid);
  903. for (i = 0; i < CXGB4_MAX_PRIORITY; i++)
  904. pg->prio_pg[7 - i] = (pgid >> (i * 4)) & 0xF;
  905. INIT_PORT_DCB_READ_PEER_CMD(pcmd, pi->port_id);
  906. pcmd.u.dcb.pgrate.type = FW_PORT_DCB_TYPE_PGRATE;
  907. err = t4_wr_mbox(adap, adap->mbox, &pcmd, sizeof(pcmd), &pcmd);
  908. if (err != FW_PORT_DCB_CFG_SUCCESS) {
  909. dev_err(adap->pdev_dev, "DCB read PGRATE failed with %d\n",
  910. -err);
  911. return err;
  912. }
  913. for (i = 0; i < CXGB4_MAX_PRIORITY; i++)
  914. pg->pg_bw[i] = pcmd.u.dcb.pgrate.pgrate[i];
  915. return 0;
  916. }
  917. /* Return Priority Flow Control information.
  918. */
  919. static int cxgb4_cee_peer_getpfc(struct net_device *dev, struct cee_pfc *pfc)
  920. {
  921. struct port_info *pi = netdev2pinfo(dev);
  922. cxgb4_getnumtcs(dev, DCB_NUMTCS_ATTR_PFC, &(pfc->tcs_supported));
  923. pfc->pfc_en = pi->dcb.pfcen;
  924. return 0;
  925. }
  926. const struct dcbnl_rtnl_ops cxgb4_dcb_ops = {
  927. .ieee_getapp = cxgb4_ieee_getapp,
  928. .ieee_setapp = cxgb4_ieee_setapp,
  929. /* CEE std */
  930. .getstate = cxgb4_getstate,
  931. .setstate = cxgb4_setstate,
  932. .getpgtccfgtx = cxgb4_getpgtccfg_tx,
  933. .getpgbwgcfgtx = cxgb4_getpgbwgcfg_tx,
  934. .getpgtccfgrx = cxgb4_getpgtccfg_rx,
  935. .getpgbwgcfgrx = cxgb4_getpgbwgcfg_rx,
  936. .setpgtccfgtx = cxgb4_setpgtccfg_tx,
  937. .setpgbwgcfgtx = cxgb4_setpgbwgcfg_tx,
  938. .setpfccfg = cxgb4_setpfccfg,
  939. .getpfccfg = cxgb4_getpfccfg,
  940. .setall = cxgb4_setall,
  941. .getcap = cxgb4_getcap,
  942. .getnumtcs = cxgb4_getnumtcs,
  943. .setnumtcs = cxgb4_setnumtcs,
  944. .getpfcstate = cxgb4_getpfcstate,
  945. .setpfcstate = cxgb4_setpfcstate,
  946. .getapp = cxgb4_getapp,
  947. .setapp = cxgb4_setapp,
  948. /* DCBX configuration */
  949. .getdcbx = cxgb4_getdcbx,
  950. .setdcbx = cxgb4_setdcbx,
  951. /* peer apps */
  952. .peer_getappinfo = cxgb4_getpeer_app,
  953. .peer_getapptable = cxgb4_getpeerapp_tbl,
  954. /* CEE peer */
  955. .cee_peer_getpg = cxgb4_cee_peer_getpg,
  956. .cee_peer_getpfc = cxgb4_cee_peer_getpfc,
  957. };