/*! \file ngknet_buff.c * * Utility routines for NGKNET packet buffer management in Linux kernel mode. * */ /* * $Copyright: Copyright 2018-2022 Broadcom. All rights reserved. * The term 'Broadcom' refers to Broadcom Inc. and/or its subsidiaries. * * This program is free software; you can redistribute it and/or * modify it under the terms of the GNU General Public License * version 2 as published by the Free Software Foundation. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * A copy of the GNU General Public License version 2 (GPLv2) can * be found in the LICENSES folder.$ */ #include #include #include #include "ngknet_main.h" #include "ngknet_buff.h" /*! * Allocate coherent memory */ static void * ngknet_ring_buf_alloc(struct pdma_dev *dev, uint32_t size, dma_addr_t *dma) { struct ngknet_dev *kdev = (struct ngknet_dev *)dev->priv; return dma_alloc_coherent(kdev->dev, size, dma, GFP_KERNEL); } /*! * Free coherent memory */ static void ngknet_ring_buf_free(struct pdma_dev *dev, uint32_t size, void *addr, dma_addr_t dma) { struct ngknet_dev *kdev = (struct ngknet_dev *)dev->priv; dma_free_coherent(kdev->dev, size, addr, dma); } /*! * Allocate Rx buffer */ static int ngknet_rx_buf_alloc(struct pdma_dev *dev, struct pdma_rx_queue *rxq, struct pdma_rx_buf *pbuf) { struct ngknet_dev *kdev = (struct ngknet_dev *)dev->priv; dma_addr_t dma; struct page *page; struct sk_buff *skb; if (rxq->buf_mode == PDMA_BUF_MODE_PAGE) { page = kal_dev_alloc_pages(rxq->page_order); if (unlikely(!page)) { return SHR_E_MEMORY; } dma = kal_dma_map_page_attrs(kdev->dev, page, 0, PAGE_SIZE * (1 << rxq->page_order), DMA_FROM_DEVICE, DMA_ATTR_SKIP_CPU_SYNC | DMA_ATTR_WEAK_ORDERING); if (unlikely(dma_mapping_error(kdev->dev, dma))) { __free_pages(page, rxq->page_order); return SHR_E_MEMORY; } pbuf->dma = dma; pbuf->page = page; pbuf->page_offset = 0; } else { skb = netdev_alloc_skb(kdev->net_dev, PDMA_RXB_RESV + pbuf->adj + rxq->buf_size); if (unlikely(!skb)) { return SHR_E_MEMORY; } skb_reserve(skb, PDMA_RXB_ALIGN - (((unsigned long)skb->data) & (PDMA_RXB_ALIGN - 1))); pbuf->skb = skb; pbuf->pkb = (struct pkt_buf *)skb->data; dma = dma_map_single(kdev->dev, &pbuf->pkb->data + pbuf->adj, rxq->buf_size, DMA_FROM_DEVICE); if (unlikely(dma_mapping_error(kdev->dev, dma))) { dev_kfree_skb_any(skb); return SHR_E_MEMORY; } pbuf->dma = dma; } return SHR_E_NONE; } /*! * Get Rx buffer DMA address */ static void ngknet_rx_buf_dma(struct pdma_dev *dev, struct pdma_rx_queue *rxq, struct pdma_rx_buf *pbuf, dma_addr_t *addr) { if (rxq->buf_mode == PDMA_BUF_MODE_PAGE) { *addr = pbuf->dma + pbuf->page_offset + PDMA_RXB_RESV + pbuf->adj; } else { *addr = pbuf->dma; } } /*! * Check Rx buffer */ static bool ngknet_rx_buf_avail(struct pdma_dev *dev, struct pdma_rx_queue *rxq, struct pdma_rx_buf *pbuf) { if (rxq->buf_mode == PDMA_BUF_MODE_PAGE) { pbuf->skb = NULL; } return (pbuf->dma != 0); } /*! * Get Rx buffer */ static struct pkt_hdr * ngknet_rx_buf_get(struct pdma_dev *dev, struct pdma_rx_queue *rxq, struct pdma_rx_buf *pbuf, int len) { struct ngknet_dev *kdev = (struct ngknet_dev *)dev->priv; struct sk_buff *skb; uint32_t pages_size; if (rxq->buf_mode == PDMA_BUF_MODE_PAGE) { if (pbuf->skb) { return &pbuf->pkb->pkh; } skb = kal_build_skb(page_address(pbuf->page) + pbuf->page_offset, PDMA_RXB_SIZE(rxq->buf_size + pbuf->adj)); if (unlikely(!skb)) { return NULL; } skb_reserve(skb, PDMA_RXB_ALIGN); pages_size = PAGE_SIZE * (1 << rxq->page_order); dma_sync_single_range_for_cpu(kdev->dev, pbuf->dma, pbuf->page_offset, pages_size >> 1, DMA_FROM_DEVICE); pbuf->skb = skb; pbuf->pkb = (struct pkt_buf *)skb->data; /* Try to reuse this page */ if (unlikely(page_count(pbuf->page) != 1) || kal_page_is_pfmemalloc(pbuf->page) || page_to_nid(pbuf->page) != numa_mem_id()) { kal_dma_unmap_page_attrs(kdev->dev, pbuf->dma, pages_size, DMA_FROM_DEVICE, DMA_ATTR_SKIP_CPU_SYNC | DMA_ATTR_WEAK_ORDERING); pbuf->dma = 0; } else { pbuf->page_offset ^= pages_size >> 1; page_ref_inc(pbuf->page); dma_sync_single_range_for_device(kdev->dev, pbuf->dma, pbuf->page_offset, pages_size >> 1, DMA_FROM_DEVICE); } } else { if (!pbuf->dma) { return &pbuf->pkb->pkh; } skb = pbuf->skb; dma_unmap_single(kdev->dev, pbuf->dma, rxq->buf_size, DMA_FROM_DEVICE); pbuf->dma = 0; } skb_put(skb, PKT_HDR_SIZE + pbuf->adj + len); return &pbuf->pkb->pkh; } /*! * Put Rx buffer */ static int ngknet_rx_buf_put(struct pdma_dev *dev, struct pdma_rx_queue *rxq, struct pdma_rx_buf *pbuf, int len) { struct ngknet_dev *kdev = (struct ngknet_dev *)dev->priv; dma_addr_t dma; struct sk_buff *skb; if (rxq->buf_mode == PDMA_BUF_MODE_PAGE) { dev_kfree_skb_any(pbuf->skb); } else { skb = pbuf->skb; if (pbuf->pkb != (struct pkt_buf *)skb->data) { dev_kfree_skb_any(skb); pbuf->dma = 0; return SHR_E_NONE; } dma = dma_map_single(kdev->dev, &pbuf->pkb->data + pbuf->adj, rxq->buf_size, DMA_FROM_DEVICE); if (unlikely(dma_mapping_error(kdev->dev, dma))) { dev_kfree_skb_any(skb); pbuf->dma = 0; return SHR_E_MEMORY; } pbuf->dma = dma; skb_trim(skb, 0); } return SHR_E_NONE; } /*! * Free Rx buffer */ static void ngknet_rx_buf_free(struct pdma_dev *dev, struct pdma_rx_queue *rxq, struct pdma_rx_buf *pbuf) { struct ngknet_dev *kdev = (struct ngknet_dev *)dev->priv; uint32_t pages_size; if (rxq->buf_mode == PDMA_BUF_MODE_PAGE) { if (!pbuf->page) { return; } pages_size = PAGE_SIZE * (1 << rxq->page_order); kal_dma_unmap_page_attrs(kdev->dev, pbuf->dma, pages_size, DMA_FROM_DEVICE, DMA_ATTR_SKIP_CPU_SYNC | DMA_ATTR_WEAK_ORDERING); __free_pages(pbuf->page, rxq->page_order); } else { if (!pbuf->skb) { return; } dma_unmap_single(kdev->dev, pbuf->dma, rxq->buf_size, DMA_FROM_DEVICE); dev_kfree_skb_any(pbuf->skb); } pbuf->dma = 0; pbuf->page = NULL; pbuf->page_offset = 0; pbuf->skb = NULL; pbuf->pkb = NULL; pbuf->adj = 0; } /*! * Get Rx buffer mode */ static enum buf_mode ngknet_rx_buf_mode(struct pdma_dev *dev, struct pdma_rx_queue *rxq) { uint32_t len, order; switch (ngknet_page_buffer_mode_get()) { case 0: /* Forced SKB mode */ return PDMA_BUF_MODE_SKB; case 1: /* Forced page mode */ break; default: /* -1 */ /* Select buffer mode based on system capability */ if (kal_support_paged_skb() == 0) { return PDMA_BUF_MODE_SKB; } break; } len = dev->rx_ph_size ? rxq->buf_size : rxq->buf_size + PDMA_RXB_META; for (order = 0; order < 32; order++) { if (PDMA_RXB_SIZE(len) * 2 <= PAGE_SIZE * (1 << order)) { rxq->page_order = order; break; } } return PDMA_BUF_MODE_PAGE; } /*! * Get Tx buffer */ static struct pkt_hdr * ngknet_tx_buf_get(struct pdma_dev *dev, struct pdma_tx_queue *txq, struct pdma_tx_buf *pbuf, void *buf) { struct ngknet_dev *kdev = (struct ngknet_dev *)dev->priv; struct sk_buff *skb = (struct sk_buff *)buf; struct pkt_buf *pkb = (struct pkt_buf *)skb->data; dma_addr_t dma; pbuf->len = pkb->pkh.data_len + (pbuf->adj ? pkb->pkh.meta_len : 0); dma = dma_map_single(kdev->dev, &pkb->data + (pbuf->adj ? 0 : pkb->pkh.meta_len), pbuf->len, DMA_TO_DEVICE); if (unlikely(dma_mapping_error(kdev->dev, dma))) { dev_kfree_skb_any(skb); return NULL; } pbuf->dma = dma; pbuf->skb = skb; pbuf->pkb = pkb; return &pkb->pkh; } /*! * Get Tx buffer DMA address */ static void ngknet_tx_buf_dma(struct pdma_dev *dev, struct pdma_tx_queue *txq, struct pdma_tx_buf *pbuf, dma_addr_t *addr) { *addr = pbuf->dma; } /*! * Free Tx buffer */ static void ngknet_tx_buf_free(struct pdma_dev *dev, struct pdma_tx_queue *txq, struct pdma_tx_buf *pbuf) { struct ngknet_dev *kdev = (struct ngknet_dev *)dev->priv; if (!pbuf->skb) { return; } dma_unmap_single(kdev->dev, pbuf->dma, pbuf->len, DMA_TO_DEVICE); if (skb_shinfo(pbuf->skb)->tx_flags & SKBTX_IN_PROGRESS) { skb_queue_tail(&kdev->ptp_tx_queue, pbuf->skb); schedule_work(&kdev->ptp_tx_work); } else { dev_kfree_skb_any(pbuf->skb); } pbuf->dma = 0; pbuf->len = 0; pbuf->skb = NULL; pbuf->pkb = NULL; pbuf->adj = 0; } static const struct pdma_buf_mngr buf_mngr = { .ring_buf_alloc = ngknet_ring_buf_alloc, .ring_buf_free = ngknet_ring_buf_free, .rx_buf_alloc = ngknet_rx_buf_alloc, .rx_buf_dma = ngknet_rx_buf_dma, .rx_buf_avail = ngknet_rx_buf_avail, .rx_buf_get = ngknet_rx_buf_get, .rx_buf_put = ngknet_rx_buf_put, .rx_buf_free = ngknet_rx_buf_free, .rx_buf_mode = ngknet_rx_buf_mode, .tx_buf_get = ngknet_tx_buf_get, .tx_buf_dma = ngknet_tx_buf_dma, .tx_buf_free = ngknet_tx_buf_free, }; /*! * Open a device */ void bcmcnet_buf_mngr_init(struct pdma_dev *dev) { dev->ctrl.buf_mngr = (struct pdma_buf_mngr *)&buf_mngr; }