diff options
author | Eric Dumazet <eric.dumazet@gmail.com> | 2009-11-12 02:44:25 -0500 |
---|---|---|
committer | David S. Miller <davem@davemloft.net> | 2009-11-13 23:46:55 -0500 |
commit | eec4df9885f7822cdeca82577a25cac4598fa7cf (patch) | |
tree | 36295f78f105e3f69b571edb3b2feddaf99dfb73 /net/ipv4 | |
parent | 342bde1b70c79bfc8509b017b3987f3c7541ff8e (diff) |
ipv4: speedup inet_dump_ifaddr()
Stephen Hemminger a écrit :
> On Thu, 12 Nov 2009 15:11:36 +0100
> Eric Dumazet <eric.dumazet@gmail.com> wrote:
>
>> When handling large number of netdevices, inet_dump_ifaddr()
>> is very slow because it has O(N^2) complexity.
>>
>> Instead of scanning one single list, we can use the NETDEV_HASHENTRIES
>> sub lists of the dev_index hash table, and RCU lookups.
>>
>> Signed-off-by: Eric Dumazet <eric.dumazet@gmail.com>
>
> You might be able to make RCU critical section smaller by moving
> it into loop.
>
Indeed. But we dump at most one skb (<= 8192 bytes ?), so rcu_read_lock
holding time is small, unless we meet many netdevices without
addresses. I wonder if its really common...
Thanks
[PATCH net-next-2.6] ipv4: speedup inet_dump_ifaddr()
When handling large number of netdevices, inet_dump_ifaddr()
is very slow because it has O(N2) complexity.
Instead of scanning one single list, we can use the NETDEV_HASHENTRIES
sub lists of the dev_index hash table, and RCU lookups.
Signed-off-by: Eric Dumazet <eric.dumazet@gmail.com>
Acked-by: Stephen Hemminger <shemminger@vyatta.com>
Signed-off-by: David S. Miller <davem@davemloft.net>
Diffstat (limited to 'net/ipv4')
-rw-r--r-- | net/ipv4/devinet.c | 61 |
1 files changed, 38 insertions, 23 deletions
diff --git a/net/ipv4/devinet.c b/net/ipv4/devinet.c index c2045f9615da..7620382058a0 100644 --- a/net/ipv4/devinet.c +++ b/net/ipv4/devinet.c | |||
@@ -1174,39 +1174,54 @@ nla_put_failure: | |||
1174 | static int inet_dump_ifaddr(struct sk_buff *skb, struct netlink_callback *cb) | 1174 | static int inet_dump_ifaddr(struct sk_buff *skb, struct netlink_callback *cb) |
1175 | { | 1175 | { |
1176 | struct net *net = sock_net(skb->sk); | 1176 | struct net *net = sock_net(skb->sk); |
1177 | int idx, ip_idx; | 1177 | int h, s_h; |
1178 | int idx, s_idx; | ||
1179 | int ip_idx, s_ip_idx; | ||
1178 | struct net_device *dev; | 1180 | struct net_device *dev; |
1179 | struct in_device *in_dev; | 1181 | struct in_device *in_dev; |
1180 | struct in_ifaddr *ifa; | 1182 | struct in_ifaddr *ifa; |
1181 | int s_ip_idx, s_idx = cb->args[0]; | 1183 | struct hlist_head *head; |
1184 | struct hlist_node *node; | ||
1182 | 1185 | ||
1183 | s_ip_idx = ip_idx = cb->args[1]; | 1186 | s_h = cb->args[0]; |
1184 | idx = 0; | 1187 | s_idx = idx = cb->args[1]; |
1185 | for_each_netdev(net, dev) { | 1188 | s_ip_idx = ip_idx = cb->args[2]; |
1186 | if (idx < s_idx) | ||
1187 | goto cont; | ||
1188 | if (idx > s_idx) | ||
1189 | s_ip_idx = 0; | ||
1190 | in_dev = __in_dev_get_rtnl(dev); | ||
1191 | if (!in_dev) | ||
1192 | goto cont; | ||
1193 | 1189 | ||
1194 | for (ifa = in_dev->ifa_list, ip_idx = 0; ifa; | 1190 | for (h = s_h; h < NETDEV_HASHENTRIES; h++, s_idx = 0) { |
1195 | ifa = ifa->ifa_next, ip_idx++) { | 1191 | idx = 0; |
1196 | if (ip_idx < s_ip_idx) | 1192 | head = &net->dev_index_head[h]; |
1197 | continue; | 1193 | rcu_read_lock(); |
1198 | if (inet_fill_ifaddr(skb, ifa, NETLINK_CB(cb->skb).pid, | 1194 | hlist_for_each_entry_rcu(dev, node, head, index_hlist) { |
1195 | if (idx < s_idx) | ||
1196 | goto cont; | ||
1197 | if (idx > s_idx) | ||
1198 | s_ip_idx = 0; | ||
1199 | in_dev = __in_dev_get_rcu(dev); | ||
1200 | if (!in_dev) | ||
1201 | goto cont; | ||
1202 | |||
1203 | for (ifa = in_dev->ifa_list, ip_idx = 0; ifa; | ||
1204 | ifa = ifa->ifa_next, ip_idx++) { | ||
1205 | if (ip_idx < s_ip_idx) | ||
1206 | continue; | ||
1207 | if (inet_fill_ifaddr(skb, ifa, | ||
1208 | NETLINK_CB(cb->skb).pid, | ||
1199 | cb->nlh->nlmsg_seq, | 1209 | cb->nlh->nlmsg_seq, |
1200 | RTM_NEWADDR, NLM_F_MULTI) <= 0) | 1210 | RTM_NEWADDR, NLM_F_MULTI) <= 0) { |
1201 | goto done; | 1211 | rcu_read_unlock(); |
1202 | } | 1212 | goto done; |
1213 | } | ||
1214 | } | ||
1203 | cont: | 1215 | cont: |
1204 | idx++; | 1216 | idx++; |
1217 | } | ||
1218 | rcu_read_unlock(); | ||
1205 | } | 1219 | } |
1206 | 1220 | ||
1207 | done: | 1221 | done: |
1208 | cb->args[0] = idx; | 1222 | cb->args[0] = h; |
1209 | cb->args[1] = ip_idx; | 1223 | cb->args[1] = idx; |
1224 | cb->args[2] = ip_idx; | ||
1210 | 1225 | ||
1211 | return skb->len; | 1226 | return skb->len; |
1212 | } | 1227 | } |