162306a36Sopenharmony_ci// SPDX-License-Identifier: GPL-2.0 262306a36Sopenharmony_ci/* Copyright (c) 2021 Mellanox Technologies. All rights reserved */ 362306a36Sopenharmony_ci 462306a36Sopenharmony_ci#include <linux/debugfs.h> 562306a36Sopenharmony_ci#include <linux/err.h> 662306a36Sopenharmony_ci#include <linux/etherdevice.h> 762306a36Sopenharmony_ci#include <linux/inet.h> 862306a36Sopenharmony_ci#include <linux/kernel.h> 962306a36Sopenharmony_ci#include <linux/random.h> 1062306a36Sopenharmony_ci#include <linux/slab.h> 1162306a36Sopenharmony_ci#include <net/devlink.h> 1262306a36Sopenharmony_ci#include <net/ip.h> 1362306a36Sopenharmony_ci#include <net/psample.h> 1462306a36Sopenharmony_ci#include <uapi/linux/ip.h> 1562306a36Sopenharmony_ci#include <uapi/linux/udp.h> 1662306a36Sopenharmony_ci 1762306a36Sopenharmony_ci#include "netdevsim.h" 1862306a36Sopenharmony_ci 1962306a36Sopenharmony_ci#define NSIM_PSAMPLE_REPORT_INTERVAL_MS 100 2062306a36Sopenharmony_ci#define NSIM_PSAMPLE_INVALID_TC 0xFFFF 2162306a36Sopenharmony_ci#define NSIM_PSAMPLE_L4_DATA_LEN 100 2262306a36Sopenharmony_ci 2362306a36Sopenharmony_cistruct nsim_dev_psample { 2462306a36Sopenharmony_ci struct delayed_work psample_dw; 2562306a36Sopenharmony_ci struct dentry *ddir; 2662306a36Sopenharmony_ci struct psample_group *group; 2762306a36Sopenharmony_ci u32 rate; 2862306a36Sopenharmony_ci u32 group_num; 2962306a36Sopenharmony_ci u32 trunc_size; 3062306a36Sopenharmony_ci int in_ifindex; 3162306a36Sopenharmony_ci int out_ifindex; 3262306a36Sopenharmony_ci u16 out_tc; 3362306a36Sopenharmony_ci u64 out_tc_occ_max; 3462306a36Sopenharmony_ci u64 latency_max; 3562306a36Sopenharmony_ci bool is_active; 3662306a36Sopenharmony_ci}; 3762306a36Sopenharmony_ci 3862306a36Sopenharmony_cistatic struct sk_buff *nsim_dev_psample_skb_build(void) 3962306a36Sopenharmony_ci{ 4062306a36Sopenharmony_ci int tot_len, data_len = NSIM_PSAMPLE_L4_DATA_LEN; 4162306a36Sopenharmony_ci struct sk_buff *skb; 4262306a36Sopenharmony_ci struct udphdr *udph; 4362306a36Sopenharmony_ci struct ethhdr *eth; 4462306a36Sopenharmony_ci struct iphdr *iph; 4562306a36Sopenharmony_ci 4662306a36Sopenharmony_ci skb = alloc_skb(NLMSG_GOODSIZE, GFP_KERNEL); 4762306a36Sopenharmony_ci if (!skb) 4862306a36Sopenharmony_ci return NULL; 4962306a36Sopenharmony_ci tot_len = sizeof(struct iphdr) + sizeof(struct udphdr) + data_len; 5062306a36Sopenharmony_ci 5162306a36Sopenharmony_ci skb_reset_mac_header(skb); 5262306a36Sopenharmony_ci eth = skb_put(skb, sizeof(struct ethhdr)); 5362306a36Sopenharmony_ci eth_random_addr(eth->h_dest); 5462306a36Sopenharmony_ci eth_random_addr(eth->h_source); 5562306a36Sopenharmony_ci eth->h_proto = htons(ETH_P_IP); 5662306a36Sopenharmony_ci skb->protocol = htons(ETH_P_IP); 5762306a36Sopenharmony_ci 5862306a36Sopenharmony_ci skb_set_network_header(skb, skb->len); 5962306a36Sopenharmony_ci iph = skb_put(skb, sizeof(struct iphdr)); 6062306a36Sopenharmony_ci iph->protocol = IPPROTO_UDP; 6162306a36Sopenharmony_ci iph->saddr = in_aton("192.0.2.1"); 6262306a36Sopenharmony_ci iph->daddr = in_aton("198.51.100.1"); 6362306a36Sopenharmony_ci iph->version = 0x4; 6462306a36Sopenharmony_ci iph->frag_off = 0; 6562306a36Sopenharmony_ci iph->ihl = 0x5; 6662306a36Sopenharmony_ci iph->tot_len = htons(tot_len); 6762306a36Sopenharmony_ci iph->id = 0; 6862306a36Sopenharmony_ci iph->ttl = 100; 6962306a36Sopenharmony_ci iph->check = 0; 7062306a36Sopenharmony_ci iph->check = ip_fast_csum((unsigned char *)iph, iph->ihl); 7162306a36Sopenharmony_ci 7262306a36Sopenharmony_ci skb_set_transport_header(skb, skb->len); 7362306a36Sopenharmony_ci udph = skb_put_zero(skb, sizeof(struct udphdr) + data_len); 7462306a36Sopenharmony_ci get_random_bytes(&udph->source, sizeof(u16)); 7562306a36Sopenharmony_ci get_random_bytes(&udph->dest, sizeof(u16)); 7662306a36Sopenharmony_ci udph->len = htons(sizeof(struct udphdr) + data_len); 7762306a36Sopenharmony_ci 7862306a36Sopenharmony_ci return skb; 7962306a36Sopenharmony_ci} 8062306a36Sopenharmony_ci 8162306a36Sopenharmony_cistatic void nsim_dev_psample_md_prepare(const struct nsim_dev_psample *psample, 8262306a36Sopenharmony_ci struct psample_metadata *md, 8362306a36Sopenharmony_ci unsigned int len) 8462306a36Sopenharmony_ci{ 8562306a36Sopenharmony_ci md->trunc_size = psample->trunc_size ? psample->trunc_size : len; 8662306a36Sopenharmony_ci md->in_ifindex = psample->in_ifindex; 8762306a36Sopenharmony_ci md->out_ifindex = psample->out_ifindex; 8862306a36Sopenharmony_ci 8962306a36Sopenharmony_ci if (psample->out_tc != NSIM_PSAMPLE_INVALID_TC) { 9062306a36Sopenharmony_ci md->out_tc = psample->out_tc; 9162306a36Sopenharmony_ci md->out_tc_valid = 1; 9262306a36Sopenharmony_ci } 9362306a36Sopenharmony_ci 9462306a36Sopenharmony_ci if (psample->out_tc_occ_max) { 9562306a36Sopenharmony_ci u64 out_tc_occ; 9662306a36Sopenharmony_ci 9762306a36Sopenharmony_ci get_random_bytes(&out_tc_occ, sizeof(u64)); 9862306a36Sopenharmony_ci md->out_tc_occ = out_tc_occ & (psample->out_tc_occ_max - 1); 9962306a36Sopenharmony_ci md->out_tc_occ_valid = 1; 10062306a36Sopenharmony_ci } 10162306a36Sopenharmony_ci 10262306a36Sopenharmony_ci if (psample->latency_max) { 10362306a36Sopenharmony_ci u64 latency; 10462306a36Sopenharmony_ci 10562306a36Sopenharmony_ci get_random_bytes(&latency, sizeof(u64)); 10662306a36Sopenharmony_ci md->latency = latency & (psample->latency_max - 1); 10762306a36Sopenharmony_ci md->latency_valid = 1; 10862306a36Sopenharmony_ci } 10962306a36Sopenharmony_ci} 11062306a36Sopenharmony_ci 11162306a36Sopenharmony_cistatic void nsim_dev_psample_report_work(struct work_struct *work) 11262306a36Sopenharmony_ci{ 11362306a36Sopenharmony_ci struct nsim_dev_psample *psample; 11462306a36Sopenharmony_ci struct psample_metadata md = {}; 11562306a36Sopenharmony_ci struct sk_buff *skb; 11662306a36Sopenharmony_ci unsigned long delay; 11762306a36Sopenharmony_ci 11862306a36Sopenharmony_ci psample = container_of(work, struct nsim_dev_psample, psample_dw.work); 11962306a36Sopenharmony_ci 12062306a36Sopenharmony_ci skb = nsim_dev_psample_skb_build(); 12162306a36Sopenharmony_ci if (!skb) 12262306a36Sopenharmony_ci goto out; 12362306a36Sopenharmony_ci 12462306a36Sopenharmony_ci nsim_dev_psample_md_prepare(psample, &md, skb->len); 12562306a36Sopenharmony_ci psample_sample_packet(psample->group, skb, psample->rate, &md); 12662306a36Sopenharmony_ci consume_skb(skb); 12762306a36Sopenharmony_ci 12862306a36Sopenharmony_ciout: 12962306a36Sopenharmony_ci delay = msecs_to_jiffies(NSIM_PSAMPLE_REPORT_INTERVAL_MS); 13062306a36Sopenharmony_ci schedule_delayed_work(&psample->psample_dw, delay); 13162306a36Sopenharmony_ci} 13262306a36Sopenharmony_ci 13362306a36Sopenharmony_cistatic int nsim_dev_psample_enable(struct nsim_dev *nsim_dev) 13462306a36Sopenharmony_ci{ 13562306a36Sopenharmony_ci struct nsim_dev_psample *psample = nsim_dev->psample; 13662306a36Sopenharmony_ci struct devlink *devlink; 13762306a36Sopenharmony_ci unsigned long delay; 13862306a36Sopenharmony_ci 13962306a36Sopenharmony_ci if (psample->is_active) 14062306a36Sopenharmony_ci return -EBUSY; 14162306a36Sopenharmony_ci 14262306a36Sopenharmony_ci devlink = priv_to_devlink(nsim_dev); 14362306a36Sopenharmony_ci psample->group = psample_group_get(devlink_net(devlink), 14462306a36Sopenharmony_ci psample->group_num); 14562306a36Sopenharmony_ci if (!psample->group) 14662306a36Sopenharmony_ci return -EINVAL; 14762306a36Sopenharmony_ci 14862306a36Sopenharmony_ci delay = msecs_to_jiffies(NSIM_PSAMPLE_REPORT_INTERVAL_MS); 14962306a36Sopenharmony_ci schedule_delayed_work(&psample->psample_dw, delay); 15062306a36Sopenharmony_ci 15162306a36Sopenharmony_ci psample->is_active = true; 15262306a36Sopenharmony_ci 15362306a36Sopenharmony_ci return 0; 15462306a36Sopenharmony_ci} 15562306a36Sopenharmony_ci 15662306a36Sopenharmony_cistatic int nsim_dev_psample_disable(struct nsim_dev *nsim_dev) 15762306a36Sopenharmony_ci{ 15862306a36Sopenharmony_ci struct nsim_dev_psample *psample = nsim_dev->psample; 15962306a36Sopenharmony_ci 16062306a36Sopenharmony_ci if (!psample->is_active) 16162306a36Sopenharmony_ci return -EINVAL; 16262306a36Sopenharmony_ci 16362306a36Sopenharmony_ci psample->is_active = false; 16462306a36Sopenharmony_ci 16562306a36Sopenharmony_ci cancel_delayed_work_sync(&psample->psample_dw); 16662306a36Sopenharmony_ci psample_group_put(psample->group); 16762306a36Sopenharmony_ci 16862306a36Sopenharmony_ci return 0; 16962306a36Sopenharmony_ci} 17062306a36Sopenharmony_ci 17162306a36Sopenharmony_cistatic ssize_t nsim_dev_psample_enable_write(struct file *file, 17262306a36Sopenharmony_ci const char __user *data, 17362306a36Sopenharmony_ci size_t count, loff_t *ppos) 17462306a36Sopenharmony_ci{ 17562306a36Sopenharmony_ci struct nsim_dev *nsim_dev = file->private_data; 17662306a36Sopenharmony_ci bool enable; 17762306a36Sopenharmony_ci int err; 17862306a36Sopenharmony_ci 17962306a36Sopenharmony_ci err = kstrtobool_from_user(data, count, &enable); 18062306a36Sopenharmony_ci if (err) 18162306a36Sopenharmony_ci return err; 18262306a36Sopenharmony_ci 18362306a36Sopenharmony_ci if (enable) 18462306a36Sopenharmony_ci err = nsim_dev_psample_enable(nsim_dev); 18562306a36Sopenharmony_ci else 18662306a36Sopenharmony_ci err = nsim_dev_psample_disable(nsim_dev); 18762306a36Sopenharmony_ci 18862306a36Sopenharmony_ci return err ? err : count; 18962306a36Sopenharmony_ci} 19062306a36Sopenharmony_ci 19162306a36Sopenharmony_cistatic const struct file_operations nsim_psample_enable_fops = { 19262306a36Sopenharmony_ci .open = simple_open, 19362306a36Sopenharmony_ci .write = nsim_dev_psample_enable_write, 19462306a36Sopenharmony_ci .llseek = generic_file_llseek, 19562306a36Sopenharmony_ci .owner = THIS_MODULE, 19662306a36Sopenharmony_ci}; 19762306a36Sopenharmony_ci 19862306a36Sopenharmony_ciint nsim_dev_psample_init(struct nsim_dev *nsim_dev) 19962306a36Sopenharmony_ci{ 20062306a36Sopenharmony_ci struct nsim_dev_psample *psample; 20162306a36Sopenharmony_ci int err; 20262306a36Sopenharmony_ci 20362306a36Sopenharmony_ci psample = kzalloc(sizeof(*psample), GFP_KERNEL); 20462306a36Sopenharmony_ci if (!psample) 20562306a36Sopenharmony_ci return -ENOMEM; 20662306a36Sopenharmony_ci nsim_dev->psample = psample; 20762306a36Sopenharmony_ci 20862306a36Sopenharmony_ci INIT_DELAYED_WORK(&psample->psample_dw, nsim_dev_psample_report_work); 20962306a36Sopenharmony_ci 21062306a36Sopenharmony_ci psample->ddir = debugfs_create_dir("psample", nsim_dev->ddir); 21162306a36Sopenharmony_ci if (IS_ERR(psample->ddir)) { 21262306a36Sopenharmony_ci err = PTR_ERR(psample->ddir); 21362306a36Sopenharmony_ci goto err_psample_free; 21462306a36Sopenharmony_ci } 21562306a36Sopenharmony_ci 21662306a36Sopenharmony_ci /* Populate sampling parameters with sane defaults. */ 21762306a36Sopenharmony_ci psample->rate = 100; 21862306a36Sopenharmony_ci debugfs_create_u32("rate", 0600, psample->ddir, &psample->rate); 21962306a36Sopenharmony_ci 22062306a36Sopenharmony_ci psample->group_num = 10; 22162306a36Sopenharmony_ci debugfs_create_u32("group_num", 0600, psample->ddir, 22262306a36Sopenharmony_ci &psample->group_num); 22362306a36Sopenharmony_ci 22462306a36Sopenharmony_ci psample->trunc_size = 0; 22562306a36Sopenharmony_ci debugfs_create_u32("trunc_size", 0600, psample->ddir, 22662306a36Sopenharmony_ci &psample->trunc_size); 22762306a36Sopenharmony_ci 22862306a36Sopenharmony_ci psample->in_ifindex = 1; 22962306a36Sopenharmony_ci debugfs_create_u32("in_ifindex", 0600, psample->ddir, 23062306a36Sopenharmony_ci &psample->in_ifindex); 23162306a36Sopenharmony_ci 23262306a36Sopenharmony_ci psample->out_ifindex = 2; 23362306a36Sopenharmony_ci debugfs_create_u32("out_ifindex", 0600, psample->ddir, 23462306a36Sopenharmony_ci &psample->out_ifindex); 23562306a36Sopenharmony_ci 23662306a36Sopenharmony_ci psample->out_tc = 0; 23762306a36Sopenharmony_ci debugfs_create_u16("out_tc", 0600, psample->ddir, &psample->out_tc); 23862306a36Sopenharmony_ci 23962306a36Sopenharmony_ci psample->out_tc_occ_max = 10000; 24062306a36Sopenharmony_ci debugfs_create_u64("out_tc_occ_max", 0600, psample->ddir, 24162306a36Sopenharmony_ci &psample->out_tc_occ_max); 24262306a36Sopenharmony_ci 24362306a36Sopenharmony_ci psample->latency_max = 50; 24462306a36Sopenharmony_ci debugfs_create_u64("latency_max", 0600, psample->ddir, 24562306a36Sopenharmony_ci &psample->latency_max); 24662306a36Sopenharmony_ci 24762306a36Sopenharmony_ci debugfs_create_file("enable", 0200, psample->ddir, nsim_dev, 24862306a36Sopenharmony_ci &nsim_psample_enable_fops); 24962306a36Sopenharmony_ci 25062306a36Sopenharmony_ci return 0; 25162306a36Sopenharmony_ci 25262306a36Sopenharmony_cierr_psample_free: 25362306a36Sopenharmony_ci kfree(nsim_dev->psample); 25462306a36Sopenharmony_ci return err; 25562306a36Sopenharmony_ci} 25662306a36Sopenharmony_ci 25762306a36Sopenharmony_civoid nsim_dev_psample_exit(struct nsim_dev *nsim_dev) 25862306a36Sopenharmony_ci{ 25962306a36Sopenharmony_ci debugfs_remove_recursive(nsim_dev->psample->ddir); 26062306a36Sopenharmony_ci if (nsim_dev->psample->is_active) { 26162306a36Sopenharmony_ci cancel_delayed_work_sync(&nsim_dev->psample->psample_dw); 26262306a36Sopenharmony_ci psample_group_put(nsim_dev->psample->group); 26362306a36Sopenharmony_ci } 26462306a36Sopenharmony_ci kfree(nsim_dev->psample); 26562306a36Sopenharmony_ci} 266