Blame view
net/core/gen_stats.c
6.71 KB
1da177e4c Linux-2.6.12-rc2 |
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 |
/* * net/core/gen_stats.c * * This program is free software; you can redistribute it and/or * modify it under the terms of the GNU General Public License * as published by the Free Software Foundation; either version * 2 of the License, or (at your option) any later version. * * Authors: Thomas Graf <tgraf@suug.ch> * Jamal Hadi Salim * Alexey Kuznetsov, <kuznet@ms2.inr.ac.ru> * * See Documentation/networking/gen_stats.txt */ #include <linux/types.h> #include <linux/kernel.h> #include <linux/module.h> #include <linux/interrupt.h> #include <linux/socket.h> #include <linux/rtnetlink.h> #include <linux/gen_stats.h> |
1e90474c3 [NET_SCHED]: Conv... |
23 |
#include <net/netlink.h> |
1da177e4c Linux-2.6.12-rc2 |
24 25 26 27 28 29 |
#include <net/gen_stats.h> static inline int gnet_stats_copy(struct gnet_dump *d, int type, void *buf, int size) { |
1e90474c3 [NET_SCHED]: Conv... |
30 |
NLA_PUT(d->skb, type, size, buf); |
1da177e4c Linux-2.6.12-rc2 |
31 |
return 0; |
1e90474c3 [NET_SCHED]: Conv... |
32 |
nla_put_failure: |
1da177e4c Linux-2.6.12-rc2 |
33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 |
spin_unlock_bh(d->lock); return -1; } /** * gnet_stats_start_copy_compat - start dumping procedure in compatibility mode * @skb: socket buffer to put statistics TLVs into * @type: TLV type for top level statistic TLV * @tc_stats_type: TLV type for backward compatibility struct tc_stats TLV * @xstats_type: TLV type for backward compatibility xstats TLV * @lock: statistics lock * @d: dumping handle * * Initializes the dumping handle, grabs the statistic lock and appends * an empty TLV header to the socket buffer for use a container for all * other statistic TLVS. * * The dumping handle is marked to be in backward compatibility mode telling * all gnet_stats_copy_XXX() functions to fill a local copy of struct tc_stats. * * Returns 0 on success or -1 if the room in the socket buffer was not sufficient. */ int gnet_stats_start_copy_compat(struct sk_buff *skb, int type, int tc_stats_type, int xstats_type, spinlock_t *lock, struct gnet_dump *d) |
9a429c498 [NET]: Add some a... |
58 |
__acquires(lock) |
1da177e4c Linux-2.6.12-rc2 |
59 60 |
{ memset(d, 0, sizeof(*d)); |
4ec93edb1 [NET] CORE: Fix w... |
61 |
|
1da177e4c Linux-2.6.12-rc2 |
62 63 64 |
spin_lock_bh(lock); d->lock = lock; if (type) |
1e90474c3 [NET_SCHED]: Conv... |
65 |
d->tail = (struct nlattr *)skb_tail_pointer(skb); |
1da177e4c Linux-2.6.12-rc2 |
66 67 68 69 70 71 72 73 74 |
d->skb = skb; d->compat_tc_stats = tc_stats_type; d->compat_xstats = xstats_type; if (d->tail) return gnet_stats_copy(d, type, NULL, 0); return 0; } |
9e34a5b51 net/core: EXPORT_... |
75 |
EXPORT_SYMBOL(gnet_stats_start_copy_compat); |
1da177e4c Linux-2.6.12-rc2 |
76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 |
/** * gnet_stats_start_copy_compat - start dumping procedure in compatibility mode * @skb: socket buffer to put statistics TLVs into * @type: TLV type for top level statistic TLV * @lock: statistics lock * @d: dumping handle * * Initializes the dumping handle, grabs the statistic lock and appends * an empty TLV header to the socket buffer for use a container for all * other statistic TLVS. * * Returns 0 on success or -1 if the room in the socket buffer was not sufficient. */ int gnet_stats_start_copy(struct sk_buff *skb, int type, spinlock_t *lock, struct gnet_dump *d) { return gnet_stats_start_copy_compat(skb, type, 0, 0, lock, d); } |
9e34a5b51 net/core: EXPORT_... |
96 |
EXPORT_SYMBOL(gnet_stats_start_copy); |
1da177e4c Linux-2.6.12-rc2 |
97 98 99 100 101 102 103 104 105 106 107 108 109 |
/** * gnet_stats_copy_basic - copy basic statistics into statistic TLV * @d: dumping handle * @b: basic statistics * * Appends the basic statistics to the top level TLV created by * gnet_stats_start_copy(). * * Returns 0 on success or -1 with the statistic lock released * if the room in the socket buffer was not sufficient. */ int |
c1a8f1f1c net: restore gnet... |
110 |
gnet_stats_copy_basic(struct gnet_dump *d, struct gnet_stats_basic_packed *b) |
1da177e4c Linux-2.6.12-rc2 |
111 112 113 114 115 |
{ if (d->compat_tc_stats) { d->tc_stats.bytes = b->bytes; d->tc_stats.packets = b->packets; } |
c1a8f1f1c net: restore gnet... |
116 117 |
if (d->tail) { struct gnet_stats_basic sb; |
1da177e4c Linux-2.6.12-rc2 |
118 |
|
c1a8f1f1c net: restore gnet... |
119 120 121 122 123 |
memset(&sb, 0, sizeof(sb)); sb.bytes = b->bytes; sb.packets = b->packets; return gnet_stats_copy(d, TCA_STATS_BASIC, &sb, sizeof(sb)); } |
1da177e4c Linux-2.6.12-rc2 |
124 125 |
return 0; } |
9e34a5b51 net/core: EXPORT_... |
126 |
EXPORT_SYMBOL(gnet_stats_copy_basic); |
1da177e4c Linux-2.6.12-rc2 |
127 128 129 130 |
/** * gnet_stats_copy_rate_est - copy rate estimator statistics into statistics TLV * @d: dumping handle |
d250a5f90 pkt_sched: gen_es... |
131 |
* @b: basic statistics |
1da177e4c Linux-2.6.12-rc2 |
132 133 134 135 136 137 138 139 140 |
* @r: rate estimator statistics * * Appends the rate estimator statistics to the top level TLV created by * gnet_stats_start_copy(). * * Returns 0 on success or -1 with the statistic lock released * if the room in the socket buffer was not sufficient. */ int |
d250a5f90 pkt_sched: gen_es... |
141 142 143 |
gnet_stats_copy_rate_est(struct gnet_dump *d, const struct gnet_stats_basic_packed *b, struct gnet_stats_rate_est *r) |
1da177e4c Linux-2.6.12-rc2 |
144 |
{ |
d250a5f90 pkt_sched: gen_es... |
145 146 |
if (b && !gen_estimator_active(b, r)) return 0; |
1da177e4c Linux-2.6.12-rc2 |
147 148 149 150 151 152 153 154 155 156 |
if (d->compat_tc_stats) { d->tc_stats.bps = r->bps; d->tc_stats.pps = r->pps; } if (d->tail) return gnet_stats_copy(d, TCA_STATS_RATE_EST, r, sizeof(*r)); return 0; } |
9e34a5b51 net/core: EXPORT_... |
157 |
EXPORT_SYMBOL(gnet_stats_copy_rate_est); |
1da177e4c Linux-2.6.12-rc2 |
158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 |
/** * gnet_stats_copy_queue - copy queue statistics into statistics TLV * @d: dumping handle * @q: queue statistics * * Appends the queue statistics to the top level TLV created by * gnet_stats_start_copy(). * * Returns 0 on success or -1 with the statistic lock released * if the room in the socket buffer was not sufficient. */ int gnet_stats_copy_queue(struct gnet_dump *d, struct gnet_stats_queue *q) { if (d->compat_tc_stats) { d->tc_stats.drops = q->drops; d->tc_stats.qlen = q->qlen; d->tc_stats.backlog = q->backlog; d->tc_stats.overlimits = q->overlimits; } if (d->tail) return gnet_stats_copy(d, TCA_STATS_QUEUE, q, sizeof(*q)); return 0; } |
9e34a5b51 net/core: EXPORT_... |
185 |
EXPORT_SYMBOL(gnet_stats_copy_queue); |
1da177e4c Linux-2.6.12-rc2 |
186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211 212 |
/** * gnet_stats_copy_app - copy application specific statistics into statistics TLV * @d: dumping handle * @st: application specific statistics data * @len: length of data * * Appends the application sepecific statistics to the top level TLV created by * gnet_stats_start_copy() and remembers the data for XSTATS if the dumping * handle is in backward compatibility mode. * * Returns 0 on success or -1 with the statistic lock released * if the room in the socket buffer was not sufficient. */ int gnet_stats_copy_app(struct gnet_dump *d, void *st, int len) { if (d->compat_xstats) { d->xstats = st; d->xstats_len = len; } if (d->tail) return gnet_stats_copy(d, TCA_STATS_APP, st, len); return 0; } |
9e34a5b51 net/core: EXPORT_... |
213 |
EXPORT_SYMBOL(gnet_stats_copy_app); |
1da177e4c Linux-2.6.12-rc2 |
214 215 216 217 218 219 220 221 222 223 224 225 226 227 228 229 230 |
/** * gnet_stats_finish_copy - finish dumping procedure * @d: dumping handle * * Corrects the length of the top level TLV to include all TLVs added * by gnet_stats_copy_XXX() calls. Adds the backward compatibility TLVs * if gnet_stats_start_copy_compat() was used and releases the statistics * lock. * * Returns 0 on success or -1 with the statistic lock released * if the room in the socket buffer was not sufficient. */ int gnet_stats_finish_copy(struct gnet_dump *d) { if (d->tail) |
1e90474c3 [NET_SCHED]: Conv... |
231 |
d->tail->nla_len = skb_tail_pointer(d->skb) - (u8 *)d->tail; |
1da177e4c Linux-2.6.12-rc2 |
232 233 234 235 236 237 238 239 240 241 242 243 244 245 246 |
if (d->compat_tc_stats) if (gnet_stats_copy(d, d->compat_tc_stats, &d->tc_stats, sizeof(d->tc_stats)) < 0) return -1; if (d->compat_xstats && d->xstats) { if (gnet_stats_copy(d, d->compat_xstats, d->xstats, d->xstats_len) < 0) return -1; } spin_unlock_bh(d->lock); return 0; } |
1da177e4c Linux-2.6.12-rc2 |
247 |
EXPORT_SYMBOL(gnet_stats_finish_copy); |