Main Page   File List   File Members  

elsacct.c File Reference

#include <linux/module.h>
#include <linux/config.h>
#include <linux/init.h>
#include <linux/errno.h>
#include <linux/fs.h>
#include <linux/proc_fs.h>
#include <linux/seq_file.h>
#include <linux/slab.h>
#include <linux/string.h>
#include <linux/blkdev.h>
#include <linux/times.h>
#include <linux/elsacct.h>
#include <linux/bank.h>
#include <asm/uaccess.h>

Go to the source code of this file.

Defines

#define dprintk(format...)
#define ELSACCT_FILE   "/var/log/bank"
#define MANTSIZE   13
#define EXPSIZE   3
#define MAXFRACT   ((1 << MANTSIZE) - 1)

Typedefs

typedef __u16 comp_t

Functions

int do_elsacct (int opcode, struct elsa_bank *b, struct elsa_data *d)
void elsacct_process_copy (struct task_struct *from, struct task_struct *to)
void elsacct_process_remove (struct task_struct *p)
int elsacct_ioctl (struct inode *inode, struct file *filp, unsigned int cmd, unsigned long arg)
 module_init (elsacct_init_module)
 module_exit (elsacct_cleanup_module)
 MODULE_DESCRIPTION ("Enhanced Linux System Accounting.")
 MODULE_AUTHOR ("Guillaume Thouvenin< guillaume.thouvenin @bull.net >")
 MODULE_LICENSE ("GPL")

Variables

file_operations elsa_fops


Define Documentation

#define dprintk format...   
 

Definition at line 35 of file elsacct.c.

Referenced by elsacct_ioctl().

#define ELSACCT_FILE   "/var/log/bank"
 

Definition at line 39 of file elsacct.c.

#define EXPSIZE   3
 

Definition at line 60 of file elsacct.c.

#define MANTSIZE   13
 

encode_comp_t - encode an unsigned long into a comp_t @value: value to encode

Description: This routine has been adopted from the encode_comp_t() function in the kern_acct.c file of the FreeBSD operating system. The encoding is a 13-bit fraction with a 3-bit (base 8) exponent. This routine is taken from kernel/acct.c

Definition at line 59 of file elsacct.c.

#define MAXFRACT   ((1 << MANTSIZE) - 1)
 

Definition at line 61 of file elsacct.c.


Typedef Documentation

typedef __u16 comp_t
 

Definition at line 63 of file elsacct.c.


Function Documentation

int do_elsacct int    opcode,
struct elsa_bank *    b,
struct elsa_data *    d
 

do_elsacct - choose the write accounting treatment @opcode: operation to perform @b: pointer to a bank @d: pointer to a data

Description: it allows to choose to update accounting information when a data is removed from a bank or to dump information if a bank is removed.

Definition at line 201 of file elsacct.c.

Referenced by elsacct_ioctl().

00202 {
00203         int retval = 0;
00204 
00205         switch (opcode) {
00206         case ELSA_BANK_CALLBACK:
00207                 BUG_ON(b == NULL);
00208                 retval = do_elsacct_bank(b);
00209                 break;
00210         case ELSA_DATA_CALLBACK:
00211                 BUG_ON(b == NULL);
00212                 BUG_ON(d == NULL);
00213                 retval = do_elsacct_data(b, d);
00214                 break;
00215         default:
00216                 printk("do_elsacct: unknown opcode\n");
00217                 retval = -ENOIOCTLCMD;
00218                 break;
00219         }
00220 
00221         return retval;
00222 }

int elsacct_ioctl struct inode *    inode,
struct file *    filp,
unsigned int    cmd,
unsigned long    arg
 

elsacct_ioctl - issue a device specific command @inode: pointer to inode structure @filp : file pointer @cmd : specific command to perform.

  • : arguments that depend of cmd.
Descritpion: Perform some actions on banks that depend of command passed to the system call

Definition at line 332 of file elsacct.c.

References do_elsacct(), dprintk, elsa_bank_add(), elsa_bank_alloc(), elsa_bank_free(), elsa_bank_remove(), elsa_data_add(), elsa_data_alloc(), elsa_data_free(), elsa_data_remove(), elsa_get_bank(), and elsa_get_data().

00334 {
00335         static int retval;      /* retval == 0 */
00336         static struct elsa_ioctl_args iarg;
00337         static struct task_struct *p;
00338         static struct elsa_bank *b;
00339         static struct elsa_data *d;
00340 
00341         /* We can make some checks */
00342         if (_IOC_TYPE(cmd) != ELSACCT_MAGIC)
00343                 return -ENOTTY;
00344         if (_IOC_NR(cmd) > ELSACCT_MAXNR)
00345                 return -ENOTTY;
00346         if ((cmd != ELSACCT_PROCESS_ADD) && (cmd != ELSACCT_PROCESS_REMOVE) &&
00347             (cmd != ELSACCT_PROCESS_REMOVE_ALL) && (cmd != ELSACCT_BANK_CLEAN))
00348                 return -ENOTTY;
00349 
00350         if (!access_ok(VERIFY_READ, (void *)arg, _IOC_SIZE(cmd)))
00351                 return -EFAULT;
00352 
00353         /* Recover ioctl parameters */
00354         if (copy_from_user(&iarg, (struct elsa_ioctl_args *)arg,
00355                            sizeof(struct elsa_ioctl_args))) {
00356                 return -EFAULT;
00357         }
00358 
00359         dprintk("ELSAIOCTL: iarg = {bid#%d - pid#%d}\n", iarg.bid, iarg.pid);
00360 
00361         switch (cmd) {
00362         case ELSACCT_PROCESS_ADD:
00363                 /*
00364                  * To add a process to a bank we need to perform 
00365                  * following actions:
00366                  *     1) Allocate memory space to data (container)
00367                  *     2) Allocate memory space to a bank or get a 
00368                  *        pointer to an existing bank if BID != 0
00369                  *     3) Update process in data field
00370                  *     4) Add the container to the bank
00371                  */
00372                 /* 
00373                  * We need the pointer to task_struct. We catch it now so if 
00374                  * it fails we will not allocate memory for new data
00375                  */
00376                 read_lock(&tasklist_lock);
00377                 p = find_task_by_pid(iarg.pid);
00378                 read_unlock(&tasklist_lock);
00379                 if (!p) {
00380                         dprintk("elsacct_ioct: PID#%d not found\n", iarg.pid);
00381                         return -EAGAIN;
00382                 }
00383 
00384                 d = elsa_data_alloc();
00385                 if (!d) {
00386                         return -ENOMEM;
00387                 }
00388 
00389                 if (iarg.bid == 0) {
00390                         struct elsa_acct *acctinfo;
00391                         int bid;
00392 
00393                         acctinfo = (struct elsa_acct *)
00394                             kmalloc(sizeof(struct elsa_acct), GFP_KERNEL);
00395                         if (!acctinfo) {
00396                                 printk
00397                                     ("ELSA: cannot allocate space for accounting \n");
00398                                 return -ENOMEM;
00399                         }
00400 
00401                         b = elsa_bank_alloc();
00402                         if (!b) {
00403                                 dprintk("elsacct_ioct: cannot create BID\n");
00404                                 /* release memory */
00405                                 elsa_data_free(d);
00406                                 kfree(acctinfo);
00407                                 return -ENOMEM;
00408                         }
00409 
00410                         /* Macro to initialize accounting informations */
00411                         elsa_acct_init(acctinfo);
00412 
00413                         spin_lock_irq(&elsa_lock);
00414                         bid = elsa_bank_add(&elsa_br, b, &do_elsacct, acctinfo);
00415                         spin_unlock_irq(&elsa_lock);
00416 
00417                         if (!bid) {
00418                                 dprintk
00419                                     ("elsacct_ioct: error during bank creation\n");
00420                                 /* release memory */
00421                                 elsa_data_free(d);
00422                                 elsa_bank_free(b);
00423                                 kfree(acctinfo);
00424                                 return -EFAULT;
00425                         }
00426                 } else {
00427                         b = elsa_get_bank(&elsa_br, iarg.bid);
00428                         if (!b) {
00429                                 dprintk("elsacct_ioct: BID#%d not found\n",
00430                                         iarg.bid);
00431                                 /* release memory */
00432                                 elsa_data_free(d);
00433                                 return -EINVAL;
00434                         }
00435                 }
00436 
00437                 /* At this point, b cannot be NULL */
00438 
00439                 d->process = p;
00440 
00441                 spin_lock_irq(&elsa_lock);
00442                 retval = elsa_data_add(b, d);
00443                 spin_unlock_irq(&elsa_lock);
00444                 break;
00445 
00446         case ELSACCT_PROCESS_REMOVE:
00447                 /*
00448                  * We want to remove a process from one given bank
00449                  * Steps are:
00450                  *      1) get a pointer to the given bank
00451                  *      2) get a pointer to the container of the process
00452                  *      3) remove the process
00453                  *      4) free memory used by it
00454                  *      5) release bank if empty
00455                  */
00456                 spin_lock_irq(&elsa_lock);
00457 
00458                 b = elsa_get_bank(&elsa_br, iarg.bid);
00459                 if (!b) {
00460                         spin_unlock_irq(&elsa_lock);
00461                         dprintk("EBR: bank not found\n");
00462                         return -EINVAL;
00463                 }
00464 
00465                 d = elsa_get_data(iarg.pid, iarg.bid);
00466                 if (!d) {
00467                         spin_unlock_irq(&elsa_lock);
00468                         dprintk("EBR: data not found\n");
00469                         return -EINVAL;
00470                 }
00471 
00472                 retval = elsa_data_remove(b, d);
00473 
00474                 elsa_data_free(d);
00475 
00476                 if (retval) {
00477                         dprintk("EBR: Bank is now empty\n");
00478 
00479                         elsa_bank_remove(b);
00480 
00481                         dprintk("EBR: bank release\n");
00482 
00483                         spin_unlock_irq(&elsa_lock);
00484                         kfree(b->info);
00485                         elsa_bank_free(b);
00486                 } else {
00487                         spin_unlock_irq(&elsa_lock);
00488                 }
00489 
00490                 retval = 0;
00491                 break;
00492 
00493         case ELSACCT_PROCESS_REMOVE_ALL:
00494                 /*
00495                  * remove a process from all banks that it
00496                  * belongs. To achieve this we need to:
00497                  *      1) get a pointer to the process
00498                  *      2) get the first data 
00499                  *      3) remove it from the bank
00500                  *      4) release the data
00501                  *      5) if necessary, remove and release bank
00502                  *
00503                  */
00504                 read_lock(&tasklist_lock);
00505                 p = find_task_by_pid(iarg.pid);
00506                 read_unlock(&tasklist_lock);
00507                 if (!p) {
00508                         dprintk("elsacct_ioct: PID#%d not found\n", iarg.pid);
00509                         return -EAGAIN;
00510                 }
00511 
00512                 retval = 1;
00513                 while (retval) {
00514                         int bid;
00515 
00516                         spin_lock_irq(&elsa_lock);
00517 
00518                         read_lock(&tasklist_lock);
00519                         if (list_empty(&p->bank_head)) {
00520                                 d = NULL;
00521                         } else {
00522                                 d = list_entry((p->bank_head).next,
00523                                                struct elsa_data, bank_list);
00524                                 dprintk("EPRA: remove ...");
00525                                 BUG_ON(d == NULL);
00526                                 dprintk(" from bank #%d ...", d->bid);
00527                                 dprintk(" pid #%d\n", d->process->pid);
00528                         }
00529                         read_unlock(&tasklist_lock);
00530 
00531                         if (d != NULL) {
00532                                 /* We don't need to check the return value */
00533                                 b = elsa_get_bank(&elsa_br, d->bid);
00534                                 bid = elsa_data_remove(b, d);
00535 
00536                                 elsa_data_free(d);
00537 
00538                                 if (bid) {
00539                                         /* bank is empty */
00540                                         elsa_bank_remove(b);
00541                                         /* 
00542                                          * unlock before releasing memory 
00543                                          * occupied by the bank 
00544                                          */
00545                                         spin_unlock_irq(&elsa_lock);
00546                                         kfree(b->info);
00547                                         elsa_bank_free(b);
00548                                 }
00549                         } else {
00550                                 dprintk
00551                                     ("EPRA: Process removed from all banks\n");
00552                                 retval = 0;
00553                                 spin_unlock_irq(&elsa_lock);
00554                         }
00555 
00556                 }
00557                 break;
00558 
00559         case ELSACCT_BANK_CLEAN:
00560                 /*
00561                  * remove all process from one given bank
00562                  * Steps are:
00563                  *      1) get a pointer to the given bank
00564                  *      2) while there is data in the bank
00565                  *              -> remove the data
00566                  *      3) release bank
00567                  */
00568                 spin_lock_irq(&elsa_lock);
00569                 b = elsa_get_bank(&elsa_br, iarg.bid);
00570 
00571                 if (!b) {
00572                         spin_unlock_irq(&elsa_lock);
00573                         dprintk("EBC: bank not found\n");
00574                         return -EINVAL;
00575                 }
00576 
00577                 retval = 1;
00578                 while (retval) {
00579 
00580                         if (list_empty(&b->data_head)) {
00581                                 d = NULL;
00582                         } else {
00583                                 d = list_entry((b->data_head).next,
00584                                                struct elsa_data, data_list);
00585                         }
00586 
00587                         if (d) {
00588                                 dprintk("EBC: Remove pid#%d from bank#%d\n",
00589                                         d->process->pid, b->bid);
00590                                 /* We don't need to check the return value */
00591                                 elsa_data_remove(b, d);
00592                                 elsa_data_free(d);
00593                         } else {
00594                                 retval = 0;
00595                                 dprintk("EBC: Bank#%d is now empty\n", b->bid);
00596                         }
00597                 }
00598 
00599                 /* bank is empty */
00600                 elsa_bank_remove(b);
00601 
00602                 spin_unlock_irq(&elsa_lock);
00603                 kfree(b->info);
00604                 elsa_bank_free(b);
00605 
00606                 break;
00607 
00608         default:
00609                 /*
00610                  * The POSIX standard, states that if an inappropriate ioctl command 
00611                  * has been issued, then -ENOTTY should be returned.
00612                  */
00613                 retval = -ENOTTY;
00614         };
00615 
00616         return retval;
00617 }

void elsacct_process_copy struct task_struct *    from,
struct task_struct *    to
 

Definition at line 225 of file elsacct.c.

References elsa_data_add(), elsa_data_alloc(), elsa_data_free(), and elsa_get_bank().

00226 {
00227         static struct list_head *entry; /* entry == NULL */
00228         static struct elsa_bank *b;     /* b == NULL */
00229         static struct elsa_data *d;     /* d == NULL */
00230         static struct elsa_data *new_d; /* d == NULL */
00231 
00232         /* 
00233          * First, initialize to->bank_head otherwise, if 
00234          * from->bank_head is NULL it won't be initialize
00235          */
00236         write_lock_irq(&tasklist_lock);
00237         INIT_LIST_HEAD(&(to->bank_head));
00238         write_unlock_irq(&tasklist_lock);
00239 
00240         list_for_each(entry, &from->bank_head) {
00241                 read_lock_irq(&tasklist_lock);
00242                 d = list_entry(entry, struct elsa_data, bank_list);
00243                 read_unlock_irq(&tasklist_lock);
00244 
00245                 new_d = elsa_data_alloc();
00246                 if (new_d) {
00247                         spin_lock_irq(&elsa_lock);
00248                         b = elsa_get_bank(&elsa_br, d->bid);
00249                         if (!b) {
00250                                 /* release memory */
00251                                 elsa_data_free(new_d);
00252                         } else {
00253                                 new_d->process = to;
00254                                 if (!elsa_data_add(b, new_d))
00255                                         elsa_data_free(new_d);
00256                         }
00257                         spin_unlock_irq(&elsa_lock);
00258                 }
00259         }
00260 }

void elsacct_process_remove struct task_struct *    p
 

Definition at line 263 of file elsacct.c.

References elsa_bank_free(), elsa_bank_remove(), elsa_data_free(), elsa_data_remove(), and elsa_get_bank().

00264 {
00265         static struct elsa_data *d;     /* d == NULL */
00266         static struct elsa_bank *b;     /* b == NULL */
00267         int empty = 0;
00268 
00269         while (!empty) {
00270                 int bid;
00271 
00272                 spin_lock_irq(&elsa_lock);
00273 
00274                 read_lock(&tasklist_lock);
00275                 if (list_empty(&p->bank_head)) {
00276                         d = NULL;
00277                 } else {
00278                         d = list_entry((p->bank_head).next,
00279                                        struct elsa_data, bank_list);
00280                 }
00281                 read_unlock(&tasklist_lock);
00282 
00283                 if (d != NULL) {
00284                         /* We don't need to check the return value */
00285                         b = elsa_get_bank(&elsa_br, d->bid);
00286                         bid = elsa_data_remove(b, d);
00287 
00288                         elsa_data_free(d);
00289 
00290                         if (bid) {
00291                                 /* bank is empty */
00292                                 elsa_bank_remove(b);
00293                                 /* 
00294                                  * unlock before releasing memory used by 
00295                                  * the bank
00296                                  */
00297                                 spin_unlock_irq(&elsa_lock);
00298                                 kfree(b->info);
00299                                 elsa_bank_free(b);
00300                         }
00301                 } else {
00302                         empty = 1;
00303                         spin_unlock_irq(&elsa_lock);
00304                 }
00305 
00306         }
00307 }

MODULE_AUTHOR "Guillaume Thouvenin< guillaume.thouvenin @bull.net >"   
 

MODULE_DESCRIPTION "Enhanced Linux System Accounting."   
 

module_exit elsacct_cleanup_module   
 

module_init elsacct_init_module   
 

MODULE_LICENSE "GPL"   
 


Variable Documentation

struct file_operations elsa_fops
 

Initial value:

 {
        .owner = THIS_MODULE,
        .ioctl = elsacct_ioctl,
}

Definition at line 619 of file elsacct.c.


Generated on Wed Jul 7 08:31:37 2004 for Enhanced Linux System Accounting by doxygen1.2.18