162306a36Sopenharmony_ci// SPDX-License-Identifier: (GPL-2.0 OR BSD-3-Clause) 262306a36Sopenharmony_ci/* 362306a36Sopenharmony_ci * proc.c - procfs support for Protocol family CAN core module 462306a36Sopenharmony_ci * 562306a36Sopenharmony_ci * Copyright (c) 2002-2007 Volkswagen Group Electronic Research 662306a36Sopenharmony_ci * All rights reserved. 762306a36Sopenharmony_ci * 862306a36Sopenharmony_ci * Redistribution and use in source and binary forms, with or without 962306a36Sopenharmony_ci * modification, are permitted provided that the following conditions 1062306a36Sopenharmony_ci * are met: 1162306a36Sopenharmony_ci * 1. Redistributions of source code must retain the above copyright 1262306a36Sopenharmony_ci * notice, this list of conditions and the following disclaimer. 1362306a36Sopenharmony_ci * 2. Redistributions in binary form must reproduce the above copyright 1462306a36Sopenharmony_ci * notice, this list of conditions and the following disclaimer in the 1562306a36Sopenharmony_ci * documentation and/or other materials provided with the distribution. 1662306a36Sopenharmony_ci * 3. Neither the name of Volkswagen nor the names of its contributors 1762306a36Sopenharmony_ci * may be used to endorse or promote products derived from this software 1862306a36Sopenharmony_ci * without specific prior written permission. 1962306a36Sopenharmony_ci * 2062306a36Sopenharmony_ci * Alternatively, provided that this notice is retained in full, this 2162306a36Sopenharmony_ci * software may be distributed under the terms of the GNU General 2262306a36Sopenharmony_ci * Public License ("GPL") version 2, in which case the provisions of the 2362306a36Sopenharmony_ci * GPL apply INSTEAD OF those given above. 2462306a36Sopenharmony_ci * 2562306a36Sopenharmony_ci * The provided data structures and external interfaces from this code 2662306a36Sopenharmony_ci * are not restricted to be used by modules with a GPL compatible license. 2762306a36Sopenharmony_ci * 2862306a36Sopenharmony_ci * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS 2962306a36Sopenharmony_ci * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT 3062306a36Sopenharmony_ci * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR 3162306a36Sopenharmony_ci * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT 3262306a36Sopenharmony_ci * OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, 3362306a36Sopenharmony_ci * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT 3462306a36Sopenharmony_ci * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, 3562306a36Sopenharmony_ci * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY 3662306a36Sopenharmony_ci * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT 3762306a36Sopenharmony_ci * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE 3862306a36Sopenharmony_ci * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH 3962306a36Sopenharmony_ci * DAMAGE. 4062306a36Sopenharmony_ci * 4162306a36Sopenharmony_ci */ 4262306a36Sopenharmony_ci 4362306a36Sopenharmony_ci#include <linux/module.h> 4462306a36Sopenharmony_ci#include <linux/proc_fs.h> 4562306a36Sopenharmony_ci#include <linux/list.h> 4662306a36Sopenharmony_ci#include <linux/rcupdate.h> 4762306a36Sopenharmony_ci#include <linux/if_arp.h> 4862306a36Sopenharmony_ci#include <linux/can/can-ml.h> 4962306a36Sopenharmony_ci#include <linux/can/core.h> 5062306a36Sopenharmony_ci 5162306a36Sopenharmony_ci#include "af_can.h" 5262306a36Sopenharmony_ci 5362306a36Sopenharmony_ci/* 5462306a36Sopenharmony_ci * proc filenames for the PF_CAN core 5562306a36Sopenharmony_ci */ 5662306a36Sopenharmony_ci 5762306a36Sopenharmony_ci#define CAN_PROC_STATS "stats" 5862306a36Sopenharmony_ci#define CAN_PROC_RESET_STATS "reset_stats" 5962306a36Sopenharmony_ci#define CAN_PROC_RCVLIST_ALL "rcvlist_all" 6062306a36Sopenharmony_ci#define CAN_PROC_RCVLIST_FIL "rcvlist_fil" 6162306a36Sopenharmony_ci#define CAN_PROC_RCVLIST_INV "rcvlist_inv" 6262306a36Sopenharmony_ci#define CAN_PROC_RCVLIST_SFF "rcvlist_sff" 6362306a36Sopenharmony_ci#define CAN_PROC_RCVLIST_EFF "rcvlist_eff" 6462306a36Sopenharmony_ci#define CAN_PROC_RCVLIST_ERR "rcvlist_err" 6562306a36Sopenharmony_ci 6662306a36Sopenharmony_cistatic int user_reset; 6762306a36Sopenharmony_ci 6862306a36Sopenharmony_cistatic const char rx_list_name[][8] = { 6962306a36Sopenharmony_ci [RX_ERR] = "rx_err", 7062306a36Sopenharmony_ci [RX_ALL] = "rx_all", 7162306a36Sopenharmony_ci [RX_FIL] = "rx_fil", 7262306a36Sopenharmony_ci [RX_INV] = "rx_inv", 7362306a36Sopenharmony_ci}; 7462306a36Sopenharmony_ci 7562306a36Sopenharmony_ci/* 7662306a36Sopenharmony_ci * af_can statistics stuff 7762306a36Sopenharmony_ci */ 7862306a36Sopenharmony_ci 7962306a36Sopenharmony_cistatic void can_init_stats(struct net *net) 8062306a36Sopenharmony_ci{ 8162306a36Sopenharmony_ci struct can_pkg_stats *pkg_stats = net->can.pkg_stats; 8262306a36Sopenharmony_ci struct can_rcv_lists_stats *rcv_lists_stats = net->can.rcv_lists_stats; 8362306a36Sopenharmony_ci /* 8462306a36Sopenharmony_ci * This memset function is called from a timer context (when 8562306a36Sopenharmony_ci * can_stattimer is active which is the default) OR in a process 8662306a36Sopenharmony_ci * context (reading the proc_fs when can_stattimer is disabled). 8762306a36Sopenharmony_ci */ 8862306a36Sopenharmony_ci memset(pkg_stats, 0, sizeof(struct can_pkg_stats)); 8962306a36Sopenharmony_ci pkg_stats->jiffies_init = jiffies; 9062306a36Sopenharmony_ci 9162306a36Sopenharmony_ci rcv_lists_stats->stats_reset++; 9262306a36Sopenharmony_ci 9362306a36Sopenharmony_ci if (user_reset) { 9462306a36Sopenharmony_ci user_reset = 0; 9562306a36Sopenharmony_ci rcv_lists_stats->user_reset++; 9662306a36Sopenharmony_ci } 9762306a36Sopenharmony_ci} 9862306a36Sopenharmony_ci 9962306a36Sopenharmony_cistatic unsigned long calc_rate(unsigned long oldjif, unsigned long newjif, 10062306a36Sopenharmony_ci unsigned long count) 10162306a36Sopenharmony_ci{ 10262306a36Sopenharmony_ci if (oldjif == newjif) 10362306a36Sopenharmony_ci return 0; 10462306a36Sopenharmony_ci 10562306a36Sopenharmony_ci /* see can_stat_update() - this should NEVER happen! */ 10662306a36Sopenharmony_ci if (count > (ULONG_MAX / HZ)) { 10762306a36Sopenharmony_ci printk(KERN_ERR "can: calc_rate: count exceeded! %ld\n", 10862306a36Sopenharmony_ci count); 10962306a36Sopenharmony_ci return 99999999; 11062306a36Sopenharmony_ci } 11162306a36Sopenharmony_ci 11262306a36Sopenharmony_ci return (count * HZ) / (newjif - oldjif); 11362306a36Sopenharmony_ci} 11462306a36Sopenharmony_ci 11562306a36Sopenharmony_civoid can_stat_update(struct timer_list *t) 11662306a36Sopenharmony_ci{ 11762306a36Sopenharmony_ci struct net *net = from_timer(net, t, can.stattimer); 11862306a36Sopenharmony_ci struct can_pkg_stats *pkg_stats = net->can.pkg_stats; 11962306a36Sopenharmony_ci unsigned long j = jiffies; /* snapshot */ 12062306a36Sopenharmony_ci 12162306a36Sopenharmony_ci /* restart counting in timer context on user request */ 12262306a36Sopenharmony_ci if (user_reset) 12362306a36Sopenharmony_ci can_init_stats(net); 12462306a36Sopenharmony_ci 12562306a36Sopenharmony_ci /* restart counting on jiffies overflow */ 12662306a36Sopenharmony_ci if (j < pkg_stats->jiffies_init) 12762306a36Sopenharmony_ci can_init_stats(net); 12862306a36Sopenharmony_ci 12962306a36Sopenharmony_ci /* prevent overflow in calc_rate() */ 13062306a36Sopenharmony_ci if (pkg_stats->rx_frames > (ULONG_MAX / HZ)) 13162306a36Sopenharmony_ci can_init_stats(net); 13262306a36Sopenharmony_ci 13362306a36Sopenharmony_ci /* prevent overflow in calc_rate() */ 13462306a36Sopenharmony_ci if (pkg_stats->tx_frames > (ULONG_MAX / HZ)) 13562306a36Sopenharmony_ci can_init_stats(net); 13662306a36Sopenharmony_ci 13762306a36Sopenharmony_ci /* matches overflow - very improbable */ 13862306a36Sopenharmony_ci if (pkg_stats->matches > (ULONG_MAX / 100)) 13962306a36Sopenharmony_ci can_init_stats(net); 14062306a36Sopenharmony_ci 14162306a36Sopenharmony_ci /* calc total values */ 14262306a36Sopenharmony_ci if (pkg_stats->rx_frames) 14362306a36Sopenharmony_ci pkg_stats->total_rx_match_ratio = (pkg_stats->matches * 100) / 14462306a36Sopenharmony_ci pkg_stats->rx_frames; 14562306a36Sopenharmony_ci 14662306a36Sopenharmony_ci pkg_stats->total_tx_rate = calc_rate(pkg_stats->jiffies_init, j, 14762306a36Sopenharmony_ci pkg_stats->tx_frames); 14862306a36Sopenharmony_ci pkg_stats->total_rx_rate = calc_rate(pkg_stats->jiffies_init, j, 14962306a36Sopenharmony_ci pkg_stats->rx_frames); 15062306a36Sopenharmony_ci 15162306a36Sopenharmony_ci /* calc current values */ 15262306a36Sopenharmony_ci if (pkg_stats->rx_frames_delta) 15362306a36Sopenharmony_ci pkg_stats->current_rx_match_ratio = 15462306a36Sopenharmony_ci (pkg_stats->matches_delta * 100) / 15562306a36Sopenharmony_ci pkg_stats->rx_frames_delta; 15662306a36Sopenharmony_ci 15762306a36Sopenharmony_ci pkg_stats->current_tx_rate = calc_rate(0, HZ, pkg_stats->tx_frames_delta); 15862306a36Sopenharmony_ci pkg_stats->current_rx_rate = calc_rate(0, HZ, pkg_stats->rx_frames_delta); 15962306a36Sopenharmony_ci 16062306a36Sopenharmony_ci /* check / update maximum values */ 16162306a36Sopenharmony_ci if (pkg_stats->max_tx_rate < pkg_stats->current_tx_rate) 16262306a36Sopenharmony_ci pkg_stats->max_tx_rate = pkg_stats->current_tx_rate; 16362306a36Sopenharmony_ci 16462306a36Sopenharmony_ci if (pkg_stats->max_rx_rate < pkg_stats->current_rx_rate) 16562306a36Sopenharmony_ci pkg_stats->max_rx_rate = pkg_stats->current_rx_rate; 16662306a36Sopenharmony_ci 16762306a36Sopenharmony_ci if (pkg_stats->max_rx_match_ratio < pkg_stats->current_rx_match_ratio) 16862306a36Sopenharmony_ci pkg_stats->max_rx_match_ratio = pkg_stats->current_rx_match_ratio; 16962306a36Sopenharmony_ci 17062306a36Sopenharmony_ci /* clear values for 'current rate' calculation */ 17162306a36Sopenharmony_ci pkg_stats->tx_frames_delta = 0; 17262306a36Sopenharmony_ci pkg_stats->rx_frames_delta = 0; 17362306a36Sopenharmony_ci pkg_stats->matches_delta = 0; 17462306a36Sopenharmony_ci 17562306a36Sopenharmony_ci /* restart timer (one second) */ 17662306a36Sopenharmony_ci mod_timer(&net->can.stattimer, round_jiffies(jiffies + HZ)); 17762306a36Sopenharmony_ci} 17862306a36Sopenharmony_ci 17962306a36Sopenharmony_ci/* 18062306a36Sopenharmony_ci * proc read functions 18162306a36Sopenharmony_ci */ 18262306a36Sopenharmony_ci 18362306a36Sopenharmony_cistatic void can_print_rcvlist(struct seq_file *m, struct hlist_head *rx_list, 18462306a36Sopenharmony_ci struct net_device *dev) 18562306a36Sopenharmony_ci{ 18662306a36Sopenharmony_ci struct receiver *r; 18762306a36Sopenharmony_ci 18862306a36Sopenharmony_ci hlist_for_each_entry_rcu(r, rx_list, list) { 18962306a36Sopenharmony_ci char *fmt = (r->can_id & CAN_EFF_FLAG)? 19062306a36Sopenharmony_ci " %-5s %08x %08x %pK %pK %8ld %s\n" : 19162306a36Sopenharmony_ci " %-5s %03x %08x %pK %pK %8ld %s\n"; 19262306a36Sopenharmony_ci 19362306a36Sopenharmony_ci seq_printf(m, fmt, DNAME(dev), r->can_id, r->mask, 19462306a36Sopenharmony_ci r->func, r->data, r->matches, r->ident); 19562306a36Sopenharmony_ci } 19662306a36Sopenharmony_ci} 19762306a36Sopenharmony_ci 19862306a36Sopenharmony_cistatic void can_print_recv_banner(struct seq_file *m) 19962306a36Sopenharmony_ci{ 20062306a36Sopenharmony_ci /* 20162306a36Sopenharmony_ci * can1. 00000000 00000000 00000000 20262306a36Sopenharmony_ci * ....... 0 tp20 20362306a36Sopenharmony_ci */ 20462306a36Sopenharmony_ci if (IS_ENABLED(CONFIG_64BIT)) 20562306a36Sopenharmony_ci seq_puts(m, " device can_id can_mask function userdata matches ident\n"); 20662306a36Sopenharmony_ci else 20762306a36Sopenharmony_ci seq_puts(m, " device can_id can_mask function userdata matches ident\n"); 20862306a36Sopenharmony_ci} 20962306a36Sopenharmony_ci 21062306a36Sopenharmony_cistatic int can_stats_proc_show(struct seq_file *m, void *v) 21162306a36Sopenharmony_ci{ 21262306a36Sopenharmony_ci struct net *net = m->private; 21362306a36Sopenharmony_ci struct can_pkg_stats *pkg_stats = net->can.pkg_stats; 21462306a36Sopenharmony_ci struct can_rcv_lists_stats *rcv_lists_stats = net->can.rcv_lists_stats; 21562306a36Sopenharmony_ci 21662306a36Sopenharmony_ci seq_putc(m, '\n'); 21762306a36Sopenharmony_ci seq_printf(m, " %8ld transmitted frames (TXF)\n", pkg_stats->tx_frames); 21862306a36Sopenharmony_ci seq_printf(m, " %8ld received frames (RXF)\n", pkg_stats->rx_frames); 21962306a36Sopenharmony_ci seq_printf(m, " %8ld matched frames (RXMF)\n", pkg_stats->matches); 22062306a36Sopenharmony_ci 22162306a36Sopenharmony_ci seq_putc(m, '\n'); 22262306a36Sopenharmony_ci 22362306a36Sopenharmony_ci if (net->can.stattimer.function == can_stat_update) { 22462306a36Sopenharmony_ci seq_printf(m, " %8ld %% total match ratio (RXMR)\n", 22562306a36Sopenharmony_ci pkg_stats->total_rx_match_ratio); 22662306a36Sopenharmony_ci 22762306a36Sopenharmony_ci seq_printf(m, " %8ld frames/s total tx rate (TXR)\n", 22862306a36Sopenharmony_ci pkg_stats->total_tx_rate); 22962306a36Sopenharmony_ci seq_printf(m, " %8ld frames/s total rx rate (RXR)\n", 23062306a36Sopenharmony_ci pkg_stats->total_rx_rate); 23162306a36Sopenharmony_ci 23262306a36Sopenharmony_ci seq_putc(m, '\n'); 23362306a36Sopenharmony_ci 23462306a36Sopenharmony_ci seq_printf(m, " %8ld %% current match ratio (CRXMR)\n", 23562306a36Sopenharmony_ci pkg_stats->current_rx_match_ratio); 23662306a36Sopenharmony_ci 23762306a36Sopenharmony_ci seq_printf(m, " %8ld frames/s current tx rate (CTXR)\n", 23862306a36Sopenharmony_ci pkg_stats->current_tx_rate); 23962306a36Sopenharmony_ci seq_printf(m, " %8ld frames/s current rx rate (CRXR)\n", 24062306a36Sopenharmony_ci pkg_stats->current_rx_rate); 24162306a36Sopenharmony_ci 24262306a36Sopenharmony_ci seq_putc(m, '\n'); 24362306a36Sopenharmony_ci 24462306a36Sopenharmony_ci seq_printf(m, " %8ld %% max match ratio (MRXMR)\n", 24562306a36Sopenharmony_ci pkg_stats->max_rx_match_ratio); 24662306a36Sopenharmony_ci 24762306a36Sopenharmony_ci seq_printf(m, " %8ld frames/s max tx rate (MTXR)\n", 24862306a36Sopenharmony_ci pkg_stats->max_tx_rate); 24962306a36Sopenharmony_ci seq_printf(m, " %8ld frames/s max rx rate (MRXR)\n", 25062306a36Sopenharmony_ci pkg_stats->max_rx_rate); 25162306a36Sopenharmony_ci 25262306a36Sopenharmony_ci seq_putc(m, '\n'); 25362306a36Sopenharmony_ci } 25462306a36Sopenharmony_ci 25562306a36Sopenharmony_ci seq_printf(m, " %8ld current receive list entries (CRCV)\n", 25662306a36Sopenharmony_ci rcv_lists_stats->rcv_entries); 25762306a36Sopenharmony_ci seq_printf(m, " %8ld maximum receive list entries (MRCV)\n", 25862306a36Sopenharmony_ci rcv_lists_stats->rcv_entries_max); 25962306a36Sopenharmony_ci 26062306a36Sopenharmony_ci if (rcv_lists_stats->stats_reset) 26162306a36Sopenharmony_ci seq_printf(m, "\n %8ld statistic resets (STR)\n", 26262306a36Sopenharmony_ci rcv_lists_stats->stats_reset); 26362306a36Sopenharmony_ci 26462306a36Sopenharmony_ci if (rcv_lists_stats->user_reset) 26562306a36Sopenharmony_ci seq_printf(m, " %8ld user statistic resets (USTR)\n", 26662306a36Sopenharmony_ci rcv_lists_stats->user_reset); 26762306a36Sopenharmony_ci 26862306a36Sopenharmony_ci seq_putc(m, '\n'); 26962306a36Sopenharmony_ci return 0; 27062306a36Sopenharmony_ci} 27162306a36Sopenharmony_ci 27262306a36Sopenharmony_cistatic int can_reset_stats_proc_show(struct seq_file *m, void *v) 27362306a36Sopenharmony_ci{ 27462306a36Sopenharmony_ci struct net *net = m->private; 27562306a36Sopenharmony_ci struct can_rcv_lists_stats *rcv_lists_stats = net->can.rcv_lists_stats; 27662306a36Sopenharmony_ci struct can_pkg_stats *pkg_stats = net->can.pkg_stats; 27762306a36Sopenharmony_ci 27862306a36Sopenharmony_ci user_reset = 1; 27962306a36Sopenharmony_ci 28062306a36Sopenharmony_ci if (net->can.stattimer.function == can_stat_update) { 28162306a36Sopenharmony_ci seq_printf(m, "Scheduled statistic reset #%ld.\n", 28262306a36Sopenharmony_ci rcv_lists_stats->stats_reset + 1); 28362306a36Sopenharmony_ci } else { 28462306a36Sopenharmony_ci if (pkg_stats->jiffies_init != jiffies) 28562306a36Sopenharmony_ci can_init_stats(net); 28662306a36Sopenharmony_ci 28762306a36Sopenharmony_ci seq_printf(m, "Performed statistic reset #%ld.\n", 28862306a36Sopenharmony_ci rcv_lists_stats->stats_reset); 28962306a36Sopenharmony_ci } 29062306a36Sopenharmony_ci return 0; 29162306a36Sopenharmony_ci} 29262306a36Sopenharmony_ci 29362306a36Sopenharmony_cistatic inline void can_rcvlist_proc_show_one(struct seq_file *m, int idx, 29462306a36Sopenharmony_ci struct net_device *dev, 29562306a36Sopenharmony_ci struct can_dev_rcv_lists *dev_rcv_lists) 29662306a36Sopenharmony_ci{ 29762306a36Sopenharmony_ci if (!hlist_empty(&dev_rcv_lists->rx[idx])) { 29862306a36Sopenharmony_ci can_print_recv_banner(m); 29962306a36Sopenharmony_ci can_print_rcvlist(m, &dev_rcv_lists->rx[idx], dev); 30062306a36Sopenharmony_ci } else 30162306a36Sopenharmony_ci seq_printf(m, " (%s: no entry)\n", DNAME(dev)); 30262306a36Sopenharmony_ci 30362306a36Sopenharmony_ci} 30462306a36Sopenharmony_ci 30562306a36Sopenharmony_cistatic int can_rcvlist_proc_show(struct seq_file *m, void *v) 30662306a36Sopenharmony_ci{ 30762306a36Sopenharmony_ci /* double cast to prevent GCC warning */ 30862306a36Sopenharmony_ci int idx = (int)(long)pde_data(m->file->f_inode); 30962306a36Sopenharmony_ci struct net_device *dev; 31062306a36Sopenharmony_ci struct can_dev_rcv_lists *dev_rcv_lists; 31162306a36Sopenharmony_ci struct net *net = m->private; 31262306a36Sopenharmony_ci 31362306a36Sopenharmony_ci seq_printf(m, "\nreceive list '%s':\n", rx_list_name[idx]); 31462306a36Sopenharmony_ci 31562306a36Sopenharmony_ci rcu_read_lock(); 31662306a36Sopenharmony_ci 31762306a36Sopenharmony_ci /* receive list for 'all' CAN devices (dev == NULL) */ 31862306a36Sopenharmony_ci dev_rcv_lists = net->can.rx_alldev_list; 31962306a36Sopenharmony_ci can_rcvlist_proc_show_one(m, idx, NULL, dev_rcv_lists); 32062306a36Sopenharmony_ci 32162306a36Sopenharmony_ci /* receive list for registered CAN devices */ 32262306a36Sopenharmony_ci for_each_netdev_rcu(net, dev) { 32362306a36Sopenharmony_ci struct can_ml_priv *can_ml = can_get_ml_priv(dev); 32462306a36Sopenharmony_ci 32562306a36Sopenharmony_ci if (can_ml) 32662306a36Sopenharmony_ci can_rcvlist_proc_show_one(m, idx, dev, 32762306a36Sopenharmony_ci &can_ml->dev_rcv_lists); 32862306a36Sopenharmony_ci } 32962306a36Sopenharmony_ci 33062306a36Sopenharmony_ci rcu_read_unlock(); 33162306a36Sopenharmony_ci 33262306a36Sopenharmony_ci seq_putc(m, '\n'); 33362306a36Sopenharmony_ci return 0; 33462306a36Sopenharmony_ci} 33562306a36Sopenharmony_ci 33662306a36Sopenharmony_cistatic inline void can_rcvlist_proc_show_array(struct seq_file *m, 33762306a36Sopenharmony_ci struct net_device *dev, 33862306a36Sopenharmony_ci struct hlist_head *rcv_array, 33962306a36Sopenharmony_ci unsigned int rcv_array_sz) 34062306a36Sopenharmony_ci{ 34162306a36Sopenharmony_ci unsigned int i; 34262306a36Sopenharmony_ci int all_empty = 1; 34362306a36Sopenharmony_ci 34462306a36Sopenharmony_ci /* check whether at least one list is non-empty */ 34562306a36Sopenharmony_ci for (i = 0; i < rcv_array_sz; i++) 34662306a36Sopenharmony_ci if (!hlist_empty(&rcv_array[i])) { 34762306a36Sopenharmony_ci all_empty = 0; 34862306a36Sopenharmony_ci break; 34962306a36Sopenharmony_ci } 35062306a36Sopenharmony_ci 35162306a36Sopenharmony_ci if (!all_empty) { 35262306a36Sopenharmony_ci can_print_recv_banner(m); 35362306a36Sopenharmony_ci for (i = 0; i < rcv_array_sz; i++) { 35462306a36Sopenharmony_ci if (!hlist_empty(&rcv_array[i])) 35562306a36Sopenharmony_ci can_print_rcvlist(m, &rcv_array[i], dev); 35662306a36Sopenharmony_ci } 35762306a36Sopenharmony_ci } else 35862306a36Sopenharmony_ci seq_printf(m, " (%s: no entry)\n", DNAME(dev)); 35962306a36Sopenharmony_ci} 36062306a36Sopenharmony_ci 36162306a36Sopenharmony_cistatic int can_rcvlist_sff_proc_show(struct seq_file *m, void *v) 36262306a36Sopenharmony_ci{ 36362306a36Sopenharmony_ci struct net_device *dev; 36462306a36Sopenharmony_ci struct can_dev_rcv_lists *dev_rcv_lists; 36562306a36Sopenharmony_ci struct net *net = m->private; 36662306a36Sopenharmony_ci 36762306a36Sopenharmony_ci /* RX_SFF */ 36862306a36Sopenharmony_ci seq_puts(m, "\nreceive list 'rx_sff':\n"); 36962306a36Sopenharmony_ci 37062306a36Sopenharmony_ci rcu_read_lock(); 37162306a36Sopenharmony_ci 37262306a36Sopenharmony_ci /* sff receive list for 'all' CAN devices (dev == NULL) */ 37362306a36Sopenharmony_ci dev_rcv_lists = net->can.rx_alldev_list; 37462306a36Sopenharmony_ci can_rcvlist_proc_show_array(m, NULL, dev_rcv_lists->rx_sff, 37562306a36Sopenharmony_ci ARRAY_SIZE(dev_rcv_lists->rx_sff)); 37662306a36Sopenharmony_ci 37762306a36Sopenharmony_ci /* sff receive list for registered CAN devices */ 37862306a36Sopenharmony_ci for_each_netdev_rcu(net, dev) { 37962306a36Sopenharmony_ci struct can_ml_priv *can_ml = can_get_ml_priv(dev); 38062306a36Sopenharmony_ci 38162306a36Sopenharmony_ci if (can_ml) { 38262306a36Sopenharmony_ci dev_rcv_lists = &can_ml->dev_rcv_lists; 38362306a36Sopenharmony_ci can_rcvlist_proc_show_array(m, dev, dev_rcv_lists->rx_sff, 38462306a36Sopenharmony_ci ARRAY_SIZE(dev_rcv_lists->rx_sff)); 38562306a36Sopenharmony_ci } 38662306a36Sopenharmony_ci } 38762306a36Sopenharmony_ci 38862306a36Sopenharmony_ci rcu_read_unlock(); 38962306a36Sopenharmony_ci 39062306a36Sopenharmony_ci seq_putc(m, '\n'); 39162306a36Sopenharmony_ci return 0; 39262306a36Sopenharmony_ci} 39362306a36Sopenharmony_ci 39462306a36Sopenharmony_cistatic int can_rcvlist_eff_proc_show(struct seq_file *m, void *v) 39562306a36Sopenharmony_ci{ 39662306a36Sopenharmony_ci struct net_device *dev; 39762306a36Sopenharmony_ci struct can_dev_rcv_lists *dev_rcv_lists; 39862306a36Sopenharmony_ci struct net *net = m->private; 39962306a36Sopenharmony_ci 40062306a36Sopenharmony_ci /* RX_EFF */ 40162306a36Sopenharmony_ci seq_puts(m, "\nreceive list 'rx_eff':\n"); 40262306a36Sopenharmony_ci 40362306a36Sopenharmony_ci rcu_read_lock(); 40462306a36Sopenharmony_ci 40562306a36Sopenharmony_ci /* eff receive list for 'all' CAN devices (dev == NULL) */ 40662306a36Sopenharmony_ci dev_rcv_lists = net->can.rx_alldev_list; 40762306a36Sopenharmony_ci can_rcvlist_proc_show_array(m, NULL, dev_rcv_lists->rx_eff, 40862306a36Sopenharmony_ci ARRAY_SIZE(dev_rcv_lists->rx_eff)); 40962306a36Sopenharmony_ci 41062306a36Sopenharmony_ci /* eff receive list for registered CAN devices */ 41162306a36Sopenharmony_ci for_each_netdev_rcu(net, dev) { 41262306a36Sopenharmony_ci struct can_ml_priv *can_ml = can_get_ml_priv(dev); 41362306a36Sopenharmony_ci 41462306a36Sopenharmony_ci if (can_ml) { 41562306a36Sopenharmony_ci dev_rcv_lists = &can_ml->dev_rcv_lists; 41662306a36Sopenharmony_ci can_rcvlist_proc_show_array(m, dev, dev_rcv_lists->rx_eff, 41762306a36Sopenharmony_ci ARRAY_SIZE(dev_rcv_lists->rx_eff)); 41862306a36Sopenharmony_ci } 41962306a36Sopenharmony_ci } 42062306a36Sopenharmony_ci 42162306a36Sopenharmony_ci rcu_read_unlock(); 42262306a36Sopenharmony_ci 42362306a36Sopenharmony_ci seq_putc(m, '\n'); 42462306a36Sopenharmony_ci return 0; 42562306a36Sopenharmony_ci} 42662306a36Sopenharmony_ci 42762306a36Sopenharmony_ci/* 42862306a36Sopenharmony_ci * can_init_proc - create main CAN proc directory and procfs entries 42962306a36Sopenharmony_ci */ 43062306a36Sopenharmony_civoid can_init_proc(struct net *net) 43162306a36Sopenharmony_ci{ 43262306a36Sopenharmony_ci /* create /proc/net/can directory */ 43362306a36Sopenharmony_ci net->can.proc_dir = proc_net_mkdir(net, "can", net->proc_net); 43462306a36Sopenharmony_ci 43562306a36Sopenharmony_ci if (!net->can.proc_dir) { 43662306a36Sopenharmony_ci printk(KERN_INFO "can: failed to create /proc/net/can . " 43762306a36Sopenharmony_ci "CONFIG_PROC_FS missing?\n"); 43862306a36Sopenharmony_ci return; 43962306a36Sopenharmony_ci } 44062306a36Sopenharmony_ci 44162306a36Sopenharmony_ci /* own procfs entries from the AF_CAN core */ 44262306a36Sopenharmony_ci net->can.pde_stats = proc_create_net_single(CAN_PROC_STATS, 0644, 44362306a36Sopenharmony_ci net->can.proc_dir, can_stats_proc_show, NULL); 44462306a36Sopenharmony_ci net->can.pde_reset_stats = proc_create_net_single(CAN_PROC_RESET_STATS, 44562306a36Sopenharmony_ci 0644, net->can.proc_dir, can_reset_stats_proc_show, 44662306a36Sopenharmony_ci NULL); 44762306a36Sopenharmony_ci net->can.pde_rcvlist_err = proc_create_net_single(CAN_PROC_RCVLIST_ERR, 44862306a36Sopenharmony_ci 0644, net->can.proc_dir, can_rcvlist_proc_show, 44962306a36Sopenharmony_ci (void *)RX_ERR); 45062306a36Sopenharmony_ci net->can.pde_rcvlist_all = proc_create_net_single(CAN_PROC_RCVLIST_ALL, 45162306a36Sopenharmony_ci 0644, net->can.proc_dir, can_rcvlist_proc_show, 45262306a36Sopenharmony_ci (void *)RX_ALL); 45362306a36Sopenharmony_ci net->can.pde_rcvlist_fil = proc_create_net_single(CAN_PROC_RCVLIST_FIL, 45462306a36Sopenharmony_ci 0644, net->can.proc_dir, can_rcvlist_proc_show, 45562306a36Sopenharmony_ci (void *)RX_FIL); 45662306a36Sopenharmony_ci net->can.pde_rcvlist_inv = proc_create_net_single(CAN_PROC_RCVLIST_INV, 45762306a36Sopenharmony_ci 0644, net->can.proc_dir, can_rcvlist_proc_show, 45862306a36Sopenharmony_ci (void *)RX_INV); 45962306a36Sopenharmony_ci net->can.pde_rcvlist_eff = proc_create_net_single(CAN_PROC_RCVLIST_EFF, 46062306a36Sopenharmony_ci 0644, net->can.proc_dir, can_rcvlist_eff_proc_show, NULL); 46162306a36Sopenharmony_ci net->can.pde_rcvlist_sff = proc_create_net_single(CAN_PROC_RCVLIST_SFF, 46262306a36Sopenharmony_ci 0644, net->can.proc_dir, can_rcvlist_sff_proc_show, NULL); 46362306a36Sopenharmony_ci} 46462306a36Sopenharmony_ci 46562306a36Sopenharmony_ci/* 46662306a36Sopenharmony_ci * can_remove_proc - remove procfs entries and main CAN proc directory 46762306a36Sopenharmony_ci */ 46862306a36Sopenharmony_civoid can_remove_proc(struct net *net) 46962306a36Sopenharmony_ci{ 47062306a36Sopenharmony_ci if (!net->can.proc_dir) 47162306a36Sopenharmony_ci return; 47262306a36Sopenharmony_ci 47362306a36Sopenharmony_ci if (net->can.pde_stats) 47462306a36Sopenharmony_ci remove_proc_entry(CAN_PROC_STATS, net->can.proc_dir); 47562306a36Sopenharmony_ci 47662306a36Sopenharmony_ci if (net->can.pde_reset_stats) 47762306a36Sopenharmony_ci remove_proc_entry(CAN_PROC_RESET_STATS, net->can.proc_dir); 47862306a36Sopenharmony_ci 47962306a36Sopenharmony_ci if (net->can.pde_rcvlist_err) 48062306a36Sopenharmony_ci remove_proc_entry(CAN_PROC_RCVLIST_ERR, net->can.proc_dir); 48162306a36Sopenharmony_ci 48262306a36Sopenharmony_ci if (net->can.pde_rcvlist_all) 48362306a36Sopenharmony_ci remove_proc_entry(CAN_PROC_RCVLIST_ALL, net->can.proc_dir); 48462306a36Sopenharmony_ci 48562306a36Sopenharmony_ci if (net->can.pde_rcvlist_fil) 48662306a36Sopenharmony_ci remove_proc_entry(CAN_PROC_RCVLIST_FIL, net->can.proc_dir); 48762306a36Sopenharmony_ci 48862306a36Sopenharmony_ci if (net->can.pde_rcvlist_inv) 48962306a36Sopenharmony_ci remove_proc_entry(CAN_PROC_RCVLIST_INV, net->can.proc_dir); 49062306a36Sopenharmony_ci 49162306a36Sopenharmony_ci if (net->can.pde_rcvlist_eff) 49262306a36Sopenharmony_ci remove_proc_entry(CAN_PROC_RCVLIST_EFF, net->can.proc_dir); 49362306a36Sopenharmony_ci 49462306a36Sopenharmony_ci if (net->can.pde_rcvlist_sff) 49562306a36Sopenharmony_ci remove_proc_entry(CAN_PROC_RCVLIST_SFF, net->can.proc_dir); 49662306a36Sopenharmony_ci 49762306a36Sopenharmony_ci remove_proc_entry("can", net->proc_net); 49862306a36Sopenharmony_ci} 499