|
@@ -105,6 +105,7 @@ int hdd_napi_create(void)
|
|
|
{
|
|
|
struct hif_opaque_softc *hif_ctx;
|
|
|
int rc = 0;
|
|
|
+ hdd_context_t *hdd_ctx;
|
|
|
|
|
|
NAPI_DEBUG("-->");
|
|
|
|
|
@@ -116,11 +117,20 @@ int hdd_napi_create(void)
|
|
|
rc = hif_napi_create(hif_ctx, hdd_napi_poll,
|
|
|
QCA_NAPI_BUDGET,
|
|
|
QCA_NAPI_DEF_SCALE);
|
|
|
- if (rc < 0)
|
|
|
+ if (rc < 0) {
|
|
|
hdd_err("ERR(%d) creating NAPI instances",
|
|
|
rc);
|
|
|
- else
|
|
|
+ } else {
|
|
|
hdd_info("napi instances were created. Map=0x%x", rc);
|
|
|
+ hdd_ctx = cds_get_context(QDF_MODULE_ID_HDD);
|
|
|
+ if (unlikely(NULL == hdd_ctx)) {
|
|
|
+ QDF_ASSERT( 0 );
|
|
|
+ rc = -EFAULT;
|
|
|
+ } else {
|
|
|
+ rc = hdd_napi_event(NAPI_EVT_INI_FILE,
|
|
|
+ (void *)hdd_ctx->napi_enable);
|
|
|
+ }
|
|
|
+ }
|
|
|
|
|
|
}
|
|
|
NAPI_DEBUG("<-- [rc=%d]", rc);
|
|
@@ -204,20 +214,16 @@ int hdd_napi_enabled(int id)
|
|
|
}
|
|
|
|
|
|
/**
|
|
|
- * hdd_napi_event() - relay the event detected by HDD to HIF NAPI decision maker
|
|
|
+ * hdd_napi_event() - relay the event detected by HDD to HIF NAPI event handler
|
|
|
* @event: event code
|
|
|
* @data : event-specific auxiliary data
|
|
|
*
|
|
|
- * Return code does not indicate a change, but whether or not NAPI is
|
|
|
- * enabled at the time of the return of the function. That is, if NAPI
|
|
|
- * was disabled before the call, and the event does not cause NAPI to be
|
|
|
- * enabled, a value of 0 will be returned indicating that it is (still)
|
|
|
- * disabled.
|
|
|
+ * See function documentation in hif_napi.c::hif_napi_event for list of events
|
|
|
+ * and how each of them is handled.
|
|
|
*
|
|
|
* Return:
|
|
|
* < 0: error code
|
|
|
- * = 0: NAPI state = disabled (after processing the event)
|
|
|
- * = 1: NAPI state = enabled (after processing the event)
|
|
|
+ * = 0: event handled successfully
|
|
|
*/
|
|
|
int hdd_napi_event(enum qca_napi_event event, void *data)
|
|
|
{
|
|
@@ -236,6 +242,61 @@ int hdd_napi_event(enum qca_napi_event event, void *data)
|
|
|
return rc;
|
|
|
}
|
|
|
|
|
|
+#ifdef HELIUMPLUS
|
|
|
+/**
|
|
|
+ * hdd_napi_apply_throughput_policy() - implement the throughput action policy
|
|
|
+ * @hddctx: HDD context
|
|
|
+ * @tx_packets: number of tx packets in the last interval
|
|
|
+ * @rx_packets: number of rx packets in the last interval
|
|
|
+ *
|
|
|
+ * Called by hdd_bus_bw_compute_cb, checks the number of packets in the last
|
|
|
+ * interval, and determines the desired napi throughput state (HI/LO). If
|
|
|
+ * the desired state is different from the current, then it invokes the
|
|
|
+ * event handler to switch to the desired state.
|
|
|
+ *
|
|
|
+ * The policy implementation is limited to this function and
|
|
|
+ * The current policy is: determine the NAPI mode based on the condition:
|
|
|
+ * (total number of packets > medium threshold)
|
|
|
+ * - tx packets are included because:
|
|
|
+ * a- tx-completions arrive at one of the rx CEs
|
|
|
+ * b- in TCP, a lof of TX implies ~(tx/2) rx (ACKs)
|
|
|
+ * c- so that we can use the same normalized criteria in ini file
|
|
|
+ * - medium-threshold (default: 500 packets / 10 ms), because
|
|
|
+ * we would like to be more reactive.
|
|
|
+ *
|
|
|
+ * Return: 0 : no action taken, or action return code
|
|
|
+ * !0: error, or action error code
|
|
|
+ */
|
|
|
+int hdd_napi_apply_throughput_policy(struct hdd_context_s *hddctx,
|
|
|
+ uint64_t tx_packets,
|
|
|
+ uint64_t rx_packets)
|
|
|
+{
|
|
|
+ int rc = 0;
|
|
|
+ uint64_t packets = tx_packets + rx_packets;
|
|
|
+ enum qca_napi_tput_state req_state;
|
|
|
+ struct qca_napi_data *napid = hdd_napi_get_all();
|
|
|
+ int enabled;
|
|
|
+
|
|
|
+ NAPI_DEBUG("-->%s(tx=%lld, rx=%lld)", __func__, tx_packets, rx_packets);
|
|
|
+
|
|
|
+ if ((napid != NULL) &&
|
|
|
+ (enabled = hdd_napi_enabled(HDD_NAPI_ANY))) {
|
|
|
+ if (packets > hddctx->config->busBandwidthHighThreshold)
|
|
|
+ req_state = QCA_NAPI_TPUT_HI;
|
|
|
+ else
|
|
|
+ req_state = QCA_NAPI_TPUT_LO;
|
|
|
+
|
|
|
+ if (req_state != napid->napi_mode)
|
|
|
+ rc = hdd_napi_event(NAPI_EVT_TPUT_STATE,
|
|
|
+ (void *)req_state);
|
|
|
+ } else {
|
|
|
+ hdd_err("ERR: napid (%p) NULL or napi_enabled (%d) FALSE",
|
|
|
+ napid, enabled);
|
|
|
+ }
|
|
|
+ return rc;
|
|
|
+}
|
|
|
+#endif
|
|
|
+
|
|
|
/**
|
|
|
* hdd_napi_poll() - NAPI poll function
|
|
|
* @napi : pointer to NAPI struct
|