|
@@ -53,14 +53,16 @@ static struct qca_napi_data *hdd_napi_ctx;
|
|
|
*/
|
|
|
struct qca_napi_data *hdd_napi_get_all(void)
|
|
|
{
|
|
|
- struct qca_napi_data *rp;
|
|
|
+ struct qca_napi_data *rp = NULL;
|
|
|
struct ol_softc *hif;
|
|
|
|
|
|
NAPI_DEBUG("-->\n");
|
|
|
|
|
|
hif = cds_get_context(CDF_MODULE_ID_HIF);
|
|
|
- CDF_ASSERT(hif != NULL);
|
|
|
- rp = hif_napi_get_all(hif);
|
|
|
+ if (unlikely(NULL == hif))
|
|
|
+ CDF_ASSERT(NULL != hif); /* WARN */
|
|
|
+ else
|
|
|
+ rp = hif_napi_get_all(hif);
|
|
|
|
|
|
NAPI_DEBUG("<-- [addr=%p]\n", rp);
|
|
|
return rp;
|
|
@@ -74,13 +76,14 @@ struct qca_napi_data *hdd_napi_get_all(void)
|
|
|
*/
|
|
|
static uint32_t hdd_napi_get_map(void)
|
|
|
{
|
|
|
- uint32_t map;
|
|
|
+ uint32_t map = 0;
|
|
|
|
|
|
NAPI_DEBUG("-->\n");
|
|
|
/* cache once, use forever */
|
|
|
if (hdd_napi_ctx == NULL)
|
|
|
hdd_napi_ctx = hdd_napi_get_all();
|
|
|
- map = hdd_napi_ctx->ce_map;
|
|
|
+ if (hdd_napi_ctx != NULL)
|
|
|
+ map = hdd_napi_ctx->ce_map;
|
|
|
|
|
|
NAPI_DEBUG("<--[map=0x%08x]\n", map);
|
|
|
return map;
|
|
@@ -105,35 +108,36 @@ int hdd_napi_create(void)
|
|
|
int ul_polled, dl_polled;
|
|
|
int rc = 0;
|
|
|
|
|
|
- hif_ctx = cds_get_context(CDF_MODULE_ID_HIF);
|
|
|
- CDF_ASSERT(hif_ctx != NULL);
|
|
|
-
|
|
|
NAPI_DEBUG("-->\n");
|
|
|
|
|
|
- /*
|
|
|
- * hif_service_to_pipe currently returns one pipe id per service.
|
|
|
- * However for Adrestea, we will need to figure out how to map
|
|
|
- * DATA service to multiple pipes. Either we will need to create
|
|
|
- * family of services, or this function may return a bitmap (of
|
|
|
- * at least #MAX_CEs bits) and then we will need to iterate on
|
|
|
- * the bitmap for hif_napi_create() calls.
|
|
|
- * For Rome, there is only one service, so there is a single call
|
|
|
- * TODO: clarify for multi-queue/Adrestea
|
|
|
- */
|
|
|
- if (CDF_STATUS_SUCCESS !=
|
|
|
- hif_map_service_to_pipe(hif_ctx, HTT_DATA_MSG_SVC,
|
|
|
- &ul, &dl, &ul_polled, &dl_polled)) {
|
|
|
- hdd_err("cannot map service to pipe");
|
|
|
- rc = -EINVAL;
|
|
|
+ hif_ctx = cds_get_context(CDF_MODULE_ID_HIF);
|
|
|
+ if (unlikely(NULL == hif_ctx)) {
|
|
|
+ CDF_ASSERT(NULL != hif_ctx);
|
|
|
+ rc = -EFAULT;
|
|
|
} else {
|
|
|
- rc = hif_napi_create(hif_ctx, dl, hdd_napi_poll,
|
|
|
- QCA_NAPI_BUDGET, QCA_NAPI_DEF_SCALE);
|
|
|
- if (rc < 0)
|
|
|
- hdd_err("ERR(%d) creating NAPI on pipe %d", rc, dl);
|
|
|
- else {
|
|
|
- hdd_info("napi instance %d created on pipe %d",
|
|
|
- rc, dl);
|
|
|
- /* rc = (0x01 << rc); -- phase 2 */
|
|
|
+ /*
|
|
|
+ * Note: hif_service_to_pipe returns one pipe id per service.
|
|
|
+ * For multi-queue NAPI for Adrastea, we will use multiple
|
|
|
+ * services/calls.
|
|
|
+ * For Rome, there is only one service, hence a single call
|
|
|
+ */
|
|
|
+ if (CDF_STATUS_SUCCESS !=
|
|
|
+ hif_map_service_to_pipe(hif_ctx, HTT_DATA_MSG_SVC,
|
|
|
+ &ul, &dl, &ul_polled, &dl_polled)) {
|
|
|
+ hdd_err("cannot map service to pipe");
|
|
|
+ rc = -EINVAL;
|
|
|
+ } else {
|
|
|
+ rc = hif_napi_create(hif_ctx, dl, hdd_napi_poll,
|
|
|
+ QCA_NAPI_BUDGET,
|
|
|
+ QCA_NAPI_DEF_SCALE);
|
|
|
+ if (rc < 0)
|
|
|
+ hdd_err("ERR(%d) creating NAPI on pipe %d",
|
|
|
+ rc, dl);
|
|
|
+ else {
|
|
|
+ hdd_info("napi instance %d created on pipe %d",
|
|
|
+ rc, dl);
|
|
|
+ /* rc = (0x01 << rc); -- phase 2 */
|
|
|
+ }
|
|
|
}
|
|
|
}
|
|
|
NAPI_DEBUG("<-- [rc=%d]\n", rc);
|
|
@@ -163,19 +167,21 @@ int hdd_napi_destroy(int force)
|
|
|
struct ol_softc *hif_ctx;
|
|
|
|
|
|
hif_ctx = cds_get_context(CDF_MODULE_ID_HIF);
|
|
|
- CDF_ASSERT(hif_ctx != NULL);
|
|
|
-
|
|
|
- for (i = 0; i < (sizeof(uint32_t)*8); i++)
|
|
|
- if (hdd_napi_map & (0x01 << i)) {
|
|
|
- if (0 <= hif_napi_destroy(hif_ctx,
|
|
|
- NAPI_PIPE2ID(i),
|
|
|
- force)) {
|
|
|
- rc++;
|
|
|
- hdd_napi_map &= ~(0x01 << i);
|
|
|
- } else
|
|
|
- hdd_err("cannot destroy napi inst %d: (pipe:%d), f=%d\n",
|
|
|
- i, NAPI_PIPE2ID(i), force);
|
|
|
- }
|
|
|
+ if (unlikely(NULL == hif_ctx))
|
|
|
+ CDF_ASSERT(NULL != hif_ctx);
|
|
|
+ else
|
|
|
+ for (i = 0; i < CE_COUNT_MAX; i++)
|
|
|
+ if (hdd_napi_map & (0x01 << i)) {
|
|
|
+ if (0 <= hif_napi_destroy(
|
|
|
+ hif_ctx,
|
|
|
+ NAPI_PIPE2ID(i), force)) {
|
|
|
+ rc++;
|
|
|
+ hdd_napi_map &= ~(0x01 << i);
|
|
|
+ } else
|
|
|
+ hdd_err("cannot destroy napi %d: (pipe:%d), f=%d\n",
|
|
|
+ i,
|
|
|
+ NAPI_PIPE2ID(i), force);
|
|
|
+ }
|
|
|
}
|
|
|
|
|
|
/* if all instances are removed, it is likely that hif_context has been
|
|
@@ -199,16 +205,19 @@ int hdd_napi_destroy(int force)
|
|
|
* int: 0 = false (NOT enabled)
|
|
|
* !0 = true (enabbled)
|
|
|
*/
|
|
|
-inline int hdd_napi_enabled(int id)
|
|
|
+int hdd_napi_enabled(int id)
|
|
|
{
|
|
|
struct ol_softc *hif;
|
|
|
+ int rc = 0; /* NOT enabled */
|
|
|
|
|
|
hif = cds_get_context(CDF_MODULE_ID_HIF);
|
|
|
- CDF_ASSERT(hif != NULL);
|
|
|
- if (-1 == id)
|
|
|
- return hif_napi_enabled(hif, id);
|
|
|
+ if (unlikely(NULL == hif))
|
|
|
+ CDF_ASSERT(hif != NULL); /* WARN_ON; rc = 0 */
|
|
|
+ else if (-1 == id)
|
|
|
+ rc = hif_napi_enabled(hif, id);
|
|
|
else
|
|
|
- return hif_napi_enabled(hif, NAPI_ID2PIPE(id));
|
|
|
+ rc = hif_napi_enabled(hif, NAPI_ID2PIPE(id));
|
|
|
+ return rc;
|
|
|
}
|
|
|
|
|
|
/**
|
|
@@ -229,14 +238,16 @@ inline int hdd_napi_enabled(int id)
|
|
|
*/
|
|
|
int hdd_napi_event(enum qca_napi_event event, void *data)
|
|
|
{
|
|
|
- int rc;
|
|
|
+ int rc = -EFAULT; /* assume err */
|
|
|
struct ol_softc *hif;
|
|
|
|
|
|
NAPI_DEBUG("-->(event=%d, aux=%p)\n", event, data);
|
|
|
|
|
|
hif = cds_get_context(CDF_MODULE_ID_HIF);
|
|
|
- CDF_ASSERT(hif != NULL);
|
|
|
- rc = hif_napi_event(hif, event, data);
|
|
|
+ if (unlikely(NULL == hif))
|
|
|
+ CDF_ASSERT(hif != NULL);
|
|
|
+ else
|
|
|
+ rc = hif_napi_event(hif, event, data);
|
|
|
|
|
|
NAPI_DEBUG("<--[rc=%d]\n", rc);
|
|
|
return rc;
|