📄 stats.c
字号:
/* * linux/net/sunrpc/stats.c * * procfs-based user access to generic RPC statistics. The stats files * reside in /proc/net/rpc. * * The read routines assume that the buffer passed in is just big enough. * If you implement an RPC service that has its own stats routine which * appends the generic RPC stats, make sure you don't exceed the PAGE_SIZE * limit. * * Copyright (C) 1995, 1996, 1997 Olaf Kirch <okir@monad.swb.de> */#define __NO_VERSION__#include <linux/module.h>#include <linux/kernel.h>#include <linux/sched.h>#include <linux/proc_fs.h>#include <linux/sunrpc/clnt.h>#include <linux/sunrpc/svcsock.h>#define RPCDBG_FACILITY RPCDBG_MISCstatic struct proc_dir_entry *proc_net_rpc = NULL;/* * Get RPC client stats */intrpc_proc_read(char *buffer, char **start, off_t offset, int count, int *eof, void *data){ struct rpc_stat *statp = (struct rpc_stat *) data; struct rpc_program *prog = statp->program; struct rpc_version *vers; int len, i, j; len = sprintf(buffer, "net %d %d %d %d\n", statp->netcnt, statp->netudpcnt, statp->nettcpcnt, statp->nettcpconn); len += sprintf(buffer + len, "rpc %d %d %d\n", statp->rpccnt, statp->rpcretrans, statp->rpcauthrefresh); for (i = 0; i < prog->nrvers; i++) { if (!(vers = prog->version[i])) continue; len += sprintf(buffer + len, "proc%d %d", vers->number, vers->nrprocs); for (j = 0; j < vers->nrprocs; j++) len += sprintf(buffer + len, " %d", vers->procs[j].p_count); buffer[len++] = '\n'; } if (offset >= len) { *start = buffer; *eof = 1; return 0; } *start = buffer + offset; if ((len -= offset) > count) return count; *eof = 1; return len;}/* * Get RPC server stats */intsvc_proc_read(char *buffer, char **start, off_t offset, int count, int *eof, void *data){ struct svc_stat *statp = (struct svc_stat *) data; struct svc_program *prog = statp->program; struct svc_procedure *proc; struct svc_version *vers; int len, i, j; len = sprintf(buffer, "net %d %d %d %d\n", statp->netcnt, statp->netudpcnt, statp->nettcpcnt, statp->nettcpconn); len += sprintf(buffer + len, "rpc %d %d %d %d %d\n", statp->rpccnt, statp->rpcbadfmt+statp->rpcbadauth+statp->rpcbadclnt, statp->rpcbadfmt, statp->rpcbadauth, statp->rpcbadclnt); for (i = 0; i < prog->pg_nvers; i++) { if (!(vers = prog->pg_vers[i]) || !(proc = vers->vs_proc)) continue; len += sprintf(buffer + len, "proc%d %d", i, vers->vs_nproc); for (j = 0; j < vers->vs_nproc; j++, proc++) len += sprintf(buffer + len, " %d", proc->pc_count); buffer[len++] = '\n'; } if (offset >= len) { *start = buffer; *eof = 1; return 0; } *start = buffer + offset; if ((len -= offset) > count) return count; *eof = 1; return len;}/* * Register/unregister RPC proc files */static inline struct proc_dir_entry *do_register(const char *name, void *data, int issvc){ rpc_proc_init(); dprintk("RPC: registering /proc/net/rpc/%s\n", name); return create_proc_read_entry(name, 0, proc_net_rpc, issvc? svc_proc_read : rpc_proc_read, data);}struct proc_dir_entry *rpc_proc_register(struct rpc_stat *statp){ return do_register(statp->program->name, statp, 0);}voidrpc_proc_unregister(const char *name){ remove_proc_entry(name, proc_net_rpc);}struct proc_dir_entry *svc_proc_register(struct svc_stat *statp){ return do_register(statp->program->pg_name, statp, 1);}voidsvc_proc_unregister(const char *name){ remove_proc_entry(name, proc_net_rpc);}voidrpc_proc_init(void){ dprintk("RPC: registering /proc/net/rpc\n"); if (!proc_net_rpc) { struct proc_dir_entry *ent; ent = proc_mkdir("net/rpc", 0); if (ent) { ent->owner = THIS_MODULE; proc_net_rpc = ent; } }}voidrpc_proc_exit(void){ dprintk("RPC: unregistering /proc/net/rpc\n"); if (proc_net_rpc) { proc_net_rpc = NULL; remove_proc_entry("net/rpc", 0); }}#ifdef MODULEintinit_module(void){#ifdef RPC_DEBUG rpc_register_sysctl();#endif rpc_proc_init(); return 0;}voidcleanup_module(void){#ifdef RPC_DEBUG rpc_unregister_sysctl();#endif rpc_proc_exit();}#endif
⌨️ 快捷键说明
复制代码
Ctrl + C
搜索代码
Ctrl + F
全屏模式
F11
切换主题
Ctrl + Shift + D
显示快捷键
?
增大字号
Ctrl + =
减小字号
Ctrl + -