diff options
author | Eric Dumazet <dada1@resalehost.networksolutions.com> | 2006-03-25 06:08:00 -0500 |
---|---|---|
committer | Linus Torvalds <torvalds@g5.osdl.org> | 2006-03-25 11:22:58 -0500 |
commit | 231bed205879236357171e50bd8965e70797ecdc (patch) | |
tree | fe3bb52388510eca3fc19548abd26f4009b94db1 /kernel/sys.c | |
parent | 76c67de460b3d00b7ab8a96bb18f07ca47d65fba (diff) |
[PATCH] No need to protect current->group_info in sys_getgroups(), in_group_p() and in_egroup_p()
While doing some benchmarks of an Apache/PHP SMP server, I noticed high
oprofile numbers in in_group_p() and _atomic_dec_and_lock().
rank percent
1 4.8911 % __link_path_walk
2 4.8503 % __d_lookup
*3 4.2911 % _atomic_dec_and_lock
4 3.9307 % __copy_to_user_ll
5 4.9004 % sysenter_past_esp
*6 3.3248 % in_group_p
It appears that in_group_p() does an uncessary
get_group_info(current->group_info); /* atomic_inc() */
... /* access current->group_info */
put_group_info(current->group_info); /* _atomic_dec_and_lock */
It is not necessary to do this, because the current task holds a reference
on its own group_info, and this reference cannot change during the lookup.
This patch deletes the get_group_info()/put_group_info() pair from
sys_getgroups(), in_group_p() and in_egroup_p() functions.
Signed-off-by: Eric Dumazet <dada1@cosmosbay.com>
Cc: Tim Hockin <thockin@hockin.org>
Signed-off-by: Andrew Morton <akpm@osdl.org>
Signed-off-by: Linus Torvalds <torvalds@osdl.org>
Diffstat (limited to 'kernel/sys.c')
-rw-r--r-- | kernel/sys.c | 6 |
1 files changed, 0 insertions, 6 deletions
diff --git a/kernel/sys.c b/kernel/sys.c index 421009cedb51..119fb0d9e24e 100644 --- a/kernel/sys.c +++ b/kernel/sys.c | |||
@@ -1421,7 +1421,6 @@ asmlinkage long sys_getgroups(int gidsetsize, gid_t __user *grouplist) | |||
1421 | return -EINVAL; | 1421 | return -EINVAL; |
1422 | 1422 | ||
1423 | /* no need to grab task_lock here; it cannot change */ | 1423 | /* no need to grab task_lock here; it cannot change */ |
1424 | get_group_info(current->group_info); | ||
1425 | i = current->group_info->ngroups; | 1424 | i = current->group_info->ngroups; |
1426 | if (gidsetsize) { | 1425 | if (gidsetsize) { |
1427 | if (i > gidsetsize) { | 1426 | if (i > gidsetsize) { |
@@ -1434,7 +1433,6 @@ asmlinkage long sys_getgroups(int gidsetsize, gid_t __user *grouplist) | |||
1434 | } | 1433 | } |
1435 | } | 1434 | } |
1436 | out: | 1435 | out: |
1437 | put_group_info(current->group_info); | ||
1438 | return i; | 1436 | return i; |
1439 | } | 1437 | } |
1440 | 1438 | ||
@@ -1475,9 +1473,7 @@ int in_group_p(gid_t grp) | |||
1475 | { | 1473 | { |
1476 | int retval = 1; | 1474 | int retval = 1; |
1477 | if (grp != current->fsgid) { | 1475 | if (grp != current->fsgid) { |
1478 | get_group_info(current->group_info); | ||
1479 | retval = groups_search(current->group_info, grp); | 1476 | retval = groups_search(current->group_info, grp); |
1480 | put_group_info(current->group_info); | ||
1481 | } | 1477 | } |
1482 | return retval; | 1478 | return retval; |
1483 | } | 1479 | } |
@@ -1488,9 +1484,7 @@ int in_egroup_p(gid_t grp) | |||
1488 | { | 1484 | { |
1489 | int retval = 1; | 1485 | int retval = 1; |
1490 | if (grp != current->egid) { | 1486 | if (grp != current->egid) { |
1491 | get_group_info(current->group_info); | ||
1492 | retval = groups_search(current->group_info, grp); | 1487 | retval = groups_search(current->group_info, grp); |
1493 | put_group_info(current->group_info); | ||
1494 | } | 1488 | } |
1495 | return retval; | 1489 | return retval; |
1496 | } | 1490 | } |