Commit cbfad782 authored by Bob Miller's avatar Bob Miller Committed by Linus Torvalds

[PATCH] 2.5.7-pre1 Code cleanup for BSD accounting.

Clean up BSD accounting locking code..
parent 98f0db7a
...@@ -89,18 +89,12 @@ struct acct_glbs { ...@@ -89,18 +89,12 @@ struct acct_glbs {
static struct acct_glbs acct_globals __cacheline_aligned = {SPIN_LOCK_UNLOCKED}; static struct acct_glbs acct_globals __cacheline_aligned = {SPIN_LOCK_UNLOCKED};
#define acct_lock acct_globals.lock
#define acct_active acct_globals.active
#define acct_needcheck acct_globals.needcheck
#define acct_file acct_globals.file
#define acct_timer acct_globals.timer
/* /*
* Called whenever the timer says to check the free space. * Called whenever the timer says to check the free space.
*/ */
static void acct_timeout(unsigned long unused) static void acct_timeout(unsigned long unused)
{ {
acct_needcheck = 1; acct_globals.needcheck = 1;
} }
/* /*
...@@ -112,11 +106,11 @@ static int check_free_space(struct file *file) ...@@ -112,11 +106,11 @@ static int check_free_space(struct file *file)
int res; int res;
int act; int act;
spin_lock(&acct_lock); spin_lock(&acct_globals.lock);
res = acct_active; res = acct_globals.active;
if (!file || !acct_needcheck) if (!file || !acct_globals.needcheck)
goto out; goto out;
spin_unlock(&acct_lock); spin_unlock(&acct_globals.lock);
/* May block */ /* May block */
if (vfs_statfs(file->f_dentry->d_inode->i_sb, &sbuf)) if (vfs_statfs(file->f_dentry->d_inode->i_sb, &sbuf))
...@@ -130,129 +124,130 @@ static int check_free_space(struct file *file) ...@@ -130,129 +124,130 @@ static int check_free_space(struct file *file)
act = 0; act = 0;
/* /*
* If some joker switched acct_file under us we'ld better be * If some joker switched acct_globals.file under us we'ld better be
* silent and _not_ touch anything. * silent and _not_ touch anything.
*/ */
spin_lock(&acct_lock); spin_lock(&acct_globals.lock);
if (file != acct_file) { if (file != acct_globals.file) {
if (act) if (act)
res = act>0; res = act>0;
goto out; goto out;
} }
if (acct_active) { if (acct_globals.active) {
if (act < 0) { if (act < 0) {
acct_active = 0; acct_globals.active = 0;
printk(KERN_INFO "Process accounting paused\n"); printk(KERN_INFO "Process accounting paused\n");
} }
} else { } else {
if (act > 0) { if (act > 0) {
acct_active = 1; acct_globals.active = 1;
printk(KERN_INFO "Process accounting resumed\n"); printk(KERN_INFO "Process accounting resumed\n");
} }
} }
del_timer(&acct_timer); del_timer(&acct_globals.timer);
acct_needcheck = 0; acct_globals.needcheck = 0;
acct_timer.expires = jiffies + ACCT_TIMEOUT*HZ; acct_globals.timer.expires = jiffies + ACCT_TIMEOUT*HZ;
add_timer(&acct_timer); add_timer(&acct_globals.timer);
res = acct_active; res = acct_globals.active;
out: out:
spin_unlock(&acct_lock); spin_unlock(&acct_globals.lock);
return res; return res;
} }
/* /*
* acct_common() is the main routine that implements process accounting. * Close the old accouting file (if currently open) and then replace
* It takes the name of the file where accounting records should be * it with file (if non-NULL).
* written. If the filename is NULL, accounting will be shutdown. *
* NOTE: acct_globals.lock MUST be held on entry and exit.
*/ */
long acct_common(const char *name, int locked) void acct_file_reopen(struct file *file)
{ {
struct file *file = NULL, *old_acct = NULL; struct file *old_acct = NULL;
char *tmp;
int error;
/* BUG_ON(!spin_is_locked(&acct_globals.lock));
* Should only have locked set when name is NULL (enforce this).
if (acct_globals.file) {
old_acct = acct_globals.file;
del_timer(&acct_globals.timer);
acct_globals.active = 0;
acct_globals.needcheck = 0;
acct_globals.file = NULL;
}
if (file) {
acct_globals.file = file;
acct_globals.needcheck = 0;
acct_globals.active = 1;
/* It's been deleted if it was used before so this is safe */
init_timer(&acct_globals.timer);
acct_globals.timer.function = acct_timeout;
acct_globals.timer.expires = jiffies + ACCT_TIMEOUT*HZ;
add_timer(&acct_globals.timer);
}
if (old_acct) {
spin_unlock(&acct_globals.lock);
do_acct_process(0, old_acct);
filp_close(old_acct, NULL);
spin_lock(&acct_globals.lock);
}
}
/*
* sys_acct() is the only system call needed to implement process
* accounting. It takes the name of the file where accounting records
* should be written. If the filename is NULL, accounting will be
* shutdown.
*/ */
BUG_ON(locked && name); asmlinkage long sys_acct(const char *name)
{
struct file *file = NULL;
char *tmp;
if (!capable(CAP_SYS_PACCT)) if (!capable(CAP_SYS_PACCT))
return -EPERM; return -EPERM;
if (name) { if (name) {
tmp = getname(name); tmp = getname(name);
error = PTR_ERR(tmp); if (IS_ERR(tmp)) {
if (IS_ERR(tmp)) return (PTR_ERR(tmp));
goto out; }
/* Difference from BSD - they don't do O_APPEND */ /* Difference from BSD - they don't do O_APPEND */
file = filp_open(tmp, O_WRONLY|O_APPEND, 0); file = filp_open(tmp, O_WRONLY|O_APPEND, 0);
putname(tmp); putname(tmp);
if (IS_ERR(file)) { if (IS_ERR(file)) {
error = PTR_ERR(file); return (PTR_ERR(file));
goto out;
} }
error = -EACCES; if (!S_ISREG(file->f_dentry->d_inode->i_mode)) {
if (!S_ISREG(file->f_dentry->d_inode->i_mode)) filp_close(file, NULL);
goto out_err; return (-EACCES);
error = -EIO;
if (!file->f_op->write)
goto out_err;
} }
error = 0; if (!file->f_op->write) {
if (!locked) filp_close(file, NULL);
spin_lock(&acct_lock); return (-EIO);
if (acct_file) {
old_acct = acct_file;
del_timer(&acct_timer);
acct_active = 0;
acct_needcheck = 0;
acct_file = NULL;
}
if (name) {
acct_file = file;
acct_needcheck = 0;
acct_active = 1;
/* It's been deleted if it was used before so this is safe */
init_timer(&acct_timer);
acct_timer.function = acct_timeout;
acct_timer.expires = jiffies + ACCT_TIMEOUT*HZ;
add_timer(&acct_timer);
} }
spin_unlock(&acct_lock);
if (old_acct) {
do_acct_process(0,old_acct);
filp_close(old_acct, NULL);
} }
out:
return error; spin_lock(&acct_globals.lock);
out_err: acct_file_reopen(file);
filp_close(file, NULL); spin_unlock(&acct_globals.lock);
goto out;
return (0);
} }
/* /*
* sys_acct() is the only system call needed to implement process * If the accouting is turned on for a file in the filesystem pointed
* accounting. It takes the name of the file where accounting records * to by sb, turn accouting off.
* should be written. If the filename is NULL, accounting will be
* shutdown.
*/ */
asmlinkage long sys_acct(const char *name)
{
return (acct_common(name, 0));
}
void acct_auto_close(struct super_block *sb) void acct_auto_close(struct super_block *sb)
{ {
spin_lock(&acct_lock); spin_lock(&acct_globals.lock);
if (acct_file && acct_file->f_dentry->d_inode->i_sb == sb) { if (acct_globals.file &&
(void) acct_common(NULL, 1); acct_globals.file->f_dentry->d_inode->i_sb == sb) {
} else { acct_file_reopen((struct file *)NULL);
spin_unlock(&acct_lock);
} }
spin_unlock(&acct_globals.lock);
} }
/* /*
...@@ -329,7 +324,8 @@ static void do_acct_process(long exitcode, struct file *file) ...@@ -329,7 +324,8 @@ static void do_acct_process(long exitcode, struct file *file)
strncpy(ac.ac_comm, current->comm, ACCT_COMM); strncpy(ac.ac_comm, current->comm, ACCT_COMM);
ac.ac_comm[ACCT_COMM - 1] = '\0'; ac.ac_comm[ACCT_COMM - 1] = '\0';
ac.ac_btime = CT_TO_SECS(current->start_time) + (xtime.tv_sec - (jiffies / HZ)); ac.ac_btime = CT_TO_SECS(current->start_time) +
(xtime.tv_sec - (jiffies / HZ));
ac.ac_etime = encode_comp_t(jiffies - current->start_time); ac.ac_etime = encode_comp_t(jiffies - current->start_time);
ac.ac_utime = encode_comp_t(current->times.tms_utime); ac.ac_utime = encode_comp_t(current->times.tms_utime);
ac.ac_stime = encode_comp_t(current->times.tms_stime); ac.ac_stime = encode_comp_t(current->times.tms_stime);
...@@ -390,15 +386,15 @@ static void do_acct_process(long exitcode, struct file *file) ...@@ -390,15 +386,15 @@ static void do_acct_process(long exitcode, struct file *file)
int acct_process(long exitcode) int acct_process(long exitcode)
{ {
struct file *file = NULL; struct file *file = NULL;
spin_lock(&acct_lock); spin_lock(&acct_globals.lock);
if (acct_file) { if (acct_globals.file) {
file = acct_file; file = acct_globals.file;
get_file(file); get_file(file);
spin_unlock(&acct_lock); spin_unlock(&acct_globals.lock);
do_acct_process(exitcode, file); do_acct_process(exitcode, file);
fput(file); fput(file);
} else } else
spin_unlock(&acct_lock); spin_unlock(&acct_globals.lock);
return 0; return 0;
} }
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment