mirror of
https://github.com/Fishwaldo/linux-bl808.git
synced 2025-06-07 15:15:29 +00:00
[VLAN]: Reduce memory consumed by vlan_groups
Currently each vlan_groupd contains 8 pointers on arrays with 512 pointers on struct net_device each :) Such a construction "in many cases ... wastes memory". My proposal is to allow for some of these arrays pointers be NULL, meaning that there are no devices in it. When a new device is added to the vlan_group, the appropriate array is allocated. The check in vlan_group_get_device's is safe, since the pointer vg->vlan_devices_arrays[x] can only switch from NULL to not-NULL. The vlan_group_prealloc_vid() is guarded with rtnl lock and is also safe. I've checked (I hope that) all the places, that use these arrays and found, that the register_vlan_dev is the only place, that can put a vlan device on an empty vlan_group. Rough calculations shows, that after the patch a setup with a single vlan dev (or up to 512 vlans with sequential vids) will occupy approximately 8 times less memory. The question I have is - does this patch makes sense, or a totally new structures are required to store the vlan_devs? Signed-off-by: Pavel Emelyanov <xemul@openvz.org> Signed-off-by: Patrick McHardy <kaber@trash.net>
This commit is contained in:
parent
789e41e6f4
commit
67727184f2
2 changed files with 24 additions and 14 deletions
|
@ -93,7 +93,7 @@ static inline struct net_device *vlan_group_get_device(struct vlan_group *vg,
|
||||||
{
|
{
|
||||||
struct net_device **array;
|
struct net_device **array;
|
||||||
array = vg->vlan_devices_arrays[vlan_id / VLAN_GROUP_ARRAY_PART_LEN];
|
array = vg->vlan_devices_arrays[vlan_id / VLAN_GROUP_ARRAY_PART_LEN];
|
||||||
return array[vlan_id % VLAN_GROUP_ARRAY_PART_LEN];
|
return array ? array[vlan_id % VLAN_GROUP_ARRAY_PART_LEN] : NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
static inline void vlan_group_set_device(struct vlan_group *vg,
|
static inline void vlan_group_set_device(struct vlan_group *vg,
|
||||||
|
|
|
@ -106,29 +106,35 @@ static void vlan_group_free(struct vlan_group *grp)
|
||||||
static struct vlan_group *vlan_group_alloc(int ifindex)
|
static struct vlan_group *vlan_group_alloc(int ifindex)
|
||||||
{
|
{
|
||||||
struct vlan_group *grp;
|
struct vlan_group *grp;
|
||||||
unsigned int size;
|
|
||||||
unsigned int i;
|
|
||||||
|
|
||||||
grp = kzalloc(sizeof(struct vlan_group), GFP_KERNEL);
|
grp = kzalloc(sizeof(struct vlan_group), GFP_KERNEL);
|
||||||
if (!grp)
|
if (!grp)
|
||||||
return NULL;
|
return NULL;
|
||||||
|
|
||||||
size = sizeof(struct net_device *) * VLAN_GROUP_ARRAY_PART_LEN;
|
|
||||||
|
|
||||||
for (i = 0; i < VLAN_GROUP_ARRAY_SPLIT_PARTS; i++) {
|
|
||||||
grp->vlan_devices_arrays[i] = kzalloc(size, GFP_KERNEL);
|
|
||||||
if (!grp->vlan_devices_arrays[i])
|
|
||||||
goto err;
|
|
||||||
}
|
|
||||||
|
|
||||||
grp->real_dev_ifindex = ifindex;
|
grp->real_dev_ifindex = ifindex;
|
||||||
hlist_add_head_rcu(&grp->hlist,
|
hlist_add_head_rcu(&grp->hlist,
|
||||||
&vlan_group_hash[vlan_grp_hashfn(ifindex)]);
|
&vlan_group_hash[vlan_grp_hashfn(ifindex)]);
|
||||||
return grp;
|
return grp;
|
||||||
|
}
|
||||||
|
|
||||||
err:
|
static int vlan_group_prealloc_vid(struct vlan_group *vg, int vid)
|
||||||
vlan_group_free(grp);
|
{
|
||||||
return NULL;
|
struct net_device **array;
|
||||||
|
unsigned int size;
|
||||||
|
|
||||||
|
ASSERT_RTNL();
|
||||||
|
|
||||||
|
array = vg->vlan_devices_arrays[vid / VLAN_GROUP_ARRAY_PART_LEN];
|
||||||
|
if (array != NULL)
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
size = sizeof(struct net_device *) * VLAN_GROUP_ARRAY_PART_LEN;
|
||||||
|
array = kzalloc(size, GFP_KERNEL);
|
||||||
|
if (array == NULL)
|
||||||
|
return -ENOBUFS;
|
||||||
|
|
||||||
|
vg->vlan_devices_arrays[vid / VLAN_GROUP_ARRAY_PART_LEN] = array;
|
||||||
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
static void vlan_rcu_free(struct rcu_head *rcu)
|
static void vlan_rcu_free(struct rcu_head *rcu)
|
||||||
|
@ -247,6 +253,10 @@ int register_vlan_dev(struct net_device *dev)
|
||||||
return -ENOBUFS;
|
return -ENOBUFS;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
err = vlan_group_prealloc_vid(grp, vlan_id);
|
||||||
|
if (err < 0)
|
||||||
|
goto out_free_group;
|
||||||
|
|
||||||
err = register_netdevice(dev);
|
err = register_netdevice(dev);
|
||||||
if (err < 0)
|
if (err < 0)
|
||||||
goto out_free_group;
|
goto out_free_group;
|
||||||
|
|
Loading…
Add table
Reference in a new issue