aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorEric Dumazet <dada1@resalehost.networksolutions.com>2006-03-25 06:08:00 -0500
committerLinus Torvalds <torvalds@g5.osdl.org>2006-03-25 11:22:58 -0500
commit231bed205879236357171e50bd8965e70797ecdc (patch)
treefe3bb52388510eca3fc19548abd26f4009b94db1
parent76c67de460b3d00b7ab8a96bb18f07ca47d65fba (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>
-rw-r--r--kernel/sys.c6
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 }
1436out: 1435out:
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}