sonic-buildimage/platform/broadcom/saibcm-modules/sdklt/linux/knetcb/ngknetcb_main.c

463 lines
12 KiB
C

/*! \file ngknetcb_main.c
*
* NGKNET Callback module entry.
*/
/*
* $Copyright: (c) 2019 Broadcom.
* Broadcom Proprietary and Confidential. All rights reserved.$
*/
#include <lkm/lkm.h>
#include <ngknet_callback.h>
#include "psample-cb.h"
/*! \cond */
MODULE_AUTHOR("Broadcom Corporation");
MODULE_DESCRIPTION("NGKNET Callback Module");
MODULE_LICENSE("GPL");
/*! \endcond */
/*! \cond */
int debug = 0;
MODULE_PARAM(debug, int, 0);
MODULE_PARM_DESC(debug,
"Debug level (default 0)");
/*! \endcond */
/*! Module information */
#define NGKNETCB_MODULE_NAME "linux_ngknetcb"
#define NGKNETCB_MODULE_MAJOR 122
/* set KNET_CB_DEBUG for debug info */
#define KNET_CB_DEBUG
/* These below need to match incoming enum values */
#define FILTER_TAG_STRIP 0
#define FILTER_TAG_KEEP 1
#define FILTER_TAG_ORIGINAL 2
/* Maintain tag strip statistics */
struct strip_stats_s {
unsigned long stripped; /* Number of packets that have been stripped */
unsigned long checked;
unsigned long skipped;
};
static struct strip_stats_s strip_stats;
static unsigned int rx_count = 0;
/* Local function prototypes */
static void strip_vlan_tag(struct sk_buff *skb);
/* Remove VLAN tag for select TPIDs */
static void
strip_vlan_tag(struct sk_buff *skb)
{
uint16_t vlan_proto;
uint8_t *pkt = skb->data;
vlan_proto = (uint16_t) ((pkt[12] << 8) | pkt[13]);
if ((vlan_proto == 0x8100) || (vlan_proto == 0x88a8) || (vlan_proto == 0x9100)) {
/* Move first 12 bytes of packet back by 4 */
memmove(&skb->data[4], skb->data, 12);
skb_pull(skb, 4); /* Remove 4 bytes from start of buffer */
}
}
/*
* The function get_tag_status() returns the tag status.
* 0 = Untagged
* 1 = Single inner-tag
* 2 = Single outer-tag
* 3 = Double tagged.
* -1 = Unsupported type
*/
static int
get_tag_status(uint32_t dev_type, uint32_t variant, void *meta)
{
uint32_t *valptr;
uint32_t fd_index;
uint32_t outer_l2_hdr;
int tag_status = -1;
uint32_t match_id_minbit = 1;
uint32_t outer_tag_match = 0x10;
if ((dev_type == 0xb880) || (dev_type == 0xb780))
{
/* Field BCM_PKTIO_RXPMD_MATCH_ID_LO has tag status in RX PMD */
fd_index = 2;
valptr = (uint32_t *)meta;
match_id_minbit = (dev_type == 0xb780) ? 2 : 1;
outer_l2_hdr = (valptr[fd_index] >> match_id_minbit & 0xFF);
outer_tag_match = ((dev_type == 0xb780 && variant == 1) ? 0x8 : 0x10);
if (outer_l2_hdr & 0x1) {
#ifdef KNET_CB_DEBUG
if (debug & 0x1) {
printk(" L2 Header Present\n");
}
#endif
tag_status = 0;
if (outer_l2_hdr & 0x4) {
#ifdef KNET_CB_DEBUG
if (debug & 0x1) {
printk(" SNAP/LLC\n");
}
#endif
tag_status = 0;
}
if (outer_l2_hdr & outer_tag_match) {
#ifdef KNET_CB_DEBUG
if (debug & 0x1) {
printk(" Outer Tagged\n");
}
#endif
tag_status = 2;
if (outer_l2_hdr & 0x20) {
#ifdef KNET_CB_DEBUG
if (debug & 0x1) {
printk(" Double Tagged\n");
}
#endif
tag_status = 3;
}
}
else if (outer_l2_hdr & 0x20) {
#ifdef KNET_CB_DEBUG
if (debug & 0x1) {
printk(" Inner Tagged\n");
}
#endif
tag_status = 1;
}
}
}
else if ((dev_type == 0xb990)|| (dev_type == 0xb996))
{
fd_index = 9;
valptr = (uint32_t *)meta;
/* On TH4, outer_l2_header means INCOMING_TAG_STATUS.
* TH4 only supports single tagging, so if TAG_STATUS
* says there's a tag, then we don't want to strip.
* Otherwise, we do.
*/
outer_l2_hdr = (valptr[fd_index] >> 13) & 3;
if (outer_l2_hdr)
{
tag_status = 2;
#ifdef KNET_CB_DEBUG
if (debug & 0x1)
{
printk(" Incoming frame tagged\n");
}
#endif
}
else
{
tag_status = 0;
#ifdef KNET_CB_DEBUG
if (debug & 0x1)
{
printk(" Incoming frame untagged\n");
}
#endif
}
}
#ifdef KNET_CB_DEBUG
if (debug & 0x1) {
printk("%s; Device Type: %d; tag status: %d\n", __func__, dev_type, tag_status);
}
#endif
return tag_status;
}
#ifdef KNET_CB_DEBUG
static void
dump_buffer(uint8_t * data, int size)
{
const char *const to_hex = "0123456789ABCDEF";
int i;
char buffer[128];
char *buffer_ptr;
int addr = 0;
buffer_ptr = buffer;
for (i = 0; i < size; i++) {
*buffer_ptr++ = ' ';
*buffer_ptr++ = to_hex[(data[i] >> 4) & 0xF];
*buffer_ptr++ = to_hex[data[i] & 0xF];
if (((i % 16) == 15) || (i == size - 1)) {
*buffer_ptr = '\0';
buffer_ptr = buffer;
printk(KERN_INFO "%04X %s\n", addr, buffer);
addr = i + 1;
}
}
}
static void
show_pmd(uint8_t *pmd, int len)
{
if (debug & 0x1) {
printk("PMD (%d bytes):\n", len);
dump_buffer(pmd, len);
}
}
static void
show_mac(uint8_t *pkt)
{
if (debug & 0x1) {
printk("DMAC=%02X:%02X:%02X:%02X:%02X:%02X\n",
pkt[0], pkt[1], pkt[2], pkt[3], pkt[4], pkt[5]);
}
}
#endif
static struct sk_buff *
strip_tag_rx_cb(struct sk_buff *skb)
{
const struct ngknet_callback_desc *cbd = NGKNET_SKB_CB(skb);
const struct ngknet_private *priv = cbd->priv;
int rcpu_mode = 0;
int tag_status;
rcpu_mode = (priv->flags & NGKNET_NETIF_F_RCPU_ENCAP)? 1 : 0;
#ifdef KNET_CB_DEBUG
if (debug & 0x1)
{
printk(KERN_INFO
"\n%4u --------------------------------------------------------------------------------\n",
rx_count);
printk(KERN_INFO
"RX KNET callback: dev_no=%1d; dev_id=0x%04X; type_str=%4s; RCPU: %3s \n",
cbd->dev_no, cbd->dev_id, cbd->type_str, rcpu_mode ? "yes" : "no");
printk(KERN_INFO " pkt_len=%4d; pmd_len=%2d; SKB len: %4d\n",
cbd->pkt_len, cbd->pmd_len, skb->len);
if (cbd->filt) {
printk(KERN_INFO "Filter user data: 0x%08x\n",
*(uint32_t *) cbd->filt->user_data);
}
printk(KERN_INFO "Before SKB (%d bytes):\n", skb->len);
dump_buffer(skb->data, skb->len);
printk("rx_cb for dev %d: id 0x%x, %s\n", cbd->dev_no, cbd->dev_id, cbd->type_str);
printk("netif user data: 0x%08x\n", *(uint32_t *)cbd->priv->user_data);
show_pmd(cbd->pmd, cbd->pmd_len);
if (rcpu_mode) {
const int RCPU_header_len = PKT_HDR_SIZE + cbd->pmd_len;
const int payload_len = skb->len - RCPU_header_len;
unsigned char *payload_start = skb->data + payload_len;
printk(KERN_INFO "Packet Payload (%d bytes):\n", payload_len);
dump_buffer(payload_start, payload_len);
} else {
printk(KERN_INFO "Packet (%d bytes):\n", cbd->pkt_len);
dump_buffer(skb->data, cbd->pkt_len);
}
}
#endif
if ((!rcpu_mode) && (cbd->filt)) {
if (FILTER_TAG_ORIGINAL == cbd->filt->user_data[0]) {
tag_status = get_tag_status(cbd->dev_id, cbd->filt->user_data[1],(void *)cbd->pmd);
if (tag_status < 0) {
strip_stats.skipped++;
goto _strip_tag_rx_cb_exit;
}
strip_stats.checked++;
if (tag_status < 2) {
strip_stats.stripped++;
strip_vlan_tag(skb);
}
}
if (FILTER_TAG_STRIP == cbd->filt->user_data[0]) {
strip_stats.stripped++;
strip_vlan_tag(skb);
}
}
_strip_tag_rx_cb_exit:
#ifdef KNET_CB_DEBUG
if (debug & 0x1) {
printk(KERN_INFO "After SKB (%d bytes):\n", skb->len);
dump_buffer(skb->data, skb->len);
printk(KERN_INFO
"\n%4u --------------------------------------------------------------------------------\n",
rx_count++);
}
#endif
return skb;
}
static struct sk_buff *
strip_tag_tx_cb(struct sk_buff *skb)
{
#ifdef KNET_CB_DEBUG
struct ngknet_callback_desc *cbd = NGKNET_SKB_CB(skb);
if (debug & 0x1) {
printk("tx_cb for dev %d: %s\n", cbd->dev_no, cbd->type_str);
}
show_pmd(cbd->pmd, cbd->pmd_len);
show_mac(cbd->pmd + cbd->pmd_len);
#endif
return skb;
}
static struct sk_buff *
ngknet_rx_cb(struct sk_buff *skb)
{
skb = strip_tag_rx_cb(skb);
#if IS_ENABLED(CONFIG_PSAMPLE)
skb = psample_rx_cb(skb);
#endif
return skb;
}
static struct sk_buff *
ngknet_tx_cb(struct sk_buff *skb)
{
skb = strip_tag_tx_cb(skb);
return skb;
}
static int
ngknet_netif_create_cb(struct net_device *dev)
{
int retv = 0;
#if IS_ENABLED(CONFIG_PSAMPLE)
retv = psample_netif_create_cb(dev);
#endif
return retv;
}
static int
ngknet_netif_destroy_cb(struct net_device *dev)
{
int retv = 0;
#if IS_ENABLED(CONFIG_PSAMPLE)
retv = psample_netif_destroy_cb(dev);
#endif
return retv;
}
/*!
* Generic module functions
*/
static int
ngknetcb_show(struct seq_file *m, void *v)
{
seq_printf(m, "Broadcom Linux NGKNET Callback: Untagged VLAN Stripper\n");
seq_printf(m, " %lu stripped packets\n", strip_stats.stripped);
seq_printf(m, " %lu packets checked\n", strip_stats.checked);
seq_printf(m, " %lu packets skipped\n", strip_stats.skipped);
return 0;
}
static int
ngknetcb_open(struct inode *inode, struct file *filp)
{
return single_open(filp, ngknetcb_show, NULL);
}
static int
ngknetcb_release(struct inode *inode, struct file *filp)
{
return 0;
}
static ssize_t
ngknetcb_write(struct file *file, const char *buf,
size_t count, loff_t *loff)
{
memset(&strip_stats, 0, sizeof(strip_stats));
printk("Cleared NGKNET callback stats\n");
return count;
}
static long
ngknetcb_ioctl(struct file *file, unsigned int cmd, unsigned long arg)
{
return 0;
}
static int
ngknetcb_mmap(struct file *filp, struct vm_area_struct *vma)
{
return 0;
}
static struct file_operations ngknetcb_fops = {
.owner = THIS_MODULE,
.open = ngknetcb_open,
.read = seq_read,
.write = ngknetcb_write,
.llseek = seq_lseek,
.release = ngknetcb_release,
.unlocked_ioctl = ngknetcb_ioctl,
.compat_ioctl = ngknetcb_ioctl,
.mmap = ngknetcb_mmap,
};
static struct proc_ops ngknetcb_proc_ops = {
.proc_open = ngknetcb_open,
.proc_read = seq_read,
.proc_write = ngknetcb_write,
.proc_lseek = seq_lseek,
.proc_release = ngknetcb_release,
.proc_ioctl = ngknetcb_ioctl,
.proc_compat_ioctl = ngknetcb_ioctl,
.proc_mmap = ngknetcb_mmap,
};
static int __init
ngknetcb_init_module(void)
{
int rv;
struct proc_dir_entry *entry = NULL;
rv = register_chrdev(NGKNETCB_MODULE_MAJOR, NGKNETCB_MODULE_NAME, &ngknetcb_fops);
if (rv < 0) {
printk(KERN_WARNING "%s: can't get major %d\n",
NGKNETCB_MODULE_NAME, NGKNETCB_MODULE_MAJOR);
return rv;
}
PROC_CREATE(entry, NGKNETCB_MODULE_NAME, 0666, NULL, &ngknetcb_proc_ops);
if (entry == NULL) {
printk(KERN_ERR "%s: proc_mkdir failed\n",
NGKNETCB_MODULE_NAME);
}
ngknet_rx_cb_register(ngknet_rx_cb);
ngknet_tx_cb_register(ngknet_tx_cb);
#if IS_ENABLED(CONFIG_PSAMPLE)
psample_init();
#endif
ngknet_netif_create_cb_register(ngknet_netif_create_cb);
ngknet_netif_destroy_cb_register(ngknet_netif_destroy_cb);
return 0;
}
static void __exit
ngknetcb_exit_module(void)
{
ngknet_netif_create_cb_unregister(ngknet_netif_create_cb);
ngknet_netif_destroy_cb_unregister(ngknet_netif_destroy_cb);
#if IS_ENABLED(CONFIG_PSAMPLE)
psample_cleanup();
#endif
ngknet_rx_cb_unregister(ngknet_rx_cb);
ngknet_tx_cb_unregister(ngknet_tx_cb);
remove_proc_entry(NGKNETCB_MODULE_NAME, NULL);
unregister_chrdev(NGKNETCB_MODULE_MAJOR, NGKNETCB_MODULE_NAME);
}
module_init(ngknetcb_init_module);
module_exit(ngknetcb_exit_module);