LCOV - code coverage report
Current view: top level - kernel - delayacct.c (source / functions) Hit Total Coverage
Test: combined.info Lines: 38 100 38.0 %
Date: 2022-04-01 14:58:12 Functions: 6 12 50.0 %
Branches: 4 28 14.3 %

           Branch data     Line data    Source code
       1                 :            : // SPDX-License-Identifier: GPL-2.0-or-later
       2                 :            : /* delayacct.c - per-task delay accounting
       3                 :            :  *
       4                 :            :  * Copyright (C) Shailabh Nagar, IBM Corp. 2006
       5                 :            :  */
       6                 :            : 
       7                 :            : #include <linux/sched.h>
       8                 :            : #include <linux/sched/task.h>
       9                 :            : #include <linux/sched/cputime.h>
      10                 :            : #include <linux/slab.h>
      11                 :            : #include <linux/taskstats.h>
      12                 :            : #include <linux/time.h>
      13                 :            : #include <linux/sysctl.h>
      14                 :            : #include <linux/delayacct.h>
      15                 :            : #include <linux/module.h>
      16                 :            : 
      17                 :            : int delayacct_on __read_mostly = 1;     /* Delay accounting turned on/off */
      18                 :            : EXPORT_SYMBOL_GPL(delayacct_on);
      19                 :            : struct kmem_cache *delayacct_cache;
      20                 :            : 
      21                 :          0 : static int __init delayacct_setup_disable(char *str)
      22                 :            : {
      23                 :          0 :         delayacct_on = 0;
      24                 :          0 :         return 1;
      25                 :            : }
      26                 :            : __setup("nodelayacct", delayacct_setup_disable);
      27                 :            : 
      28                 :          3 : void delayacct_init(void)
      29                 :            : {
      30                 :          3 :         delayacct_cache = KMEM_CACHE(task_delay_info, SLAB_PANIC|SLAB_ACCOUNT);
      31                 :          3 :         delayacct_tsk_init(&init_task);
      32                 :          3 : }
      33                 :            : 
      34                 :       2402 : void __delayacct_tsk_init(struct task_struct *tsk)
      35                 :            : {
      36                 :       2402 :         tsk->delays = kmem_cache_zalloc(delayacct_cache, GFP_KERNEL);
      37   [ +  -  +  - ]:       2402 :         if (tsk->delays)
      38                 :       2402 :                 raw_spin_lock_init(&tsk->delays->lock);
      39                 :       2399 : }
      40                 :            : 
      41                 :            : /*
      42                 :            :  * Finish delay accounting for a statistic using its timestamps (@start),
      43                 :            :  * accumalator (@total) and @count
      44                 :            :  */
      45                 :       2599 : static void delayacct_end(raw_spinlock_t *lock, u64 *start, u64 *total,
      46                 :            :                           u32 *count)
      47                 :            : {
      48                 :       2599 :         s64 ns = ktime_get_ns() - *start;
      49                 :       2599 :         unsigned long flags;
      50                 :            : 
      51         [ +  - ]:       2599 :         if (ns > 0) {
      52                 :       2599 :                 raw_spin_lock_irqsave(lock, flags);
      53                 :       2599 :                 *total += ns;
      54                 :       2599 :                 (*count)++;
      55                 :       2599 :                 raw_spin_unlock_irqrestore(lock, flags);
      56                 :            :         }
      57                 :       2599 : }
      58                 :            : 
      59                 :       2599 : void __delayacct_blkio_start(void)
      60                 :            : {
      61                 :       2599 :         current->delays->blkio_start = ktime_get_ns();
      62                 :       2599 : }
      63                 :            : 
      64                 :            : /*
      65                 :            :  * We cannot rely on the `current` macro, as we haven't yet switched back to
      66                 :            :  * the process being woken.
      67                 :            :  */
      68                 :       2599 : void __delayacct_blkio_end(struct task_struct *p)
      69                 :            : {
      70                 :       2599 :         struct task_delay_info *delays = p->delays;
      71                 :       2599 :         u64 *total;
      72                 :       2599 :         u32 *count;
      73                 :            : 
      74         [ -  + ]:       2599 :         if (p->delays->flags & DELAYACCT_PF_SWAPIN) {
      75                 :          0 :                 total = &delays->swapin_delay;
      76                 :          0 :                 count = &delays->swapin_count;
      77                 :            :         } else {
      78                 :       2599 :                 total = &delays->blkio_delay;
      79                 :       2599 :                 count = &delays->blkio_count;
      80                 :            :         }
      81                 :            : 
      82                 :       2599 :         delayacct_end(&delays->lock, &delays->blkio_start, total, count);
      83                 :       2599 : }
      84                 :            : 
      85                 :          0 : int __delayacct_add_tsk(struct taskstats *d, struct task_struct *tsk)
      86                 :            : {
      87                 :          0 :         u64 utime, stime, stimescaled, utimescaled;
      88                 :          0 :         unsigned long long t2, t3;
      89                 :          0 :         unsigned long flags, t1;
      90                 :          0 :         s64 tmp;
      91                 :            : 
      92         [ #  # ]:          0 :         task_cputime(tsk, &utime, &stime);
      93                 :          0 :         tmp = (s64)d->cpu_run_real_total;
      94                 :          0 :         tmp += utime + stime;
      95         [ #  # ]:          0 :         d->cpu_run_real_total = (tmp < (s64)d->cpu_run_real_total) ? 0 : tmp;
      96                 :            : 
      97         [ #  # ]:          0 :         task_cputime_scaled(tsk, &utimescaled, &stimescaled);
      98                 :          0 :         tmp = (s64)d->cpu_scaled_run_real_total;
      99                 :          0 :         tmp += utimescaled + stimescaled;
     100                 :          0 :         d->cpu_scaled_run_real_total =
     101         [ #  # ]:          0 :                 (tmp < (s64)d->cpu_scaled_run_real_total) ? 0 : tmp;
     102                 :            : 
     103                 :            :         /*
     104                 :            :          * No locking available for sched_info (and too expensive to add one)
     105                 :            :          * Mitigate by taking snapshot of values
     106                 :            :          */
     107                 :          0 :         t1 = tsk->sched_info.pcount;
     108                 :          0 :         t2 = tsk->sched_info.run_delay;
     109                 :          0 :         t3 = tsk->se.sum_exec_runtime;
     110                 :            : 
     111                 :          0 :         d->cpu_count += t1;
     112                 :            : 
     113                 :          0 :         tmp = (s64)d->cpu_delay_total + t2;
     114         [ #  # ]:          0 :         d->cpu_delay_total = (tmp < (s64)d->cpu_delay_total) ? 0 : tmp;
     115                 :            : 
     116                 :          0 :         tmp = (s64)d->cpu_run_virtual_total + t3;
     117                 :          0 :         d->cpu_run_virtual_total =
     118         [ #  # ]:          0 :                 (tmp < (s64)d->cpu_run_virtual_total) ?   0 : tmp;
     119                 :            : 
     120                 :            :         /* zero XXX_total, non-zero XXX_count implies XXX stat overflowed */
     121                 :            : 
     122                 :          0 :         raw_spin_lock_irqsave(&tsk->delays->lock, flags);
     123                 :          0 :         tmp = d->blkio_delay_total + tsk->delays->blkio_delay;
     124         [ #  # ]:          0 :         d->blkio_delay_total = (tmp < d->blkio_delay_total) ? 0 : tmp;
     125                 :          0 :         tmp = d->swapin_delay_total + tsk->delays->swapin_delay;
     126         [ #  # ]:          0 :         d->swapin_delay_total = (tmp < d->swapin_delay_total) ? 0 : tmp;
     127                 :          0 :         tmp = d->freepages_delay_total + tsk->delays->freepages_delay;
     128         [ #  # ]:          0 :         d->freepages_delay_total = (tmp < d->freepages_delay_total) ? 0 : tmp;
     129                 :          0 :         tmp = d->thrashing_delay_total + tsk->delays->thrashing_delay;
     130         [ #  # ]:          0 :         d->thrashing_delay_total = (tmp < d->thrashing_delay_total) ? 0 : tmp;
     131                 :          0 :         d->blkio_count += tsk->delays->blkio_count;
     132                 :          0 :         d->swapin_count += tsk->delays->swapin_count;
     133                 :          0 :         d->freepages_count += tsk->delays->freepages_count;
     134                 :          0 :         d->thrashing_count += tsk->delays->thrashing_count;
     135                 :          0 :         raw_spin_unlock_irqrestore(&tsk->delays->lock, flags);
     136                 :            : 
     137                 :          0 :         return 0;
     138                 :            : }
     139                 :            : 
     140                 :        296 : __u64 __delayacct_blkio_ticks(struct task_struct *tsk)
     141                 :            : {
     142                 :        296 :         __u64 ret;
     143                 :        296 :         unsigned long flags;
     144                 :            : 
     145                 :        296 :         raw_spin_lock_irqsave(&tsk->delays->lock, flags);
     146                 :        296 :         ret = nsec_to_clock_t(tsk->delays->blkio_delay +
     147                 :        296 :                                 tsk->delays->swapin_delay);
     148                 :        296 :         raw_spin_unlock_irqrestore(&tsk->delays->lock, flags);
     149                 :        296 :         return ret;
     150                 :            : }
     151                 :            : 
     152                 :          0 : void __delayacct_freepages_start(void)
     153                 :            : {
     154                 :          0 :         current->delays->freepages_start = ktime_get_ns();
     155                 :          0 : }
     156                 :            : 
     157                 :          0 : void __delayacct_freepages_end(void)
     158                 :            : {
     159                 :          0 :         delayacct_end(
     160                 :          0 :                 &current->delays->lock,
     161                 :          0 :                 &current->delays->freepages_start,
     162                 :          0 :                 &current->delays->freepages_delay,
     163                 :          0 :                 &current->delays->freepages_count);
     164                 :          0 : }
     165                 :            : 
     166                 :          0 : void __delayacct_thrashing_start(void)
     167                 :            : {
     168                 :          0 :         current->delays->thrashing_start = ktime_get_ns();
     169                 :          0 : }
     170                 :            : 
     171                 :          0 : void __delayacct_thrashing_end(void)
     172                 :            : {
     173                 :          0 :         delayacct_end(&current->delays->lock,
     174                 :          0 :                       &current->delays->thrashing_start,
     175                 :          0 :                       &current->delays->thrashing_delay,
     176                 :          0 :                       &current->delays->thrashing_count);
     177                 :          0 : }

Generated by: LCOV version 1.14