Author | Tokens | Token Proportion | Commits | Commit Proportion |
---|---|---|---|---|
Haijun Liu | 5822 | 99.97% | 3 | 75.00% |
Ricardo Martinez | 2 | 0.03% | 1 | 25.00% |
Total | 5824 | 4 |
// SPDX-License-Identifier: GPL-2.0-only /* * Copyright (c) 2021, MediaTek Inc. * Copyright (c) 2021-2022, Intel Corporation. * * Authors: * Amir Hanania <amir.hanania@intel.com> * Haijun Liu <haijun.liu@mediatek.com> * Eliot Lee <eliot.lee@intel.com> * Moises Veleta <moises.veleta@intel.com> * Ricardo Martinez <ricardo.martinez@linux.intel.com> * * Contributors: * Andy Shevchenko <andriy.shevchenko@linux.intel.com> * Chiranjeevi Rapolu <chiranjeevi.rapolu@intel.com> * Sreehari Kancharla <sreehari.kancharla@intel.com> */ #include <linux/atomic.h> #include <linux/bitfield.h> #include <linux/bitops.h> #include <linux/device.h> #include <linux/dma-direction.h> #include <linux/dma-mapping.h> #include <linux/gfp.h> #include <linux/err.h> #include <linux/iopoll.h> #include <linux/jiffies.h> #include <linux/kernel.h> #include <linux/kthread.h> #include <linux/list.h> #include <linux/minmax.h> #include <linux/mm.h> #include <linux/netdevice.h> #include <linux/pm_runtime.h> #include <linux/sched.h> #include <linux/skbuff.h> #include <linux/slab.h> #include <linux/spinlock.h> #include <linux/string.h> #include <linux/types.h> #include <linux/wait.h> #include <linux/workqueue.h> #include "t7xx_dpmaif.h" #include "t7xx_hif_dpmaif.h" #include "t7xx_hif_dpmaif_rx.h" #include "t7xx_pci.h" #define DPMAIF_BAT_COUNT 8192 #define DPMAIF_FRG_COUNT 4814 #define DPMAIF_PIT_COUNT (DPMAIF_BAT_COUNT * 2) #define DPMAIF_BAT_CNT_THRESHOLD 30 #define DPMAIF_PIT_CNT_THRESHOLD 60 #define DPMAIF_RX_PUSH_THRESHOLD_MASK GENMASK(2, 0) #define DPMAIF_NOTIFY_RELEASE_COUNT 128 #define DPMAIF_POLL_PIT_TIME_US 20 #define DPMAIF_POLL_PIT_MAX_TIME_US 2000 #define DPMAIF_WQ_TIME_LIMIT_MS 2 #define DPMAIF_CS_RESULT_PASS 0 /* Packet type */ #define DES_PT_PD 0 #define DES_PT_MSG 1 /* Buffer type */ #define PKT_BUF_FRAG 1 static unsigned int t7xx_normal_pit_bid(const struct dpmaif_pit *pit_info) { u32 value; value = FIELD_GET(PD_PIT_H_BID, le32_to_cpu(pit_info->pd.footer)); value <<= 13; value += FIELD_GET(PD_PIT_BUFFER_ID, le32_to_cpu(pit_info->header)); return value; } static int t7xx_dpmaif_net_rx_push_thread(void *arg) { struct dpmaif_rx_queue *q = arg; struct dpmaif_ctrl *hif_ctrl; struct dpmaif_callbacks *cb; hif_ctrl = q->dpmaif_ctrl; cb = hif_ctrl->callbacks; while (!kthread_should_stop()) { struct sk_buff *skb; unsigned long flags; if (skb_queue_empty(&q->skb_list)) { if (wait_event_interruptible(q->rx_wq, !skb_queue_empty(&q->skb_list) || kthread_should_stop())) continue; if (kthread_should_stop()) break; } spin_lock_irqsave(&q->skb_list.lock, flags); skb = __skb_dequeue(&q->skb_list); spin_unlock_irqrestore(&q->skb_list.lock, flags); if (!skb) continue; cb->recv_skb(hif_ctrl->t7xx_dev, skb); cond_resched(); } return 0; } static int t7xx_dpmaif_update_bat_wr_idx(struct dpmaif_ctrl *dpmaif_ctrl, const unsigned int q_num, const unsigned int bat_cnt) { struct dpmaif_rx_queue *rxq = &dpmaif_ctrl->rxq[q_num]; struct dpmaif_bat_request *bat_req = rxq->bat_req; unsigned int old_rl_idx, new_wr_idx, old_wr_idx; if (!rxq->que_started) { dev_err(dpmaif_ctrl->dev, "RX queue %d has not been started\n", rxq->index); return -EINVAL; } old_rl_idx = bat_req->bat_release_rd_idx; old_wr_idx = bat_req->bat_wr_idx; new_wr_idx = old_wr_idx + bat_cnt; if (old_rl_idx > old_wr_idx && new_wr_idx >= old_rl_idx) goto err_flow; if (new_wr_idx >= bat_req->bat_size_cnt) { new_wr_idx -= bat_req->bat_size_cnt; if (new_wr_idx >= old_rl_idx) goto err_flow; } bat_req->bat_wr_idx = new_wr_idx; return 0; err_flow: dev_err(dpmaif_ctrl->dev, "RX BAT flow check fail\n"); return -EINVAL; } static bool t7xx_alloc_and_map_skb_info(const struct dpmaif_ctrl *dpmaif_ctrl, const unsigned int size, struct dpmaif_bat_skb *cur_skb) { dma_addr_t data_bus_addr; struct sk_buff *skb; skb = __dev_alloc_skb(size, GFP_KERNEL); if (!skb) return false; data_bus_addr = dma_map_single(dpmaif_ctrl->dev, skb->data, size, DMA_FROM_DEVICE); if (dma_mapping_error(dpmaif_ctrl->dev, data_bus_addr)) { dev_err_ratelimited(dpmaif_ctrl->dev, "DMA mapping error\n"); dev_kfree_skb_any(skb); return false; } cur_skb->skb = skb; cur_skb->data_bus_addr = data_bus_addr; cur_skb->data_len = size; return true; } static void t7xx_unmap_bat_skb(struct device *dev, struct dpmaif_bat_skb *bat_skb_base, unsigned int index) { struct dpmaif_bat_skb *bat_skb = bat_skb_base + index; if (bat_skb->skb) { dma_unmap_single(dev, bat_skb->data_bus_addr, bat_skb->data_len, DMA_FROM_DEVICE); dev_kfree_skb(bat_skb->skb); bat_skb->skb = NULL; } } /** * t7xx_dpmaif_rx_buf_alloc() - Allocate buffers for the BAT ring. * @dpmaif_ctrl: Pointer to DPMAIF context structure. * @bat_req: Pointer to BAT request structure. * @q_num: Queue number. * @buf_cnt: Number of buffers to allocate. * @initial: Indicates if the ring is being populated for the first time. * * Allocate skb and store the start address of the data buffer into the BAT ring. * If this is not the initial call, notify the HW about the new entries. * * Return: * * 0 - Success. * * -ERROR - Error code. */ int t7xx_dpmaif_rx_buf_alloc(struct dpmaif_ctrl *dpmaif_ctrl, const struct dpmaif_bat_request *bat_req, const unsigned int q_num, const unsigned int buf_cnt, const bool initial) { unsigned int i, bat_cnt, bat_max_cnt, bat_start_idx; int ret; if (!buf_cnt || buf_cnt > bat_req->bat_size_cnt) return -EINVAL; /* Check BAT buffer space */ bat_max_cnt = bat_req->bat_size_cnt; bat_cnt = t7xx_ring_buf_rd_wr_count(bat_max_cnt, bat_req->bat_release_rd_idx, bat_req->bat_wr_idx, DPMAIF_WRITE); if (buf_cnt > bat_cnt) return -ENOMEM; bat_start_idx = bat_req->bat_wr_idx; for (i = 0; i < buf_cnt; i++) { unsigned int cur_bat_idx = bat_start_idx + i; struct dpmaif_bat_skb *cur_skb; struct dpmaif_bat *cur_bat; if (cur_bat_idx >= bat_max_cnt) cur_bat_idx -= bat_max_cnt; cur_skb = (struct dpmaif_bat_skb *)bat_req->bat_skb + cur_bat_idx; if (!cur_skb->skb && !t7xx_alloc_and_map_skb_info(dpmaif_ctrl, bat_req->pkt_buf_sz, cur_skb)) break; cur_bat = (struct dpmaif_bat *)bat_req->bat_base + cur_bat_idx; cur_bat->buffer_addr_ext = upper_32_bits(cur_skb->data_bus_addr); cur_bat->p_buffer_addr = lower_32_bits(cur_skb->data_bus_addr); } if (!i) return -ENOMEM; ret = t7xx_dpmaif_update_bat_wr_idx(dpmaif_ctrl, q_num, i); if (ret) goto err_unmap_skbs; if (!initial) { unsigned int hw_wr_idx; ret = t7xx_dpmaif_dl_snd_hw_bat_cnt(&dpmaif_ctrl->hw_info, i); if (ret) goto err_unmap_skbs; hw_wr_idx = t7xx_dpmaif_dl_get_bat_wr_idx(&dpmaif_ctrl->hw_info, DPF_RX_QNO_DFT); if (hw_wr_idx != bat_req->bat_wr_idx) { ret = -EFAULT; dev_err(dpmaif_ctrl->dev, "Write index mismatch in RX ring\n"); goto err_unmap_skbs; } } return 0; err_unmap_skbs: while (--i > 0) t7xx_unmap_bat_skb(dpmaif_ctrl->dev, bat_req->bat_skb, i); return ret; } static int t7xx_dpmaifq_release_pit_entry(struct dpmaif_rx_queue *rxq, const unsigned int rel_entry_num) { struct dpmaif_hw_info *hw_info = &rxq->dpmaif_ctrl->hw_info; unsigned int old_rel_idx, new_rel_idx, hw_wr_idx; int ret; if (!rxq->que_started) return 0; if (rel_entry_num >= rxq->pit_size_cnt) { dev_err(rxq->dpmaif_ctrl->dev, "Invalid PIT release index\n"); return -EINVAL; } old_rel_idx = rxq->pit_release_rd_idx; new_rel_idx = old_rel_idx + rel_entry_num; hw_wr_idx = rxq->pit_wr_idx; if (hw_wr_idx < old_rel_idx && new_rel_idx >= rxq->pit_size_cnt) new_rel_idx -= rxq->pit_size_cnt; ret = t7xx_dpmaif_dlq_add_pit_remain_cnt(hw_info, rxq->index, rel_entry_num); if (ret) { dev_err(rxq->dpmaif_ctrl->dev, "PIT release failure: %d\n", ret); return ret; } rxq->pit_release_rd_idx = new_rel_idx; return 0; } static void t7xx_dpmaif_set_bat_mask(struct dpmaif_bat_request *bat_req, unsigned int idx) { unsigned long flags; spin_lock_irqsave(&bat_req->mask_lock, flags); set_bit(idx, bat_req->bat_bitmap); spin_unlock_irqrestore(&bat_req->mask_lock, flags); } static int t7xx_frag_bat_cur_bid_check(struct dpmaif_rx_queue *rxq, const unsigned int cur_bid) { struct dpmaif_bat_request *bat_frag = rxq->bat_frag; struct dpmaif_bat_page *bat_page; if (cur_bid >= DPMAIF_FRG_COUNT) return -EINVAL; bat_page = bat_frag->bat_skb + cur_bid; if (!bat_page->page) return -EINVAL; return 0; } static void t7xx_unmap_bat_page(struct device *dev, struct dpmaif_bat_page *bat_page_base, unsigned int index) { struct dpmaif_bat_page *bat_page = bat_page_base + index; if (bat_page->page) { dma_unmap_page(dev, bat_page->data_bus_addr, bat_page->data_len, DMA_FROM_DEVICE); put_page(bat_page->page); bat_page->page = NULL; } } /** * t7xx_dpmaif_rx_frag_alloc() - Allocates buffers for the Fragment BAT ring. * @dpmaif_ctrl: Pointer to DPMAIF context structure. * @bat_req: Pointer to BAT request structure. * @buf_cnt: Number of buffers to allocate. * @initial: Indicates if the ring is being populated for the first time. * * Fragment BAT is used when the received packet does not fit in a normal BAT entry. * This function allocates a page fragment and stores the start address of the page * into the Fragment BAT ring. * If this is not the initial call, notify the HW about the new entries. * * Return: * * 0 - Success. * * -ERROR - Error code. */ int t7xx_dpmaif_rx_frag_alloc(struct dpmaif_ctrl *dpmaif_ctrl, struct dpmaif_bat_request *bat_req, const unsigned int buf_cnt, const bool initial) { unsigned int buf_space, cur_bat_idx = bat_req->bat_wr_idx; struct dpmaif_bat_page *bat_skb = bat_req->bat_skb; int ret = 0, i; if (!buf_cnt || buf_cnt > bat_req->bat_size_cnt) return -EINVAL; buf_space = t7xx_ring_buf_rd_wr_count(bat_req->bat_size_cnt, bat_req->bat_release_rd_idx, bat_req->bat_wr_idx, DPMAIF_WRITE); if (buf_cnt > buf_space) { dev_err(dpmaif_ctrl->dev, "Requested more buffers than the space available in RX frag ring\n"); return -EINVAL; } for (i = 0; i < buf_cnt; i++) { struct dpmaif_bat_page *cur_page = bat_skb + cur_bat_idx; struct dpmaif_bat *cur_bat; dma_addr_t data_base_addr; if (!cur_page->page) { unsigned long offset; struct page *page; void *data; data = netdev_alloc_frag(bat_req->pkt_buf_sz); if (!data) break; page = virt_to_head_page(data); offset = data - page_address(page); data_base_addr = dma_map_page(dpmaif_ctrl->dev, page, offset, bat_req->pkt_buf_sz, DMA_FROM_DEVICE); if (dma_mapping_error(dpmaif_ctrl->dev, data_base_addr)) { put_page(virt_to_head_page(data)); dev_err(dpmaif_ctrl->dev, "DMA mapping fail\n"); break; } cur_page->page = page; cur_page->data_bus_addr = data_base_addr; cur_page->offset = offset; cur_page->data_len = bat_req->pkt_buf_sz; } data_base_addr = cur_page->data_bus_addr; cur_bat = (struct dpmaif_bat *)bat_req->bat_base + cur_bat_idx; cur_bat->buffer_addr_ext = upper_32_bits(data_base_addr); cur_bat->p_buffer_addr = lower_32_bits(data_base_addr); cur_bat_idx = t7xx_ring_buf_get_next_wr_idx(bat_req->bat_size_cnt, cur_bat_idx); } bat_req->bat_wr_idx = cur_bat_idx; if (!initial) t7xx_dpmaif_dl_snd_hw_frg_cnt(&dpmaif_ctrl->hw_info, i); if (i < buf_cnt) { ret = -ENOMEM; if (initial) { while (--i > 0) t7xx_unmap_bat_page(dpmaif_ctrl->dev, bat_req->bat_skb, i); } } return ret; } static int t7xx_dpmaif_set_frag_to_skb(const struct dpmaif_rx_queue *rxq, const struct dpmaif_pit *pkt_info, struct sk_buff *skb) { unsigned long long data_bus_addr, data_base_addr; struct device *dev = rxq->dpmaif_ctrl->dev; struct dpmaif_bat_page *page_info; unsigned int data_len; int data_offset; page_info = rxq->bat_frag->bat_skb; page_info += t7xx_normal_pit_bid(pkt_info); dma_unmap_page(dev, page_info->data_bus_addr, page_info->data_len, DMA_FROM_DEVICE); if (!page_info->page) return -EINVAL; data_bus_addr = le32_to_cpu(pkt_info->pd.data_addr_h); data_bus_addr = (data_bus_addr << 32) + le32_to_cpu(pkt_info->pd.data_addr_l); data_base_addr = page_info->data_bus_addr; data_offset = data_bus_addr - data_base_addr; data_offset += page_info->offset; data_len = FIELD_GET(PD_PIT_DATA_LEN, le32_to_cpu(pkt_info->header)); skb_add_rx_frag(skb, skb_shinfo(skb)->nr_frags, page_info->page, data_offset, data_len, page_info->data_len); page_info->page = NULL; page_info->offset = 0; page_info->data_len = 0; return 0; } static int t7xx_dpmaif_get_frag(struct dpmaif_rx_queue *rxq, const struct dpmaif_pit *pkt_info, const struct dpmaif_cur_rx_skb_info *skb_info) { unsigned int cur_bid = t7xx_normal_pit_bid(pkt_info); int ret; ret = t7xx_frag_bat_cur_bid_check(rxq, cur_bid); if (ret < 0) return ret; ret = t7xx_dpmaif_set_frag_to_skb(rxq, pkt_info, skb_info->cur_skb); if (ret < 0) { dev_err(rxq->dpmaif_ctrl->dev, "Failed to set frag data to skb: %d\n", ret); return ret; } t7xx_dpmaif_set_bat_mask(rxq->bat_frag, cur_bid); return 0; } static int t7xx_bat_cur_bid_check(struct dpmaif_rx_queue *rxq, const unsigned int cur_bid) { struct dpmaif_bat_skb *bat_skb = rxq->bat_req->bat_skb; bat_skb += cur_bid; if (cur_bid >= DPMAIF_BAT_COUNT || !bat_skb->skb) return -EINVAL; return 0; } static int t7xx_dpmaif_read_pit_seq(const struct dpmaif_pit *pit) { return FIELD_GET(PD_PIT_PIT_SEQ, le32_to_cpu(pit->pd.footer)); } static int t7xx_dpmaif_check_pit_seq(struct dpmaif_rx_queue *rxq, const struct dpmaif_pit *pit) { unsigned int cur_pit_seq, expect_pit_seq = rxq->expect_pit_seq; if (read_poll_timeout_atomic(t7xx_dpmaif_read_pit_seq, cur_pit_seq, cur_pit_seq == expect_pit_seq, DPMAIF_POLL_PIT_TIME_US, DPMAIF_POLL_PIT_MAX_TIME_US, false, pit)) return -EFAULT; rxq->expect_pit_seq++; if (rxq->expect_pit_seq >= DPMAIF_DL_PIT_SEQ_VALUE) rxq->expect_pit_seq = 0; return 0; } static unsigned int t7xx_dpmaif_avail_pkt_bat_cnt(struct dpmaif_bat_request *bat_req) { unsigned int zero_index; unsigned long flags; spin_lock_irqsave(&bat_req->mask_lock, flags); zero_index = find_next_zero_bit(bat_req->bat_bitmap, bat_req->bat_size_cnt, bat_req->bat_release_rd_idx); if (zero_index < bat_req->bat_size_cnt) { spin_unlock_irqrestore(&bat_req->mask_lock, flags); return zero_index - bat_req->bat_release_rd_idx; } /* limiting the search till bat_release_rd_idx */ zero_index = find_first_zero_bit(bat_req->bat_bitmap, bat_req->bat_release_rd_idx); spin_unlock_irqrestore(&bat_req->mask_lock, flags); return bat_req->bat_size_cnt - bat_req->bat_release_rd_idx + zero_index; } static int t7xx_dpmaif_release_bat_entry(const struct dpmaif_rx_queue *rxq, const unsigned int rel_entry_num, const enum bat_type buf_type) { struct dpmaif_hw_info *hw_info = &rxq->dpmaif_ctrl->hw_info; unsigned int old_rel_idx, new_rel_idx, hw_rd_idx, i; struct dpmaif_bat_request *bat; unsigned long flags; if (!rxq->que_started || !rel_entry_num) return -EINVAL; if (buf_type == BAT_TYPE_FRAG) { bat = rxq->bat_frag; hw_rd_idx = t7xx_dpmaif_dl_get_frg_rd_idx(hw_info, rxq->index); } else { bat = rxq->bat_req; hw_rd_idx = t7xx_dpmaif_dl_get_bat_rd_idx(hw_info, rxq->index); } if (rel_entry_num >= bat->bat_size_cnt) return -EINVAL; old_rel_idx = bat->bat_release_rd_idx; new_rel_idx = old_rel_idx + rel_entry_num; /* Do not need to release if the queue is empty */ if (bat->bat_wr_idx == old_rel_idx) return 0; if (hw_rd_idx >= old_rel_idx) { if (new_rel_idx > hw_rd_idx) return -EINVAL; } if (new_rel_idx >= bat->bat_size_cnt) { new_rel_idx -= bat->bat_size_cnt; if (new_rel_idx > hw_rd_idx) return -EINVAL; } spin_lock_irqsave(&bat->mask_lock, flags); for (i = 0; i < rel_entry_num; i++) { unsigned int index = bat->bat_release_rd_idx + i; if (index >= bat->bat_size_cnt) index -= bat->bat_size_cnt; clear_bit(index, bat->bat_bitmap); } spin_unlock_irqrestore(&bat->mask_lock, flags); bat->bat_release_rd_idx = new_rel_idx; return rel_entry_num; } static int t7xx_dpmaif_pit_release_and_add(struct dpmaif_rx_queue *rxq) { int ret; if (rxq->pit_remain_release_cnt < DPMAIF_PIT_CNT_THRESHOLD) return 0; ret = t7xx_dpmaifq_release_pit_entry(rxq, rxq->pit_remain_release_cnt); if (ret) return ret; rxq->pit_remain_release_cnt = 0; return 0; } static int t7xx_dpmaif_bat_release_and_add(const struct dpmaif_rx_queue *rxq) { unsigned int bid_cnt; int ret; bid_cnt = t7xx_dpmaif_avail_pkt_bat_cnt(rxq->bat_req); if (bid_cnt < DPMAIF_BAT_CNT_THRESHOLD) return 0; ret = t7xx_dpmaif_release_bat_entry(rxq, bid_cnt, BAT_TYPE_NORMAL); if (ret <= 0) { dev_err(rxq->dpmaif_ctrl->dev, "Release PKT BAT failed: %d\n", ret); return ret; } ret = t7xx_dpmaif_rx_buf_alloc(rxq->dpmaif_ctrl, rxq->bat_req, rxq->index, bid_cnt, false); if (ret < 0) dev_err(rxq->dpmaif_ctrl->dev, "Allocate new RX buffer failed: %d\n", ret); return ret; } static int t7xx_dpmaif_frag_bat_release_and_add(const struct dpmaif_rx_queue *rxq) { unsigned int bid_cnt; int ret; bid_cnt = t7xx_dpmaif_avail_pkt_bat_cnt(rxq->bat_frag); if (bid_cnt < DPMAIF_BAT_CNT_THRESHOLD) return 0; ret = t7xx_dpmaif_release_bat_entry(rxq, bid_cnt, BAT_TYPE_FRAG); if (ret <= 0) { dev_err(rxq->dpmaif_ctrl->dev, "Release BAT entry failed: %d\n", ret); return ret; } return t7xx_dpmaif_rx_frag_alloc(rxq->dpmaif_ctrl, rxq->bat_frag, bid_cnt, false); } static void t7xx_dpmaif_parse_msg_pit(const struct dpmaif_rx_queue *rxq, const struct dpmaif_pit *msg_pit, struct dpmaif_cur_rx_skb_info *skb_info) { int header = le32_to_cpu(msg_pit->header); skb_info->cur_chn_idx = FIELD_GET(MSG_PIT_CHANNEL_ID, header); skb_info->check_sum = FIELD_GET(MSG_PIT_CHECKSUM, header); skb_info->pit_dp = FIELD_GET(MSG_PIT_DP, header); skb_info->pkt_type = FIELD_GET(MSG_PIT_IP, le32_to_cpu(msg_pit->msg.params_3)); } static int t7xx_dpmaif_set_data_to_skb(const struct dpmaif_rx_queue *rxq, const struct dpmaif_pit *pkt_info, struct dpmaif_cur_rx_skb_info *skb_info) { unsigned long long data_bus_addr, data_base_addr; struct device *dev = rxq->dpmaif_ctrl->dev; struct dpmaif_bat_skb *bat_skb; unsigned int data_len; struct sk_buff *skb; int data_offset; bat_skb = rxq->bat_req->bat_skb; bat_skb += t7xx_normal_pit_bid(pkt_info); dma_unmap_single(dev, bat_skb->data_bus_addr, bat_skb->data_len, DMA_FROM_DEVICE); data_bus_addr = le32_to_cpu(pkt_info->pd.data_addr_h); data_bus_addr = (data_bus_addr << 32) + le32_to_cpu(pkt_info->pd.data_addr_l); data_base_addr = bat_skb->data_bus_addr; data_offset = data_bus_addr - data_base_addr; data_len = FIELD_GET(PD_PIT_DATA_LEN, le32_to_cpu(pkt_info->header)); skb = bat_skb->skb; skb->len = 0; skb_reset_tail_pointer(skb); skb_reserve(skb, data_offset); if (skb->tail + data_len > skb->end) { dev_err(dev, "No buffer space available\n"); return -ENOBUFS; } skb_put(skb, data_len); skb_info->cur_skb = skb; bat_skb->skb = NULL; return 0; } static int t7xx_dpmaif_get_rx_pkt(struct dpmaif_rx_queue *rxq, const struct dpmaif_pit *pkt_info, struct dpmaif_cur_rx_skb_info *skb_info) { unsigned int cur_bid = t7xx_normal_pit_bid(pkt_info); int ret; ret = t7xx_bat_cur_bid_check(rxq, cur_bid); if (ret < 0) return ret; ret = t7xx_dpmaif_set_data_to_skb(rxq, pkt_info, skb_info); if (ret < 0) { dev_err(rxq->dpmaif_ctrl->dev, "RX set data to skb failed: %d\n", ret); return ret; } t7xx_dpmaif_set_bat_mask(rxq->bat_req, cur_bid); return 0; } static int t7xx_dpmaifq_rx_notify_hw(struct dpmaif_rx_queue *rxq) { struct dpmaif_ctrl *dpmaif_ctrl = rxq->dpmaif_ctrl; int ret; queue_work(dpmaif_ctrl->bat_release_wq, &dpmaif_ctrl->bat_release_work); ret = t7xx_dpmaif_pit_release_and_add(rxq); if (ret < 0) dev_err(dpmaif_ctrl->dev, "RXQ%u update PIT failed: %d\n", rxq->index, ret); return ret; } static void t7xx_dpmaif_rx_skb_enqueue(struct dpmaif_rx_queue *rxq, struct sk_buff *skb) { unsigned long flags; spin_lock_irqsave(&rxq->skb_list.lock, flags); if (rxq->skb_list.qlen < rxq->skb_list_max_len) __skb_queue_tail(&rxq->skb_list, skb); else dev_kfree_skb_any(skb); spin_unlock_irqrestore(&rxq->skb_list.lock, flags); } static void t7xx_dpmaif_rx_skb(struct dpmaif_rx_queue *rxq, struct dpmaif_cur_rx_skb_info *skb_info) { struct sk_buff *skb = skb_info->cur_skb; struct t7xx_skb_cb *skb_cb; u8 netif_id; skb_info->cur_skb = NULL; if (skb_info->pit_dp) { dev_kfree_skb_any(skb); return; } skb->ip_summed = skb_info->check_sum == DPMAIF_CS_RESULT_PASS ? CHECKSUM_UNNECESSARY : CHECKSUM_NONE; netif_id = FIELD_GET(NETIF_MASK, skb_info->cur_chn_idx); skb_cb = T7XX_SKB_CB(skb); skb_cb->netif_idx = netif_id; skb_cb->rx_pkt_type = skb_info->pkt_type; t7xx_dpmaif_rx_skb_enqueue(rxq, skb); } static int t7xx_dpmaif_rx_start(struct dpmaif_rx_queue *rxq, const unsigned int pit_cnt, const unsigned long timeout) { unsigned int cur_pit, pit_len, rx_cnt, recv_skb_cnt = 0; struct device *dev = rxq->dpmaif_ctrl->dev; struct dpmaif_cur_rx_skb_info *skb_info; int ret = 0; pit_len = rxq->pit_size_cnt; skb_info = &rxq->rx_data_info; cur_pit = rxq->pit_rd_idx; for (rx_cnt = 0; rx_cnt < pit_cnt; rx_cnt++) { struct dpmaif_pit *pkt_info; u32 val; if (!skb_info->msg_pit_received && time_after_eq(jiffies, timeout)) break; pkt_info = (struct dpmaif_pit *)rxq->pit_base + cur_pit; if (t7xx_dpmaif_check_pit_seq(rxq, pkt_info)) { dev_err_ratelimited(dev, "RXQ%u checks PIT SEQ fail\n", rxq->index); return -EAGAIN; } val = FIELD_GET(PD_PIT_PACKET_TYPE, le32_to_cpu(pkt_info->header)); if (val == DES_PT_MSG) { if (skb_info->msg_pit_received) dev_err(dev, "RXQ%u received repeated PIT\n", rxq->index); skb_info->msg_pit_received = true; t7xx_dpmaif_parse_msg_pit(rxq, pkt_info, skb_info); } else { /* DES_PT_PD */ val = FIELD_GET(PD_PIT_BUFFER_TYPE, le32_to_cpu(pkt_info->header)); if (val != PKT_BUF_FRAG) ret = t7xx_dpmaif_get_rx_pkt(rxq, pkt_info, skb_info); else if (!skb_info->cur_skb) ret = -EINVAL; else ret = t7xx_dpmaif_get_frag(rxq, pkt_info, skb_info); if (ret < 0) { skb_info->err_payload = 1; dev_err_ratelimited(dev, "RXQ%u error payload\n", rxq->index); } val = FIELD_GET(PD_PIT_CONT, le32_to_cpu(pkt_info->header)); if (!val) { if (!skb_info->err_payload) { t7xx_dpmaif_rx_skb(rxq, skb_info); } else if (skb_info->cur_skb) { dev_kfree_skb_any(skb_info->cur_skb); skb_info->cur_skb = NULL; } memset(skb_info, 0, sizeof(*skb_info)); recv_skb_cnt++; if (!(recv_skb_cnt & DPMAIF_RX_PUSH_THRESHOLD_MASK)) { wake_up_all(&rxq->rx_wq); recv_skb_cnt = 0; } } } cur_pit = t7xx_ring_buf_get_next_wr_idx(pit_len, cur_pit); rxq->pit_rd_idx = cur_pit; rxq->pit_remain_release_cnt++; if (rx_cnt > 0 && !(rx_cnt % DPMAIF_NOTIFY_RELEASE_COUNT)) { ret = t7xx_dpmaifq_rx_notify_hw(rxq); if (ret < 0) break; } } if (recv_skb_cnt) wake_up_all(&rxq->rx_wq); if (!ret) ret = t7xx_dpmaifq_rx_notify_hw(rxq); if (ret) return ret; return rx_cnt; } static unsigned int t7xx_dpmaifq_poll_pit(struct dpmaif_rx_queue *rxq) { unsigned int hw_wr_idx, pit_cnt; if (!rxq->que_started) return 0; hw_wr_idx = t7xx_dpmaif_dl_dlq_pit_get_wr_idx(&rxq->dpmaif_ctrl->hw_info, rxq->index); pit_cnt = t7xx_ring_buf_rd_wr_count(rxq->pit_size_cnt, rxq->pit_rd_idx, hw_wr_idx, DPMAIF_READ); rxq->pit_wr_idx = hw_wr_idx; return pit_cnt; } static int t7xx_dpmaif_rx_data_collect(struct dpmaif_ctrl *dpmaif_ctrl, const unsigned int q_num, const unsigned int budget) { struct dpmaif_rx_queue *rxq = &dpmaif_ctrl->rxq[q_num]; unsigned long time_limit; unsigned int cnt; time_limit = jiffies + msecs_to_jiffies(DPMAIF_WQ_TIME_LIMIT_MS); while ((cnt = t7xx_dpmaifq_poll_pit(rxq))) { unsigned int rd_cnt; int real_cnt; rd_cnt = min(cnt, budget); real_cnt = t7xx_dpmaif_rx_start(rxq, rd_cnt, time_limit); if (real_cnt < 0) return real_cnt; if (real_cnt < cnt) return -EAGAIN; } return 0; } static void t7xx_dpmaif_do_rx(struct dpmaif_ctrl *dpmaif_ctrl, struct dpmaif_rx_queue *rxq) { struct dpmaif_hw_info *hw_info = &dpmaif_ctrl->hw_info; int ret; ret = t7xx_dpmaif_rx_data_collect(dpmaif_ctrl, rxq->index, rxq->budget); if (ret < 0) { /* Try one more time */ queue_work(rxq->worker, &rxq->dpmaif_rxq_work); t7xx_dpmaif_clr_ip_busy_sts(hw_info); } else { t7xx_dpmaif_clr_ip_busy_sts(hw_info); t7xx_dpmaif_dlq_unmask_rx_done(hw_info, rxq->index); } } static void t7xx_dpmaif_rxq_work(struct work_struct *work) { struct dpmaif_rx_queue *rxq = container_of(work, struct dpmaif_rx_queue, dpmaif_rxq_work); struct dpmaif_ctrl *dpmaif_ctrl = rxq->dpmaif_ctrl; int ret; atomic_set(&rxq->rx_processing, 1); /* Ensure rx_processing is changed to 1 before actually begin RX flow */ smp_mb(); if (!rxq->que_started) { atomic_set(&rxq->rx_processing, 0); dev_err(dpmaif_ctrl->dev, "Work RXQ: %d has not been started\n", rxq->index); return; } ret = pm_runtime_resume_and_get(dpmaif_ctrl->dev); if (ret < 0 && ret != -EACCES) return; t7xx_pci_disable_sleep(dpmaif_ctrl->t7xx_dev); if (t7xx_pci_sleep_disable_complete(dpmaif_ctrl->t7xx_dev)) t7xx_dpmaif_do_rx(dpmaif_ctrl, rxq); t7xx_pci_enable_sleep(dpmaif_ctrl->t7xx_dev); pm_runtime_mark_last_busy(dpmaif_ctrl->dev); pm_runtime_put_autosuspend(dpmaif_ctrl->dev); atomic_set(&rxq->rx_processing, 0); } void t7xx_dpmaif_irq_rx_done(struct dpmaif_ctrl *dpmaif_ctrl, const unsigned int que_mask) { struct dpmaif_rx_queue *rxq; int qno; qno = ffs(que_mask) - 1; if (qno < 0 || qno > DPMAIF_RXQ_NUM - 1) { dev_err(dpmaif_ctrl->dev, "Invalid RXQ number: %u\n", qno); return; } rxq = &dpmaif_ctrl->rxq[qno]; queue_work(rxq->worker, &rxq->dpmaif_rxq_work); } static void t7xx_dpmaif_base_free(const struct dpmaif_ctrl *dpmaif_ctrl, const struct dpmaif_bat_request *bat_req) { if (bat_req->bat_base) dma_free_coherent(dpmaif_ctrl->dev, bat_req->bat_size_cnt * sizeof(struct dpmaif_bat), bat_req->bat_base, bat_req->bat_bus_addr); } /** * t7xx_dpmaif_bat_alloc() - Allocate the BAT ring buffer. * @dpmaif_ctrl: Pointer to DPMAIF context structure. * @bat_req: Pointer to BAT request structure. * @buf_type: BAT ring type. * * This function allocates the BAT ring buffer shared with the HW device, also allocates * a buffer used to store information about the BAT skbs for further release. * * Return: * * 0 - Success. * * -ERROR - Error code. */ int t7xx_dpmaif_bat_alloc(const struct dpmaif_ctrl *dpmaif_ctrl, struct dpmaif_bat_request *bat_req, const enum bat_type buf_type) { int sw_buf_size; if (buf_type == BAT_TYPE_FRAG) { sw_buf_size = sizeof(struct dpmaif_bat_page); bat_req->bat_size_cnt = DPMAIF_FRG_COUNT; bat_req->pkt_buf_sz = DPMAIF_HW_FRG_PKTBUF; } else { sw_buf_size = sizeof(struct dpmaif_bat_skb); bat_req->bat_size_cnt = DPMAIF_BAT_COUNT; bat_req->pkt_buf_sz = DPMAIF_HW_BAT_PKTBUF; } bat_req->type = buf_type; bat_req->bat_wr_idx = 0; bat_req->bat_release_rd_idx = 0; bat_req->bat_base = dma_alloc_coherent(dpmaif_ctrl->dev, bat_req->bat_size_cnt * sizeof(struct dpmaif_bat), &bat_req->bat_bus_addr, GFP_KERNEL | __GFP_ZERO); if (!bat_req->bat_base) return -ENOMEM; /* For AP SW to record skb information */ bat_req->bat_skb = devm_kzalloc(dpmaif_ctrl->dev, bat_req->bat_size_cnt * sw_buf_size, GFP_KERNEL); if (!bat_req->bat_skb) goto err_free_dma_mem; bat_req->bat_bitmap = bitmap_zalloc(bat_req->bat_size_cnt, GFP_KERNEL); if (!bat_req->bat_bitmap) goto err_free_dma_mem; spin_lock_init(&bat_req->mask_lock); atomic_set(&bat_req->refcnt, 0); return 0; err_free_dma_mem: t7xx_dpmaif_base_free(dpmaif_ctrl, bat_req); return -ENOMEM; } void t7xx_dpmaif_bat_free(const struct dpmaif_ctrl *dpmaif_ctrl, struct dpmaif_bat_request *bat_req) { if (!bat_req || !atomic_dec_and_test(&bat_req->refcnt)) return; bitmap_free(bat_req->bat_bitmap); bat_req->bat_bitmap = NULL; if (bat_req->bat_skb) { unsigned int i; for (i = 0; i < bat_req->bat_size_cnt; i++) { if (bat_req->type == BAT_TYPE_FRAG) t7xx_unmap_bat_page(dpmaif_ctrl->dev, bat_req->bat_skb, i); else t7xx_unmap_bat_skb(dpmaif_ctrl->dev, bat_req->bat_skb, i); } } t7xx_dpmaif_base_free(dpmaif_ctrl, bat_req); } static int t7xx_dpmaif_rx_alloc(struct dpmaif_rx_queue *rxq) { rxq->pit_size_cnt = DPMAIF_PIT_COUNT; rxq->pit_rd_idx = 0; rxq->pit_wr_idx = 0; rxq->pit_release_rd_idx = 0; rxq->expect_pit_seq = 0; rxq->pit_remain_release_cnt = 0; memset(&rxq->rx_data_info, 0, sizeof(rxq->rx_data_info)); rxq->pit_base = dma_alloc_coherent(rxq->dpmaif_ctrl->dev, rxq->pit_size_cnt * sizeof(struct dpmaif_pit), &rxq->pit_bus_addr, GFP_KERNEL | __GFP_ZERO); if (!rxq->pit_base) return -ENOMEM; rxq->bat_req = &rxq->dpmaif_ctrl->bat_req; atomic_inc(&rxq->bat_req->refcnt); rxq->bat_frag = &rxq->dpmaif_ctrl->bat_frag; atomic_inc(&rxq->bat_frag->refcnt); return 0; } static void t7xx_dpmaif_rx_buf_free(const struct dpmaif_rx_queue *rxq) { if (!rxq->dpmaif_ctrl) return; t7xx_dpmaif_bat_free(rxq->dpmaif_ctrl, rxq->bat_req); t7xx_dpmaif_bat_free(rxq->dpmaif_ctrl, rxq->bat_frag); if (rxq->pit_base) dma_free_coherent(rxq->dpmaif_ctrl->dev, rxq->pit_size_cnt * sizeof(struct dpmaif_pit), rxq->pit_base, rxq->pit_bus_addr); } int t7xx_dpmaif_rxq_init(struct dpmaif_rx_queue *queue) { int ret; ret = t7xx_dpmaif_rx_alloc(queue); if (ret < 0) { dev_err(queue->dpmaif_ctrl->dev, "Failed to allocate RX buffers: %d\n", ret); return ret; } INIT_WORK(&queue->dpmaif_rxq_work, t7xx_dpmaif_rxq_work); queue->worker = alloc_workqueue("dpmaif_rx%d_worker", WQ_UNBOUND | WQ_MEM_RECLAIM | WQ_HIGHPRI, 1, queue->index); if (!queue->worker) { ret = -ENOMEM; goto err_free_rx_buffer; } init_waitqueue_head(&queue->rx_wq); skb_queue_head_init(&queue->skb_list); queue->skb_list_max_len = queue->bat_req->pkt_buf_sz; queue->rx_thread = kthread_run(t7xx_dpmaif_net_rx_push_thread, queue, "dpmaif_rx%d_push", queue->index); ret = PTR_ERR_OR_ZERO(queue->rx_thread); if (ret) goto err_free_workqueue; return 0; err_free_workqueue: destroy_workqueue(queue->worker); err_free_rx_buffer: t7xx_dpmaif_rx_buf_free(queue); return ret; } void t7xx_dpmaif_rxq_free(struct dpmaif_rx_queue *queue) { if (queue->worker) destroy_workqueue(queue->worker); if (queue->rx_thread) kthread_stop(queue->rx_thread); skb_queue_purge(&queue->skb_list); t7xx_dpmaif_rx_buf_free(queue); } static void t7xx_dpmaif_bat_release_work(struct work_struct *work) { struct dpmaif_ctrl *dpmaif_ctrl = container_of(work, struct dpmaif_ctrl, bat_release_work); struct dpmaif_rx_queue *rxq; int ret; ret = pm_runtime_resume_and_get(dpmaif_ctrl->dev); if (ret < 0 && ret != -EACCES) return; t7xx_pci_disable_sleep(dpmaif_ctrl->t7xx_dev); /* ALL RXQ use one BAT table, so choose DPF_RX_QNO_DFT */ rxq = &dpmaif_ctrl->rxq[DPF_RX_QNO_DFT]; if (t7xx_pci_sleep_disable_complete(dpmaif_ctrl->t7xx_dev)) { t7xx_dpmaif_bat_release_and_add(rxq); t7xx_dpmaif_frag_bat_release_and_add(rxq); } t7xx_pci_enable_sleep(dpmaif_ctrl->t7xx_dev); pm_runtime_mark_last_busy(dpmaif_ctrl->dev); pm_runtime_put_autosuspend(dpmaif_ctrl->dev); } int t7xx_dpmaif_bat_rel_wq_alloc(struct dpmaif_ctrl *dpmaif_ctrl) { dpmaif_ctrl->bat_release_wq = alloc_workqueue("dpmaif_bat_release_work_queue", WQ_MEM_RECLAIM, 1); if (!dpmaif_ctrl->bat_release_wq) return -ENOMEM; INIT_WORK(&dpmaif_ctrl->bat_release_work, t7xx_dpmaif_bat_release_work); return 0; } void t7xx_dpmaif_bat_wq_rel(struct dpmaif_ctrl *dpmaif_ctrl) { flush_work(&dpmaif_ctrl->bat_release_work); if (dpmaif_ctrl->bat_release_wq) { destroy_workqueue(dpmaif_ctrl->bat_release_wq); dpmaif_ctrl->bat_release_wq = NULL; } } /** * t7xx_dpmaif_rx_stop() - Suspend RX flow. * @dpmaif_ctrl: Pointer to data path control struct dpmaif_ctrl. * * Wait for all the RX work to finish executing and mark the RX queue as paused. */ void t7xx_dpmaif_rx_stop(struct dpmaif_ctrl *dpmaif_ctrl) { unsigned int i; for (i = 0; i < DPMAIF_RXQ_NUM; i++) { struct dpmaif_rx_queue *rxq = &dpmaif_ctrl->rxq[i]; int timeout, value; flush_work(&rxq->dpmaif_rxq_work); timeout = readx_poll_timeout_atomic(atomic_read, &rxq->rx_processing, value, !value, 0, DPMAIF_CHECK_INIT_TIMEOUT_US); if (timeout) dev_err(dpmaif_ctrl->dev, "Stop RX SW failed\n"); /* Ensure RX processing has stopped before we set rxq->que_started to false */ smp_mb(); rxq->que_started = false; } } static void t7xx_dpmaif_stop_rxq(struct dpmaif_rx_queue *rxq) { int cnt, j = 0; flush_work(&rxq->dpmaif_rxq_work); rxq->que_started = false; do { cnt = t7xx_ring_buf_rd_wr_count(rxq->pit_size_cnt, rxq->pit_rd_idx, rxq->pit_wr_idx, DPMAIF_READ); if (++j >= DPMAIF_MAX_CHECK_COUNT) { dev_err(rxq->dpmaif_ctrl->dev, "Stop RX SW failed, %d\n", cnt); break; } } while (cnt); memset(rxq->pit_base, 0, rxq->pit_size_cnt * sizeof(struct dpmaif_pit)); memset(rxq->bat_req->bat_base, 0, rxq->bat_req->bat_size_cnt * sizeof(struct dpmaif_bat)); bitmap_zero(rxq->bat_req->bat_bitmap, rxq->bat_req->bat_size_cnt); memset(&rxq->rx_data_info, 0, sizeof(rxq->rx_data_info)); rxq->pit_rd_idx = 0; rxq->pit_wr_idx = 0; rxq->pit_release_rd_idx = 0; rxq->expect_pit_seq = 0; rxq->pit_remain_release_cnt = 0; rxq->bat_req->bat_release_rd_idx = 0; rxq->bat_req->bat_wr_idx = 0; rxq->bat_frag->bat_release_rd_idx = 0; rxq->bat_frag->bat_wr_idx = 0; } void t7xx_dpmaif_rx_clear(struct dpmaif_ctrl *dpmaif_ctrl) { int i; for (i = 0; i < DPMAIF_RXQ_NUM; i++) t7xx_dpmaif_stop_rxq(&dpmaif_ctrl->rxq[i]); }
Information contained on this website is for historical information purposes only and does not indicate or represent copyright ownership.
Created with Cregit http://github.com/cregit/cregit
Version 2.0-RC1