hif_napi.c 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473
  1. /*
  2. * Copyright (c) 2015-2016 The Linux Foundation. All rights reserved.
  3. *
  4. * Previously licensed under the ISC license by Qualcomm Atheros, Inc.
  5. *
  6. *
  7. * Permission to use, copy, modify, and/or distribute this software for
  8. * any purpose with or without fee is hereby granted, provided that the
  9. * above copyright notice and this permission notice appear in all
  10. * copies.
  11. *
  12. * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL
  13. * WARRANTIES WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED
  14. * WARRANTIES OF MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE
  15. * AUTHOR BE LIABLE FOR ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL
  16. * DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR
  17. * PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE OR OTHER
  18. * TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR
  19. * PERFORMANCE OF THIS SOFTWARE.
  20. */
  21. /*
  22. * This file was originally distributed by Qualcomm Atheros, Inc.
  23. * under proprietary terms before Copyright ownership was assigned
  24. * to the Linux Foundation.
  25. */
  26. /**
  27. * DOC: hif_napi.c
  28. *
  29. * HIF NAPI interface implementation
  30. */
  31. #include <string.h> /* memset */
  32. #include <hif_napi.h>
  33. #include <hif_debug.h>
  34. #include <hif_io32.h>
  35. #include <ce_api.h>
  36. #include <ce_internal.h>
  37. enum napi_decision_vector {
  38. HIF_NAPI_NOEVENT = 0,
  39. HIF_NAPI_INITED = 1,
  40. HIF_NAPI_CONF_UP = 2
  41. };
  42. #define ENABLE_NAPI_MASK (HIF_NAPI_INITED | HIF_NAPI_CONF_UP)
  43. /**
  44. * hif_napi_create() - creates the NAPI structures for a given CE
  45. * @hif : pointer to hif context
  46. * @pipe_id: the CE id on which the instance will be created
  47. * @poll : poll function to be used for this NAPI instance
  48. * @budget : budget to be registered with the NAPI instance
  49. * @scale : scale factor on the weight (to scaler budget to 1000)
  50. *
  51. * Description:
  52. * Creates NAPI instances. This function is called
  53. * unconditionally during initialization. It creates
  54. * napi structures through the proper HTC/HIF calls.
  55. * The structures are disabled on creation.
  56. * Note that for each NAPI instance a separate dummy netdev is used
  57. *
  58. * Return:
  59. * < 0: error
  60. * = 0: <should never happen>
  61. * > 0: id of the created object (for multi-NAPI, number of objects created)
  62. */
  63. int hif_napi_create(struct hif_opaque_softc *hif_ctx,
  64. uint8_t pipe_id,
  65. int (*poll)(struct napi_struct *, int),
  66. int budget,
  67. int scale)
  68. {
  69. struct qca_napi_data *napid;
  70. struct qca_napi_info *napii;
  71. struct hif_softc *hif = HIF_GET_SOFTC(hif_ctx);
  72. NAPI_DEBUG("-->(pipe=%d, budget=%d, scale=%d)",
  73. pipe_id, budget, scale);
  74. NAPI_DEBUG("hif->napi_data.state = 0x%08x",
  75. hif->napi_data.state);
  76. NAPI_DEBUG("hif->napi_data.ce_map = 0x%08x",
  77. hif->napi_data.ce_map);
  78. napid = &(hif->napi_data);
  79. if (0 == (napid->state & HIF_NAPI_INITED)) {
  80. memset(napid, 0, sizeof(struct qca_napi_data));
  81. mutex_init(&(napid->mutex));
  82. init_dummy_netdev(&(napid->netdev));
  83. napid->state |= HIF_NAPI_INITED;
  84. HIF_INFO("%s: NAPI structures initialized", __func__);
  85. NAPI_DEBUG("NAPI structures initialized");
  86. }
  87. napii = &(napid->napis[pipe_id]);
  88. memset(napii, 0, sizeof(struct qca_napi_info));
  89. napii->scale = scale;
  90. napii->id = NAPI_PIPE2ID(pipe_id);
  91. NAPI_DEBUG("adding napi=%p to netdev=%p (poll=%p, bdgt=%d)",
  92. &(napii->napi), &(napid->netdev), poll, budget);
  93. netif_napi_add(&(napid->netdev), &(napii->napi), poll, budget);
  94. NAPI_DEBUG("after napi_add");
  95. NAPI_DEBUG("napi=0x%p, netdev=0x%p",
  96. &(napii->napi), &(napid->netdev));
  97. NAPI_DEBUG("napi.dev_list.prev=0x%p, next=0x%p",
  98. napii->napi.dev_list.prev, napii->napi.dev_list.next);
  99. NAPI_DEBUG("dev.napi_list.prev=0x%p, next=0x%p",
  100. napid->netdev.napi_list.prev, napid->netdev.napi_list.next);
  101. /* It is OK to change the state variable below without protection
  102. * as there should be no-one around yet
  103. */
  104. napid->ce_map |= (0x01 << pipe_id);
  105. HIF_INFO("%s: NAPI id %d created for pipe %d", __func__,
  106. napii->id, pipe_id);
  107. NAPI_DEBUG("NAPI id %d created for pipe %d", napii->id, pipe_id);
  108. NAPI_DEBUG("<--napi_id=%d]", napii->id);
  109. return napii->id;
  110. }
  111. /**
  112. *
  113. * hif_napi_destroy() - destroys the NAPI structures for a given instance
  114. * @hif : pointer to hif context
  115. * @ce_id : the CE id whose napi instance will be destroyed
  116. * @force : if set, will destroy even if entry is active (de-activates)
  117. *
  118. * Description:
  119. * Destroy a given NAPI instance. This function is called
  120. * unconditionally during cleanup.
  121. * Refuses to destroy an entry of it is still enabled (unless force=1)
  122. * Marks the whole napi_data invalid if all instances are destroyed.
  123. *
  124. * Return:
  125. * -EINVAL: specific entry has not been created
  126. * -EPERM : specific entry is still active
  127. * 0 < : error
  128. * 0 = : success
  129. */
  130. int hif_napi_destroy(struct hif_opaque_softc *hif_ctx,
  131. uint8_t id,
  132. int force)
  133. {
  134. uint8_t ce = NAPI_ID2PIPE(id);
  135. int rc = 0;
  136. struct hif_softc *hif = HIF_GET_SOFTC(hif_ctx);
  137. NAPI_DEBUG("-->(id=%d, force=%d)", id, force);
  138. if (0 == (hif->napi_data.state & HIF_NAPI_INITED)) {
  139. HIF_ERROR("%s: NAPI not initialized or entry %d not created\n",
  140. __func__, id);
  141. rc = -EINVAL;
  142. } else if (0 == (hif->napi_data.ce_map & (0x01 << ce))) {
  143. HIF_ERROR("%s: NAPI instance %d (pipe %d) not created\n",
  144. __func__, id, ce);
  145. rc = -EINVAL;
  146. } else {
  147. struct qca_napi_data *napid;
  148. struct qca_napi_info *napii;
  149. napid = &(hif->napi_data);
  150. napii = &(napid->napis[ce]);
  151. if (hif->napi_data.state == HIF_NAPI_CONF_UP) {
  152. if (force) {
  153. napi_disable(&(napii->napi));
  154. HIF_INFO("%s: NAPI entry %d force disabled\n",
  155. __func__, id);
  156. NAPI_DEBUG("NAPI %d force disabled", id);
  157. } else {
  158. HIF_ERROR("%s: Cannot destroy active NAPI %d\n",
  159. __func__, id);
  160. rc = -EPERM;
  161. }
  162. }
  163. if (0 == rc) {
  164. NAPI_DEBUG("before napi_del");
  165. NAPI_DEBUG("napi.dlist.prv=0x%p, next=0x%p",
  166. napii->napi.dev_list.prev,
  167. napii->napi.dev_list.next);
  168. NAPI_DEBUG("dev.napi_l.prv=0x%p, next=0x%p",
  169. napid->netdev.napi_list.prev,
  170. napid->netdev.napi_list.next);
  171. netif_napi_del(&(napii->napi));
  172. napid->ce_map &= ~(0x01 << ce);
  173. napii->scale = 0;
  174. HIF_INFO("%s: NAPI %d destroyed\n", __func__, id);
  175. /* if there are no active instances and
  176. * if they are all destroyed,
  177. * set the whole structure to uninitialized state
  178. */
  179. if (napid->ce_map == 0) {
  180. /* hif->napi_data.state = 0; */
  181. memset(napid,
  182. 0, sizeof(struct qca_napi_data));
  183. HIF_INFO("%s: no NAPI instances. Zapped.\n",
  184. __func__);
  185. }
  186. }
  187. }
  188. return rc;
  189. }
  190. /**
  191. *
  192. * hif_napi_get_all() - returns the address of the whole HIF NAPI structure
  193. * @hif: pointer to hif context
  194. *
  195. * Description:
  196. * Returns the address of the whole structure
  197. *
  198. * Return:
  199. * <addr>: address of the whole HIF NAPI structure
  200. */
  201. inline struct qca_napi_data *hif_napi_get_all(struct hif_opaque_softc *hif_ctx)
  202. {
  203. struct hif_softc *hif = HIF_GET_SOFTC(hif_ctx);
  204. return &(hif->napi_data);
  205. }
  206. /**
  207. *
  208. * hif_napi_event() - Decision-maker to enable/disable NAPI.
  209. * @hif : pointer to hif context
  210. * @evnt: event that has been detected
  211. * @data: more data regarding the event
  212. *
  213. * Description:
  214. * This function decides whether or not NAPI should be enabled.
  215. * NAPI will be enabled, if all the following is satisfied.
  216. * 1- has been enabled administratively:
  217. * the .ini file has the enabled setting and it has not been disabled
  218. * by an vendor command override later
  219. *
  220. * Return:
  221. * < 0: some error
  222. * = 0: NAPI is now disabled
  223. * = 1: NAPI is now enabled
  224. */
  225. int hif_napi_event(struct hif_opaque_softc *hif_ctx, enum qca_napi_event event,
  226. void *data)
  227. {
  228. int rc;
  229. uint32_t prev_state;
  230. int i;
  231. struct napi_struct *napi;
  232. struct hif_softc *hif = HIF_GET_SOFTC(hif_ctx);
  233. NAPI_DEBUG("-->(event=%d, aux=%p)", event, data);
  234. mutex_lock(&(hif->napi_data.mutex));
  235. prev_state = hif->napi_data.state;
  236. switch (event) {
  237. case NAPI_EVT_INI_FILE:
  238. case NAPI_EVT_CMD_STATE: {
  239. int on = (data != ((void *)0));
  240. HIF_INFO("%s: received evnt: CONF %s; v = %d (state=0x%0x)\n",
  241. __func__,
  242. (event == NAPI_EVT_INI_FILE)?".ini file":"cmd",
  243. on, prev_state);
  244. if (on)
  245. if (prev_state & HIF_NAPI_CONF_UP) {
  246. HIF_INFO("%s: duplicate NAPI conf ON msg\n",
  247. __func__);
  248. } else {
  249. HIF_INFO("%s: setting configuration to ON\n",
  250. __func__);
  251. hif->napi_data.state |= HIF_NAPI_CONF_UP;
  252. }
  253. else /* off request */
  254. if (prev_state & HIF_NAPI_CONF_UP) {
  255. HIF_INFO("%s: setting configuration to OFF\n",
  256. __func__);
  257. hif->napi_data.state &= ~HIF_NAPI_CONF_UP;
  258. } else {
  259. HIF_INFO("%s: duplicate NAPI conf OFF msg\n",
  260. __func__);
  261. }
  262. break;
  263. }
  264. /* case NAPI_INIT_FILE/CMD_STATE */
  265. default: {
  266. HIF_ERROR("%s: unknown event: %d (data=0x%0lx)\n",
  267. __func__, event, (unsigned long) data);
  268. break;
  269. } /* default */
  270. }; /* switch */
  271. mutex_unlock(&(hif->napi_data.mutex));
  272. if (prev_state != hif->napi_data.state) {
  273. if (hif->napi_data.state == ENABLE_NAPI_MASK) {
  274. rc = 1;
  275. for (i = 0; i < CE_COUNT_MAX; i++)
  276. if ((hif->napi_data.ce_map & (0x01 << i))) {
  277. napi = &(hif->napi_data.napis[i].napi);
  278. NAPI_DEBUG("enabling NAPI %d", i);
  279. napi_enable(napi);
  280. }
  281. } else {
  282. rc = 0;
  283. for (i = 0; i < CE_COUNT_MAX; i++)
  284. if (hif->napi_data.ce_map & (0x01 << i)) {
  285. napi = &(hif->napi_data.napis[i].napi);
  286. NAPI_DEBUG("disabling NAPI %d", i);
  287. napi_disable(napi);
  288. }
  289. }
  290. } else {
  291. HIF_INFO("%s: no change in hif napi state (still %d)\n",
  292. __func__, prev_state);
  293. rc = (hif->napi_data.state == ENABLE_NAPI_MASK);
  294. }
  295. NAPI_DEBUG("<--[rc=%d]", rc);
  296. return rc;
  297. }
  298. /**
  299. * hif_napi_enabled() - checks whether NAPI is enabled for given ce or not
  300. * @hif: hif context
  301. * @ce : CE instance (or -1, to check if any CEs are enabled)
  302. *
  303. * Return: bool
  304. */
  305. int hif_napi_enabled(struct hif_opaque_softc *hif_ctx, int ce)
  306. {
  307. int rc;
  308. struct hif_softc *hif = HIF_GET_SOFTC(hif_ctx);
  309. if (-1 == ce)
  310. rc = ((hif->napi_data.state == ENABLE_NAPI_MASK));
  311. else
  312. rc = ((hif->napi_data.state == ENABLE_NAPI_MASK) &&
  313. (hif->napi_data.ce_map & (0x01 << ce)));
  314. return rc;
  315. };
  316. /**
  317. * hif_napi_enable_irq() - enables bus interrupts after napi_complete
  318. *
  319. * @hif: hif context
  320. * @id : id of NAPI instance calling this (used to determine the CE)
  321. *
  322. * Return: void
  323. */
  324. inline void hif_napi_enable_irq(struct hif_opaque_softc *hif, int id)
  325. {
  326. struct hif_softc *scn = HIF_GET_SOFTC(hif);
  327. ce_irq_enable(scn, NAPI_ID2PIPE(id));
  328. }
  329. /**
  330. * hif_napi_schedule() - schedules napi, updates stats
  331. * @scn: hif context
  332. * @ce_id: index of napi instance
  333. *
  334. * Return: void
  335. */
  336. int hif_napi_schedule(struct hif_opaque_softc *hif_ctx, int ce_id)
  337. {
  338. int cpu = smp_processor_id();
  339. struct hif_softc *scn = HIF_GET_SOFTC(hif_ctx);
  340. scn->napi_data.napis[ce_id].stats[cpu].napi_schedules++;
  341. NAPI_DEBUG("scheduling napi %d (ce:%d)",
  342. scn->napi_data.napis[ce_id].id, ce_id);
  343. napi_schedule(&(scn->napi_data.napis[ce_id].napi));
  344. return true;
  345. }
  346. /**
  347. * hif_napi_poll() - NAPI poll routine
  348. * @napi : pointer to NAPI struct as kernel holds it
  349. * @budget:
  350. *
  351. * This is the body of the poll function.
  352. * The poll function is called by kernel. So, there is a wrapper
  353. * function in HDD, which in turn calls this function.
  354. * Two main reasons why the whole thing is not implemented in HDD:
  355. * a) references to things like ce_service that HDD is not aware of
  356. * b) proximity to the implementation of ce_tasklet, which the body
  357. * of this function should be very close to.
  358. *
  359. * NOTE TO THE MAINTAINER:
  360. * Consider this function and ce_tasklet very tightly coupled pairs.
  361. * Any changes to ce_tasklet or this function may likely need to be
  362. * reflected in the counterpart.
  363. *
  364. * Returns:
  365. * int: the amount of work done in this poll ( <= budget)
  366. */
  367. int hif_napi_poll(struct hif_opaque_softc *hif_ctx, struct napi_struct *napi,
  368. int budget)
  369. {
  370. int rc = 0; /* default: no work done, also takes care of error */
  371. int normalized, bucket;
  372. int cpu = smp_processor_id();
  373. struct hif_softc *hif = HIF_GET_SOFTC(hif_ctx);
  374. struct qca_napi_info *napi_info;
  375. struct CE_state *ce_state;
  376. NAPI_DEBUG("%s -->(.., budget=%d)", budget);
  377. napi_info = (struct qca_napi_info *)
  378. container_of(napi, struct qca_napi_info, napi);
  379. napi_info->stats[cpu].napi_polls++;
  380. if (unlikely(NULL == hif))
  381. QDF_ASSERT(hif != NULL); /* emit a warning if hif NULL */
  382. else {
  383. rc = ce_per_engine_service(hif, NAPI_ID2PIPE(napi_info->id));
  384. HIF_INFO_HI("%s: ce_per_engine_service processed %d msgs",
  385. __func__, rc);
  386. }
  387. napi_info->stats[cpu].napi_workdone += rc;
  388. normalized = (rc / napi_info->scale);
  389. if (NULL != hif) {
  390. ce_state = hif->ce_id_to_state[NAPI_ID2PIPE(napi_info->id)];
  391. if (ce_state->lro_flush_cb != NULL) {
  392. ce_state->lro_flush_cb(ce_state->lro_data);
  393. }
  394. }
  395. /* do not return 0, if there was some work done,
  396. * even if it is below the scale
  397. */
  398. if (rc)
  399. normalized++;
  400. bucket = (normalized / QCA_NAPI_DEF_SCALE);
  401. napi_info->stats[cpu].napi_budget_uses[bucket]++;
  402. /* if ce_per engine reports 0, then poll should be terminated */
  403. if (0 == rc)
  404. NAPI_DEBUG("%s:%d: nothing processed by CE. Completing NAPI",
  405. __func__, __LINE__);
  406. if (rc <= HIF_NAPI_MAX_RECEIVES) {
  407. napi_info->stats[cpu].napi_completes++;
  408. /* enable interrupts */
  409. napi_complete(napi);
  410. if (NULL != hif) {
  411. hif_napi_enable_irq(hif_ctx, napi_info->id);
  412. /* support suspend/resume */
  413. qdf_atomic_dec(&(hif->active_tasklet_cnt));
  414. }
  415. NAPI_DEBUG("%s:%d: napi_complete + enabling the interrupts",
  416. __func__, __LINE__);
  417. }
  418. NAPI_DEBUG("%s <--[normalized=%d]", _func__, normalized);
  419. return normalized;
  420. }