Bläddra i källkod

qcacmn: Genoa: SDIO: Handle Rx Padding

In Genoa SDIO ADMA implementation FW sends bundled packets
which is terminated by a pad block

Handle the Rx Bundling using the pad block

CRs-Fixed: 2506247
Change-Id: I50f1c79a07398ed0fcfba358fc53aa02ad3c6964
Visweswara Tanuku 5 år sedan
förälder
incheckning
8c730b08c0
1 ändrade filer med 39 tillägg och 68 borttagningar
  1. 39 68
      hif/src/sdio/transfer/adma.c

+ 39 - 68
hif/src/sdio/transfer/adma.c

@@ -259,8 +259,6 @@ int hif_get_send_address(struct hif_sdio_device *pdev,
 {
 	struct sdio_al_channel_handle *chan = NULL;
 
-	HIF_INFO("pipe: %u", pipe);
-
 	if (!addr)
 		return -EINVAL;
 
@@ -287,7 +285,6 @@ void hif_fixup_write_param(struct hif_sdio_dev *pdev, uint32_t req,
 			   uint32_t *length, uint32_t *addr)
 {
 	HIF_ENTER();
-	/* ADMA-TODO */
 	HIF_EXIT();
 }
 
@@ -376,9 +373,10 @@ QDF_STATUS hif_enable_func(struct hif_softc *ol_sc, struct hif_sdio_dev *device,
  *
  * Return - NULL if out of buffers, else qdf_nbuf_t
  */
-#define HEAD_ROOM 256
-#define len_head_room(len) ((len) + HEAD_ROOM)
-#define is_pad_block(buf) (*((uint32_t *)buf) == 0xbabababa)
+#define HEAD_ROOM		256
+#define RESERVE_LEN		256
+#define is_pad_block(buf)	(*((uint32_t *)buf) == 0xbabababa)
+
 #if HIF_MAX_RX_Q_ALLOC
 qdf_nbuf_t hif_sdio_get_nbuf(struct hif_sdio_dev *dev)
 {
@@ -414,7 +412,7 @@ qdf_nbuf_t hif_sdio_get_nbuf(struct hif_sdio_dev *dev)
 {
 	qdf_nbuf_t nbuf;
 
-	nbuf = qdf_nbuf_alloc(NULL, HIF_SDIO_RX_BUFFER_SIZE + HEAD_ROOM,
+	nbuf = qdf_nbuf_alloc(NULL, HIF_SDIO_RX_BUFFER_SIZE + RESERVE_LEN,
 			      HEAD_ROOM, 4, 0);
 	return nbuf;
 }
@@ -443,7 +441,7 @@ void hif_sdio_rx_q_alloc(void *ctx)
 		}
 
 		rx_q_elem->nbuf = qdf_nbuf_alloc(NULL, HIF_SDIO_RX_BUFFER_SIZE +
-						 HEAD_ROOM, HEAD_ROOM, 4, 0);
+						 RESERVE_LEN, HEAD_ROOM, 4, 0);
 		if (!rx_q_elem->nbuf) {
 			HIF_ERROR("%s: failed to alloc nbuf for rx", __func__);
 			qdf_mem_free(rx_q_elem);
@@ -622,12 +620,6 @@ hif_read_write(struct hif_sdio_dev *dev,
 	/* Request SDIO AL to do transfer */
 	dir = (request & HIF_SDIO_WRITE) ? SDIO_AL_TX : SDIO_AL_RX;
 
-	if (request & HIF_SDIO_WRITE)
-		HIF_TRACE("%s: Tx len %d", __func__, length);
-
-	if (request & HIF_SDIO_READ)
-		HIF_TRACE("%s: Rx len %d", __func__, length);
-
 	if (request & HIF_SYNCHRONOUS) {
 		ret = sdio_al_queue_transfer(ch,
 					     dir,
@@ -673,7 +665,7 @@ hif_read_write(struct hif_sdio_dev *dev,
 }
 
 /**
- * ul_xfer_cb() - Completion call back for asyncronous transfer
+ * ul_xfer_cb() - Completion call back for asynchronous transfer
  * @ch_handle: The sdio al channel handle
  * @result: The result of the operation
  * @context: pointer to request context
@@ -719,7 +711,7 @@ void ul_xfer_cb(struct sdio_al_channel_handle *ch_handle,
  * completing READ in the transfer done callback later which
  * runs in sdio al thread context. If we do the syncronous
  * transfer here, the thread context won't be available and
- * perhaps a new thread may be reaquired here.
+ * perhaps a new thread may be required here.
  */
 void dl_data_avail_cb(struct sdio_al_channel_handle *ch_handle,
 		      unsigned int len)
@@ -747,7 +739,7 @@ void dl_data_avail_cb(struct sdio_al_channel_handle *ch_handle,
 	 * processed in the transfer done callback.
 	 */
 	/* TODO, use global buffer instead of runtime allocations */
-	nbuf = qdf_nbuf_alloc(NULL, len_head_room(len), HEAD_ROOM, 4, 0);
+	nbuf = qdf_nbuf_alloc(NULL, len + RESERVE_LEN, HEAD_ROOM, 4, 0);
 
 	if (!nbuf) {
 		HIF_ERROR("%s: Unable to alloc netbuf %u bytes", __func__, len);
@@ -772,19 +764,21 @@ void dl_xfer_cb(struct sdio_al_channel_handle *ch_handle,
 {
 	unsigned char *buf;
 	qdf_nbuf_t nbuf;
-	uint32_t len, nbuflen, payload_len = 0;
-	uint8_t *nbufdata;
+	uint32_t len, payload_len = 0;
 	struct hif_sdio_dev *dev;
 	struct hif_sdio_device *device;
 	struct bus_request *bus_req = (struct bus_request *)ctx;
-	bool last_htc_packet = false;
 	QDF_STATUS (*rx_completion)(void *, qdf_nbuf_t, uint8_t);
 
-	if (!ctx)
-		HIF_ERROR("%s: Net buf context NULL!!!", __func__);
+	if (!bus_req) {
+		HIF_ERROR("%s: Bus Req NULL!!!", __func__);
+		qdf_assert_always(0);
+		return;
+	}
 
-	if (!ctx || !ch_handle || !result) {
-		HIF_ERROR("%s: Invalid args", __func__);
+	if (!ch_handle || !result) {
+		HIF_ERROR("%s: Invalid args %pK %pK", __func__,
+			  ch_handle, result);
 		qdf_assert_always(0);
 		return;
 	}
@@ -793,10 +787,8 @@ void dl_xfer_cb(struct sdio_al_channel_handle *ch_handle,
 	if (result->xfer_status) {
 		HIF_ERROR("%s: ASYNC Rx failed %d", __func__,
 			  result->xfer_status);
-		/* TODO - Free nbuf if hif_sdio_get_nbuf is used, when bundling
-		 * is enabled
-		 */
-		hif_free_bus_request(dev, (struct bus_request *)ctx);
+		qdf_nbuf_free((qdf_nbuf_t)bus_req->context);
+		hif_free_bus_request(dev, bus_req);
 		return;
 	}
 
@@ -806,57 +798,45 @@ void dl_xfer_cb(struct sdio_al_channel_handle *ch_handle,
 	buf = (unsigned char *)result->buf_addr;
 	len = (unsigned int)result->xfer_len;
 
-	/* ADMA-TODO - Discard the padding data
-	 * Its still not decided that the padding is informed
-	 * via the hdr->flags or a padding magic inline in the
-	 * buffer. So, lets see.
-	 */
 	while (len > 0 && len >= sizeof(HTC_FRAME_HDR)) {
-		if (last_htc_packet) {
-			HIF_ERROR("ERRR last htc_packet processed already\n");
+		if (is_pad_block(buf)) {
+			/* End of Rx Buffer */
 			break;
 		}
+
 		if (HTC_GET_FIELD(buf, HTC_FRAME_HDR, ENDPOINTID) >=
 		    ENDPOINT_MAX) {
 			HIF_ERROR("%s: invalid endpoint id: %u", __func__,
 				  HTC_GET_FIELD(buf, HTC_FRAME_HDR,
 						ENDPOINTID));
-			hif_free_bus_request(dev, (struct bus_request *)ctx);
-			return;
+			break;
 		}
-		/* last_htc_packet is currently used to test non bundling case
-		 * on Rumi Emulation platform.
-		 * TODO - Remove last_htc_packet logic when bundling is
-		 * enabled and use is_pad_block for bundling
-		 */
-		last_htc_packet = 1;
-		/* TODO - get net buf using hif_sdio_get_nbuf, when bundling is
-		 * enabled
-		 */
-		nbuf = (qdf_nbuf_t)bus_req->context;
+
+		nbuf = hif_sdio_get_nbuf(dev);
 		if (!nbuf) {
 			HIF_ERROR("%s: failed to alloc rx buffer", __func__);
 			break;
 		}
-		nbufdata = qdf_nbuf_data(nbuf);
-		nbuflen = qdf_nbuf_len(nbuf);
 
 		/* Copy the HTC frame to the alloc'd packet buffer */
 		payload_len = HTC_GET_FIELD(buf, HTC_FRAME_HDR, PAYLOADLEN);
 		if (!payload_len) {
 			HIF_ERROR("%s:Invalid Payload len %d bytes", __func__,
 				  payload_len);
+			qdf_nbuf_free(nbuf);
 			break;
 		}
 
-		/* TODO - Check if payload fits in netbuf data */
-		if (0) { //nbuflen < payload_len) {
-			HIF_ERROR("%s: nbuf %d < payload %d bytes", __func__,
-				  nbuflen, payload_len);
+		/* Check if payload fits in skb */
+		if (qdf_nbuf_tailroom(nbuf) < payload_len + HTC_HEADER_LEN) {
+			HIF_ERROR("%s: Payload + HTC_HDR %d > skb tailroom %d",
+				  __func__, (payload_len + 8),
+				  qdf_nbuf_tailroom(nbuf));
+			qdf_nbuf_free(nbuf);
 			break;
 		}
 
-		qdf_mem_copy(nbufdata, buf,
+		qdf_mem_copy((uint8_t *)qdf_nbuf_data(nbuf), buf,
 			     payload_len + HTC_HEADER_LEN);
 
 		qdf_nbuf_set_pktlen(nbuf, payload_len + HTC_HDR_LENGTH);
@@ -864,19 +844,10 @@ void dl_xfer_cb(struct sdio_al_channel_handle *ch_handle,
 			      nbuf,
 			      0); /* don't care, not used */
 
-		if (len && last_htc_packet) {
-			unsigned int pad;
-
-			pad = DEV_CALC_RECV_PADDED_LEN(device, payload_len +
-						       HTC_HEADER_LEN);
-			/* Decrement the length to be processed yet */
-			len -= pad;
-			/* Move the data pointer */
-			buf += pad;
-		} else {
-			len -= payload_len + HTC_HDR_LENGTH;
-			buf += payload_len + HTC_HDR_LENGTH;
-		}
+		len -= payload_len + HTC_HDR_LENGTH;
+		buf += payload_len + HTC_HDR_LENGTH;
 	}
-	hif_free_bus_request(dev, (struct bus_request *)ctx);
+
+	qdf_nbuf_free((qdf_nbuf_t)bus_req->context);
+	hif_free_bus_request(dev, bus_req);
 }