diff options
author | Eric W. Biederman <ebiederm@xmission.com> | 2007-09-27 01:08:12 -0400 |
---|---|---|
committer | David S. Miller <davem@sunset.davemloft.net> | 2007-10-10 19:52:47 -0400 |
commit | 5f6d88b9149d537f3db0798f7d312be632422e15 (patch) | |
tree | 8e85e14c152a5b6c7f61ef2176cb6ed53854e280 /drivers | |
parent | 9dd776b6d7b0b85966b6ddd03e2b2aae59012ab1 (diff) |
[NET]: Dynamically allocate the per cpu counters for the loopback device.
This patch add support for dynamically allocating the statistics counters
for the loopback device and adds appropriate device methods for allocating
and freeing the loopback device.
This completes support for creating multiple instances of the loopback
device, in preparation for creating per network namespace instances.
Signed-off-by: Eric W. Biederman <ebiederm@xmission.com>
Signed-off-by: David S. Miller <davem@davemloft.net>
Diffstat (limited to 'drivers')
-rw-r--r-- | drivers/net/loopback.c | 32 |
1 files changed, 28 insertions, 4 deletions
diff --git a/drivers/net/loopback.c b/drivers/net/loopback.c index 4b6f7b2abea5..f3018bb7570d 100644 --- a/drivers/net/loopback.c +++ b/drivers/net/loopback.c | |||
@@ -62,7 +62,6 @@ struct pcpu_lstats { | |||
62 | unsigned long packets; | 62 | unsigned long packets; |
63 | unsigned long bytes; | 63 | unsigned long bytes; |
64 | }; | 64 | }; |
65 | static DEFINE_PER_CPU(struct pcpu_lstats, pcpu_lstats); | ||
66 | 65 | ||
67 | #define LOOPBACK_OVERHEAD (128 + MAX_HEADER + 16 + 16) | 66 | #define LOOPBACK_OVERHEAD (128 + MAX_HEADER + 16 + 16) |
68 | 67 | ||
@@ -134,7 +133,7 @@ static void emulate_large_send_offload(struct sk_buff *skb) | |||
134 | */ | 133 | */ |
135 | static int loopback_xmit(struct sk_buff *skb, struct net_device *dev) | 134 | static int loopback_xmit(struct sk_buff *skb, struct net_device *dev) |
136 | { | 135 | { |
137 | struct pcpu_lstats *lb_stats; | 136 | struct pcpu_lstats *pcpu_lstats, *lb_stats; |
138 | 137 | ||
139 | skb_orphan(skb); | 138 | skb_orphan(skb); |
140 | 139 | ||
@@ -155,7 +154,8 @@ static int loopback_xmit(struct sk_buff *skb, struct net_device *dev) | |||
155 | dev->last_rx = jiffies; | 154 | dev->last_rx = jiffies; |
156 | 155 | ||
157 | /* it's OK to use __get_cpu_var() because BHs are off */ | 156 | /* it's OK to use __get_cpu_var() because BHs are off */ |
158 | lb_stats = &__get_cpu_var(pcpu_lstats); | 157 | pcpu_lstats = netdev_priv(dev); |
158 | lb_stats = per_cpu_ptr(pcpu_lstats, smp_processor_id()); | ||
159 | lb_stats->bytes += skb->len; | 159 | lb_stats->bytes += skb->len; |
160 | lb_stats->packets++; | 160 | lb_stats->packets++; |
161 | 161 | ||
@@ -166,15 +166,17 @@ static int loopback_xmit(struct sk_buff *skb, struct net_device *dev) | |||
166 | 166 | ||
167 | static struct net_device_stats *get_stats(struct net_device *dev) | 167 | static struct net_device_stats *get_stats(struct net_device *dev) |
168 | { | 168 | { |
169 | const struct pcpu_lstats *pcpu_lstats; | ||
169 | struct net_device_stats *stats = &dev->stats; | 170 | struct net_device_stats *stats = &dev->stats; |
170 | unsigned long bytes = 0; | 171 | unsigned long bytes = 0; |
171 | unsigned long packets = 0; | 172 | unsigned long packets = 0; |
172 | int i; | 173 | int i; |
173 | 174 | ||
175 | pcpu_lstats = netdev_priv(dev); | ||
174 | for_each_possible_cpu(i) { | 176 | for_each_possible_cpu(i) { |
175 | const struct pcpu_lstats *lb_stats; | 177 | const struct pcpu_lstats *lb_stats; |
176 | 178 | ||
177 | lb_stats = &per_cpu(pcpu_lstats, i); | 179 | lb_stats = per_cpu_ptr(pcpu_lstats, i); |
178 | bytes += lb_stats->bytes; | 180 | bytes += lb_stats->bytes; |
179 | packets += lb_stats->packets; | 181 | packets += lb_stats->packets; |
180 | } | 182 | } |
@@ -198,6 +200,26 @@ static const struct ethtool_ops loopback_ethtool_ops = { | |||
198 | .get_rx_csum = always_on, | 200 | .get_rx_csum = always_on, |
199 | }; | 201 | }; |
200 | 202 | ||
203 | static int loopback_dev_init(struct net_device *dev) | ||
204 | { | ||
205 | struct pcpu_lstats *lstats; | ||
206 | |||
207 | lstats = alloc_percpu(struct pcpu_lstats); | ||
208 | if (!lstats) | ||
209 | return -ENOMEM; | ||
210 | |||
211 | dev->priv = lstats; | ||
212 | return 0; | ||
213 | } | ||
214 | |||
215 | static void loopback_dev_free(struct net_device *dev) | ||
216 | { | ||
217 | struct pcpu_lstats *lstats = netdev_priv(dev); | ||
218 | |||
219 | free_percpu(lstats); | ||
220 | free_netdev(dev); | ||
221 | } | ||
222 | |||
201 | /* | 223 | /* |
202 | * The loopback device is special. There is only one instance and | 224 | * The loopback device is special. There is only one instance and |
203 | * it is statically allocated. Don't do this for other devices. | 225 | * it is statically allocated. Don't do this for other devices. |
@@ -225,6 +247,8 @@ static void loopback_setup(struct net_device *dev) | |||
225 | | NETIF_F_LLTX | 247 | | NETIF_F_LLTX |
226 | | NETIF_F_NETNS_LOCAL, | 248 | | NETIF_F_NETNS_LOCAL, |
227 | dev->ethtool_ops = &loopback_ethtool_ops; | 249 | dev->ethtool_ops = &loopback_ethtool_ops; |
250 | dev->init = loopback_dev_init; | ||
251 | dev->destructor = loopback_dev_free; | ||
228 | } | 252 | } |
229 | 253 | ||
230 | /* Setup and register the loopback device. */ | 254 | /* Setup and register the loopback device. */ |