stats.c 8.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348
  1. // SPDX-License-Identifier: GPL-2.0-only
  2. /*
  3. * linux/net/sunrpc/stats.c
  4. *
  5. * procfs-based user access to generic RPC statistics. The stats files
  6. * reside in /proc/net/rpc.
  7. *
  8. * The read routines assume that the buffer passed in is just big enough.
  9. * If you implement an RPC service that has its own stats routine which
  10. * appends the generic RPC stats, make sure you don't exceed the PAGE_SIZE
  11. * limit.
  12. *
  13. * Copyright (C) 1995, 1996, 1997 Olaf Kirch <okir@monad.swb.de>
  14. */
  15. #include <linux/module.h>
  16. #include <linux/slab.h>
  17. #include <linux/init.h>
  18. #include <linux/kernel.h>
  19. #include <linux/proc_fs.h>
  20. #include <linux/seq_file.h>
  21. #include <linux/sunrpc/clnt.h>
  22. #include <linux/sunrpc/svcsock.h>
  23. #include <linux/sunrpc/metrics.h>
  24. #include <linux/rcupdate.h>
  25. #include <trace/events/sunrpc.h>
  26. #include "netns.h"
  27. #define RPCDBG_FACILITY RPCDBG_MISC
  28. /*
  29. * Get RPC client stats
  30. */
  31. static int rpc_proc_show(struct seq_file *seq, void *v) {
  32. const struct rpc_stat *statp = seq->private;
  33. const struct rpc_program *prog = statp->program;
  34. unsigned int i, j;
  35. seq_printf(seq,
  36. "net %u %u %u %u\n",
  37. statp->netcnt,
  38. statp->netudpcnt,
  39. statp->nettcpcnt,
  40. statp->nettcpconn);
  41. seq_printf(seq,
  42. "rpc %u %u %u\n",
  43. statp->rpccnt,
  44. statp->rpcretrans,
  45. statp->rpcauthrefresh);
  46. for (i = 0; i < prog->nrvers; i++) {
  47. const struct rpc_version *vers = prog->version[i];
  48. if (!vers)
  49. continue;
  50. seq_printf(seq, "proc%u %u",
  51. vers->number, vers->nrprocs);
  52. for (j = 0; j < vers->nrprocs; j++)
  53. seq_printf(seq, " %u", vers->counts[j]);
  54. seq_putc(seq, '\n');
  55. }
  56. return 0;
  57. }
  58. static int rpc_proc_open(struct inode *inode, struct file *file)
  59. {
  60. return single_open(file, rpc_proc_show, pde_data(inode));
  61. }
  62. static const struct proc_ops rpc_proc_ops = {
  63. .proc_open = rpc_proc_open,
  64. .proc_read = seq_read,
  65. .proc_lseek = seq_lseek,
  66. .proc_release = single_release,
  67. };
  68. /*
  69. * Get RPC server stats
  70. */
  71. void svc_seq_show(struct seq_file *seq, const struct svc_stat *statp)
  72. {
  73. const struct svc_program *prog = statp->program;
  74. const struct svc_version *vers;
  75. unsigned int i, j, k;
  76. unsigned long count;
  77. seq_printf(seq,
  78. "net %u %u %u %u\n",
  79. statp->netcnt,
  80. statp->netudpcnt,
  81. statp->nettcpcnt,
  82. statp->nettcpconn);
  83. seq_printf(seq,
  84. "rpc %u %u %u %u %u\n",
  85. statp->rpccnt,
  86. statp->rpcbadfmt+statp->rpcbadauth+statp->rpcbadclnt,
  87. statp->rpcbadfmt,
  88. statp->rpcbadauth,
  89. statp->rpcbadclnt);
  90. for (i = 0; i < prog->pg_nvers; i++) {
  91. vers = prog->pg_vers[i];
  92. if (!vers)
  93. continue;
  94. seq_printf(seq, "proc%d %u", i, vers->vs_nproc);
  95. for (j = 0; j < vers->vs_nproc; j++) {
  96. count = 0;
  97. for_each_possible_cpu(k)
  98. count += per_cpu(vers->vs_count[j], k);
  99. seq_printf(seq, " %lu", count);
  100. }
  101. seq_putc(seq, '\n');
  102. }
  103. }
  104. EXPORT_SYMBOL_GPL(svc_seq_show);
  105. /**
  106. * rpc_alloc_iostats - allocate an rpc_iostats structure
  107. * @clnt: RPC program, version, and xprt
  108. *
  109. */
  110. struct rpc_iostats *rpc_alloc_iostats(struct rpc_clnt *clnt)
  111. {
  112. struct rpc_iostats *stats;
  113. int i;
  114. stats = kzalloc_objs(*stats, clnt->cl_maxproc);
  115. if (stats) {
  116. for (i = 0; i < clnt->cl_maxproc; i++)
  117. spin_lock_init(&stats[i].om_lock);
  118. }
  119. return stats;
  120. }
  121. EXPORT_SYMBOL_GPL(rpc_alloc_iostats);
  122. /**
  123. * rpc_free_iostats - release an rpc_iostats structure
  124. * @stats: doomed rpc_iostats structure
  125. *
  126. */
  127. void rpc_free_iostats(struct rpc_iostats *stats)
  128. {
  129. kfree(stats);
  130. }
  131. EXPORT_SYMBOL_GPL(rpc_free_iostats);
  132. /**
  133. * rpc_count_iostats_metrics - tally up per-task stats
  134. * @task: completed rpc_task
  135. * @op_metrics: stat structure for OP that will accumulate stats from @task
  136. */
  137. void rpc_count_iostats_metrics(const struct rpc_task *task,
  138. struct rpc_iostats *op_metrics)
  139. {
  140. struct rpc_rqst *req = task->tk_rqstp;
  141. ktime_t backlog, execute, now;
  142. if (!op_metrics || !req)
  143. return;
  144. now = ktime_get();
  145. spin_lock(&op_metrics->om_lock);
  146. op_metrics->om_ops++;
  147. /* kernel API: om_ops must never become larger than om_ntrans */
  148. op_metrics->om_ntrans += max(req->rq_ntrans, 1);
  149. op_metrics->om_timeouts += task->tk_timeouts;
  150. op_metrics->om_bytes_sent += req->rq_xmit_bytes_sent;
  151. op_metrics->om_bytes_recv += req->rq_reply_bytes_recvd;
  152. backlog = 0;
  153. if (ktime_to_ns(req->rq_xtime)) {
  154. backlog = ktime_sub(req->rq_xtime, task->tk_start);
  155. op_metrics->om_queue = ktime_add(op_metrics->om_queue, backlog);
  156. }
  157. op_metrics->om_rtt = ktime_add(op_metrics->om_rtt, req->rq_rtt);
  158. execute = ktime_sub(now, task->tk_start);
  159. op_metrics->om_execute = ktime_add(op_metrics->om_execute, execute);
  160. if (task->tk_status < 0)
  161. op_metrics->om_error_status++;
  162. spin_unlock(&op_metrics->om_lock);
  163. trace_rpc_stats_latency(req->rq_task, backlog, req->rq_rtt, execute);
  164. }
  165. EXPORT_SYMBOL_GPL(rpc_count_iostats_metrics);
  166. /**
  167. * rpc_count_iostats - tally up per-task stats
  168. * @task: completed rpc_task
  169. * @stats: array of stat structures
  170. *
  171. * Uses the statidx from @task
  172. */
  173. void rpc_count_iostats(const struct rpc_task *task, struct rpc_iostats *stats)
  174. {
  175. rpc_count_iostats_metrics(task,
  176. &stats[task->tk_msg.rpc_proc->p_statidx]);
  177. }
  178. EXPORT_SYMBOL_GPL(rpc_count_iostats);
  179. static void _print_name(struct seq_file *seq, unsigned int op,
  180. const struct rpc_procinfo *procs)
  181. {
  182. if (procs[op].p_name)
  183. seq_printf(seq, "\t%12s: ", procs[op].p_name);
  184. else if (op == 0)
  185. seq_printf(seq, "\t NULL: ");
  186. else
  187. seq_printf(seq, "\t%12u: ", op);
  188. }
  189. static void _add_rpc_iostats(struct rpc_iostats *a, struct rpc_iostats *b)
  190. {
  191. a->om_ops += b->om_ops;
  192. a->om_ntrans += b->om_ntrans;
  193. a->om_timeouts += b->om_timeouts;
  194. a->om_bytes_sent += b->om_bytes_sent;
  195. a->om_bytes_recv += b->om_bytes_recv;
  196. a->om_queue = ktime_add(a->om_queue, b->om_queue);
  197. a->om_rtt = ktime_add(a->om_rtt, b->om_rtt);
  198. a->om_execute = ktime_add(a->om_execute, b->om_execute);
  199. a->om_error_status += b->om_error_status;
  200. }
  201. static void _print_rpc_iostats(struct seq_file *seq, struct rpc_iostats *stats,
  202. int op, const struct rpc_procinfo *procs)
  203. {
  204. _print_name(seq, op, procs);
  205. seq_printf(seq, "%lu %lu %lu %llu %llu %llu %llu %llu %lu\n",
  206. stats->om_ops,
  207. stats->om_ntrans,
  208. stats->om_timeouts,
  209. stats->om_bytes_sent,
  210. stats->om_bytes_recv,
  211. ktime_to_ms(stats->om_queue),
  212. ktime_to_ms(stats->om_rtt),
  213. ktime_to_ms(stats->om_execute),
  214. stats->om_error_status);
  215. }
  216. static int do_print_stats(struct rpc_clnt *clnt, struct rpc_xprt *xprt, void *seqv)
  217. {
  218. struct seq_file *seq = seqv;
  219. xprt->ops->print_stats(xprt, seq);
  220. return 0;
  221. }
  222. void rpc_clnt_show_stats(struct seq_file *seq, struct rpc_clnt *clnt)
  223. {
  224. unsigned int op, maxproc = clnt->cl_maxproc;
  225. if (!clnt->cl_metrics)
  226. return;
  227. seq_printf(seq, "\tRPC iostats version: %s ", RPC_IOSTATS_VERS);
  228. seq_printf(seq, "p/v: %u/%u (%s)\n",
  229. clnt->cl_prog, clnt->cl_vers, clnt->cl_program->name);
  230. rpc_clnt_iterate_for_each_xprt(clnt, do_print_stats, seq);
  231. seq_printf(seq, "\tper-op statistics\n");
  232. for (op = 0; op < maxproc; op++) {
  233. struct rpc_iostats stats = {};
  234. struct rpc_clnt *next = clnt;
  235. do {
  236. _add_rpc_iostats(&stats, &next->cl_metrics[op]);
  237. if (next == next->cl_parent)
  238. break;
  239. next = next->cl_parent;
  240. } while (next);
  241. _print_rpc_iostats(seq, &stats, op, clnt->cl_procinfo);
  242. }
  243. }
  244. EXPORT_SYMBOL_GPL(rpc_clnt_show_stats);
  245. /*
  246. * Register/unregister RPC proc files
  247. */
  248. static inline struct proc_dir_entry *
  249. do_register(struct net *net, const char *name, void *data,
  250. const struct proc_ops *proc_ops)
  251. {
  252. struct sunrpc_net *sn;
  253. dprintk("RPC: registering /proc/net/rpc/%s\n", name);
  254. sn = net_generic(net, sunrpc_net_id);
  255. return proc_create_data(name, 0, sn->proc_net_rpc, proc_ops, data);
  256. }
  257. struct proc_dir_entry *
  258. rpc_proc_register(struct net *net, struct rpc_stat *statp)
  259. {
  260. return do_register(net, statp->program->name, statp, &rpc_proc_ops);
  261. }
  262. EXPORT_SYMBOL_GPL(rpc_proc_register);
  263. void
  264. rpc_proc_unregister(struct net *net, const char *name)
  265. {
  266. struct sunrpc_net *sn;
  267. sn = net_generic(net, sunrpc_net_id);
  268. remove_proc_entry(name, sn->proc_net_rpc);
  269. }
  270. EXPORT_SYMBOL_GPL(rpc_proc_unregister);
  271. struct proc_dir_entry *
  272. svc_proc_register(struct net *net, struct svc_stat *statp, const struct proc_ops *proc_ops)
  273. {
  274. return do_register(net, statp->program->pg_name, net, proc_ops);
  275. }
  276. EXPORT_SYMBOL_GPL(svc_proc_register);
  277. void
  278. svc_proc_unregister(struct net *net, const char *name)
  279. {
  280. struct sunrpc_net *sn;
  281. sn = net_generic(net, sunrpc_net_id);
  282. remove_proc_entry(name, sn->proc_net_rpc);
  283. }
  284. EXPORT_SYMBOL_GPL(svc_proc_unregister);
  285. int rpc_proc_init(struct net *net)
  286. {
  287. struct sunrpc_net *sn;
  288. dprintk("RPC: registering /proc/net/rpc\n");
  289. sn = net_generic(net, sunrpc_net_id);
  290. sn->proc_net_rpc = proc_mkdir("rpc", net->proc_net);
  291. if (sn->proc_net_rpc == NULL)
  292. return -ENOMEM;
  293. return 0;
  294. }
  295. void rpc_proc_exit(struct net *net)
  296. {
  297. dprintk("RPC: unregistering /proc/net/rpc\n");
  298. remove_proc_entry("rpc", net->proc_net);
  299. }