cregit-Linux how code gets into the kernel

Release 4.11 kernel/sched/cputime.c

Directory: kernel/sched
#include <linux/export.h>
#include <linux/sched.h>
#include <linux/tsacct_kern.h>
#include <linux/kernel_stat.h>
#include <linux/static_key.h>
#include <linux/context_tracking.h>
#include <linux/sched/cputime.h>
#include "sched.h"

#ifdef CONFIG_IRQ_TIME_ACCOUNTING

/*
 * There are no locks covering percpu hardirq/softirq time.
 * They are only modified in vtime_account, on corresponding CPU
 * with interrupts disabled. So, writes are safe.
 * They are read and saved off onto struct rq in update_rq_clock().
 * This may result in other CPU reading this CPU's irq time and can
 * race with irq/vtime_account on this CPU. We would either get old
 * or new value with a side effect of accounting a slice of irq time to wrong
 * task when irq is in progress while we read rq->clock. That is a worthy
 * compromise in place of having locks on each irq in account_system_time.
 */
DEFINE_PER_CPU(struct irqtime, cpu_irqtime);


static int sched_clock_irqtime;


void enable_sched_clock_irqtime(void) { sched_clock_irqtime = 1; }

Contributors

PersonTokensPropCommitsCommitProp
Frédéric Weisbecker11100.00%1100.00%
Total11100.00%1100.00%


void disable_sched_clock_irqtime(void) { sched_clock_irqtime = 0; }

Contributors

PersonTokensPropCommitsCommitProp
Frédéric Weisbecker11100.00%1100.00%
Total11100.00%1100.00%


static void irqtime_account_delta(struct irqtime *irqtime, u64 delta, enum cpu_usage_stat idx) { u64 *cpustat = kcpustat_this_cpu->cpustat; u64_stats_update_begin(&irqtime->sync); cpustat[idx] += delta; irqtime->total += delta; irqtime->tick_delta += delta; u64_stats_update_end(&irqtime->sync); }

Contributors

PersonTokensPropCommitsCommitProp
Frédéric Weisbecker61100.00%1100.00%
Total61100.00%1100.00%

/* * Called before incrementing preempt_count on {soft,}irq_enter * and before decrementing preempt_count on {soft,}irq_exit. */
void irqtime_account_irq(struct task_struct *curr) { struct irqtime *irqtime = this_cpu_ptr(&cpu_irqtime); s64 delta; int cpu; if (!sched_clock_irqtime) return; cpu = smp_processor_id(); delta = sched_clock_cpu(cpu) - irqtime->irq_start_time; irqtime->irq_start_time += delta; /* * We do not account for softirq time from ksoftirqd here. * We want to continue accounting softirq time to ksoftirqd thread * in that case, so as not to confuse scheduler with a special task * that do not consume any time, but still wants to run. */ if (hardirq_count()) irqtime_account_delta(irqtime, delta, CPUTIME_IRQ); else if (in_serving_softirq() && curr != this_cpu_ksoftirqd()) irqtime_account_delta(irqtime, delta, CPUTIME_SOFTIRQ); }

Contributors

PersonTokensPropCommitsCommitProp
Frédéric Weisbecker90100.00%4100.00%
Total90100.00%4100.00%

EXPORT_SYMBOL_GPL(irqtime_account_irq);
static u64 irqtime_tick_accounted(u64 maxtime) { struct irqtime *irqtime = this_cpu_ptr(&cpu_irqtime); u64 delta; delta = min(irqtime->tick_delta, maxtime); irqtime->tick_delta -= delta; return delta; }

Contributors

PersonTokensPropCommitsCommitProp
Frédéric Weisbecker3990.70%583.33%
Rik Van Riel49.30%116.67%
Total43100.00%6100.00%

#else /* CONFIG_IRQ_TIME_ACCOUNTING */ #define sched_clock_irqtime (0)
static u64 irqtime_tick_accounted(u64 dummy) { return 0; }

Contributors

PersonTokensPropCommitsCommitProp
Rik Van Riel975.00%133.33%
Frédéric Weisbecker325.00%266.67%
Total12100.00%3100.00%

#endif /* !CONFIG_IRQ_TIME_ACCOUNTING */
static inline void task_group_account_field(struct task_struct *p, int index, u64 tmp) { /* * Since all updates are sure to touch the root cgroup, we * get ourselves ahead and touch it first. If the root cgroup * is the only cgroup, then nothing else should be necessary. * */ __this_cpu_add(kernel_cpustat.cpustat[index], tmp); cpuacct_account_field(p, index, tmp); }

Contributors

PersonTokensPropCommitsCommitProp
Frédéric Weisbecker3485.00%133.33%
Li Zefan37.50%133.33%
Christoph Lameter37.50%133.33%
Total40100.00%3100.00%

/* * Account user cpu time to a process. * @p: the process that the cpu time gets accounted to * @cputime: the cpu time spent in user space since the last update */
void account_user_time(struct task_struct *p, u64 cputime) { int index; /* Add user time to process. */ p->utime += cputime; account_group_user_time(p, cputime); index = (task_nice(p) > 0) ? CPUTIME_NICE : CPUTIME_USER; /* Add user time to cpustat. */ task_group_account_field(p, index, cputime); /* Account for user time used */ acct_account_cputime(p); }

Contributors

PersonTokensPropCommitsCommitProp
Frédéric Weisbecker6098.36%375.00%
Dongsheng Yang11.64%125.00%
Total61100.00%4100.00%

/* * Account guest cpu time to a process. * @p: the process that the cpu time gets accounted to * @cputime: the cpu time spent in virtual machine since the last update */
void account_guest_time(struct task_struct *p, u64 cputime) { u64 *cpustat = kcpustat_this_cpu->cpustat; /* Add guest time to process. */ p->utime += cputime; account_group_user_time(p, cputime); p->gtime += cputime; /* Add guest time to cpustat. */ if (task_nice(p) > 0) { cpustat[CPUTIME_NICE] += cputime; cpustat[CPUTIME_GUEST_NICE] += cputime; } else { cpustat[CPUTIME_USER] += cputime; cpustat[CPUTIME_GUEST] += cputime; } }

Contributors

PersonTokensPropCommitsCommitProp
Frédéric Weisbecker8398.81%266.67%
Dongsheng Yang11.19%133.33%
Total84100.00%3100.00%

/* * Account system cpu time to a process and desired cpustat field * @p: the process that the cpu time gets accounted to * @cputime: the cpu time spent in kernel space since the last update * @index: pointer to cpustat field that has to be updated */
void account_system_index_time(struct task_struct *p, u64 cputime, enum cpu_usage_stat index) { /* Add system time to process. */ p->stime += cputime; account_group_system_time(p, cputime); /* Add system time to cpustat. */ task_group_account_field(p, index, cputime); /* Account for system time used */ acct_account_cputime(p); }

Contributors

PersonTokensPropCommitsCommitProp
Frédéric Weisbecker47100.00%4100.00%
Total47100.00%4100.00%

/* * Account system cpu time to a process. * @p: the process that the cpu time gets accounted to * @hardirq_offset: the offset to subtract from hardirq_count() * @cputime: the cpu time spent in kernel space since the last update */
void account_system_time(struct task_struct *p, int hardirq_offset, u64 cputime) { int index; if ((p->flags & PF_VCPU) && (irq_count() - hardirq_offset == 0)) { account_guest_time(p, cputime); return; } if (hardirq_count() - hardirq_offset) index = CPUTIME_IRQ; else if (in_serving_softirq()) index = CPUTIME_SOFTIRQ; else index = CPUTIME_SYSTEM; account_system_index_time(p, cputime, index); }

Contributors

PersonTokensPropCommitsCommitProp
Frédéric Weisbecker83100.00%3100.00%
Total83100.00%3100.00%

/* * Account for involuntary wait time. * @cputime: the cpu time spent in involuntary wait */
void account_steal_time(u64 cputime) { u64 *cpustat = kcpustat_this_cpu->cpustat; cpustat[CPUTIME_STEAL] += cputime; }

Contributors

PersonTokensPropCommitsCommitProp
Frédéric Weisbecker23100.00%2100.00%
Total23100.00%2100.00%

/* * Account for idle time. * @cputime: the cpu time spent in idle wait */
void account_idle_time(u64 cputime) { u64 *cpustat = kcpustat_this_cpu->cpustat; struct rq *rq = this_rq(); if (atomic_read(&rq->nr_iowait) > 0) cpustat[CPUTIME_IOWAIT] += cputime; else cpustat[CPUTIME_IDLE] += cputime; }

Contributors

PersonTokensPropCommitsCommitProp
Frédéric Weisbecker51100.00%2100.00%
Total51100.00%2100.00%

/* * When a guest is interrupted for a longer amount of time, missed clock * ticks are not redelivered later. Due to that, this function may on * occasion account more time than the calling functions think elapsed. */
static __always_inline u64 steal_account_process_time(u64 maxtime) { #ifdef CONFIG_PARAVIRT if (static_key_false(&paravirt_steal_enabled)) { u64 steal; steal = paravirt_steal_clock(smp_processor_id()); steal -= this_rq()->prev_steal_time; steal = min(steal, maxtime); account_steal_time(steal); this_rq()->prev_steal_time += steal; return steal; } #endif return 0; }

Contributors

PersonTokensPropCommitsCommitProp
Frédéric Weisbecker6187.14%466.67%
Rik Van Riel57.14%116.67%
Wanpeng Li45.71%116.67%
Total70100.00%6100.00%

/* * Account how much elapsed time was spent in steal, irq, or softirq time. */
static inline u64 account_other_time(u64 max) { u64 accounted; /* Shall be converted to a lockdep-enabled lightweight check */ WARN_ON_ONCE(!irqs_disabled()); accounted = steal_account_process_time(max); if (accounted < max) accounted += irqtime_tick_accounted(max - accounted); return accounted; }

Contributors

PersonTokensPropCommitsCommitProp
Rik Van Riel3473.91%125.00%
Frédéric Weisbecker1226.09%375.00%
Total46100.00%4100.00%

#ifdef CONFIG_64BIT
static inline u64 read_sum_exec_runtime(struct task_struct *t) { return t->se.sum_exec_runtime; }

Contributors

PersonTokensPropCommitsCommitProp
Stanislaw Gruszka19100.00%1100.00%
Total19100.00%1100.00%

#else
static u64 read_sum_exec_runtime(struct task_struct *t) { u64 ns; struct rq_flags rf; struct rq *rq; rq = task_rq_lock(t, &rf); ns = t->se.sum_exec_runtime; task_rq_unlock(rq, t, &rf); return ns; }

Contributors

PersonTokensPropCommitsCommitProp
Stanislaw Gruszka54100.00%1100.00%
Total54100.00%1100.00%

#endif /* * Accumulate raw cputime values of dead tasks (sig->[us]time) and live * tasks (sum on group iteration) belonging to @tsk's group. */
void thread_group_cputime(struct task_struct *tsk, struct task_cputime *times) { struct signal_struct *sig = tsk->signal; u64 utime, stime; struct task_struct *t; unsigned int seq, nextseq; unsigned long flags; /* * Update current task runtime to account pending time since last * scheduler action or thread_group_cputime() call. This thread group * might have other running tasks on different CPUs, but updating * their runtime can affect syscall performance, so we skip account * those pending times and rely only on values updated on tick or * other scheduler action. */ if (same_thread_group(current, tsk)) (void) task_sched_runtime(current); rcu_read_lock(); /* Attempt a lockless read on the first round. */ nextseq = 0; do { seq = nextseq; flags = read_seqbegin_or_lock_irqsave(&sig->stats_lock, &seq); times->utime = sig->utime; times->stime = sig->stime; times->sum_exec_runtime = sig->sum_sched_runtime; for_each_thread(tsk, t) { task_cputime(t, &utime, &stime); times->utime += utime; times->stime += stime; times->sum_exec_runtime += read_sum_exec_runtime(t); } /* If lockless access failed, take the lock. */ nextseq = 1; } while (need_seqretry(&sig->stats_lock, seq)); done_seqretry_irqrestore(&sig->stats_lock, seq, flags); rcu_read_unlock(); }

Contributors

PersonTokensPropCommitsCommitProp
Frédéric Weisbecker9450.27%337.50%
Rik Van Riel6836.36%225.00%
Stanislaw Gruszka2010.70%225.00%
Oleg Nesterov52.67%112.50%
Total187100.00%8100.00%

#ifdef CONFIG_IRQ_TIME_ACCOUNTING /* * Account a tick to a process and cpustat * @p: the process that the cpu time gets accounted to * @user_tick: is the tick from userspace * @rq: the pointer to rq * * Tick demultiplexing follows the order * - pending hardirq update * - pending softirq update * - user_time * - idle_time * - system time * - check for guest_time * - else account as system_time * * Check for hardirq is done both for system and user time as there is * no timer going off while we are on hardirq and hence we may never get an * opportunity to update it solely in system time. * p->stime and friends are only updated on system time and not on irq * softirq as those do not count in task exec_runtime any more. */
static void irqtime_account_process_tick(struct task_struct *p, int user_tick, struct rq *rq, int ticks) { u64 other, cputime = TICK_NSEC * ticks; /* * When returning from idle, many ticks can get accounted at * once, including some ticks of steal, irq, and softirq time. * Subtract those ticks from the amount of time accounted to * idle, or potentially user or system time. Due to rounding, * other time can exceed ticks occasionally. */ other = account_other_time(ULONG_MAX); if (other >= cputime) return; cputime -= other; if (this_cpu_ksoftirqd() == p) { /* * ksoftirqd time do not get accounted in cpu_softirq_time. * So, we have to handle it separately here. * Also, p->stime needs to be updated for ksoftirqd. */ account_system_index_time(p, cputime, CPUTIME_SOFTIRQ); } else if (user_tick) { account_user_time(p, cputime); } else if (p == rq->idle) { account_idle_time(cputime); } else if (p->flags & PF_VCPU) { /* System time or guest time */ account_guest_time(p, cputime); } else { account_system_index_time(p, cputime, CPUTIME_SYSTEM); } }

Contributors

PersonTokensPropCommitsCommitProp
Frédéric Weisbecker10883.08%550.00%
Rik Van Riel129.23%110.00%
Thomas Gleixner75.38%110.00%
Wanpeng Li21.54%220.00%
Oleg Nesterov10.77%110.00%
Total130100.00%10100.00%


static void irqtime_account_idle_ticks(int ticks) { struct rq *rq = this_rq(); irqtime_account_process_tick(current, 0, rq, ticks); }

Contributors

PersonTokensPropCommitsCommitProp
Frédéric Weisbecker2692.86%150.00%
Thomas Gleixner27.14%150.00%
Total28100.00%2100.00%

#else /* CONFIG_IRQ_TIME_ACCOUNTING */
static inline void irqtime_account_idle_ticks(int ticks) {}

Contributors

PersonTokensPropCommitsCommitProp
Frédéric Weisbecker9100.00%2100.00%
Total9100.00%2100.00%


static inline void irqtime_account_process_tick(struct task_struct *p, int user_tick, struct rq *rq, int nr_ticks) {}

Contributors

PersonTokensPropCommitsCommitProp
Frédéric Weisbecker1986.36%266.67%
Thomas Gleixner313.64%133.33%
Total22100.00%3100.00%

#endif /* CONFIG_IRQ_TIME_ACCOUNTING */ /* * Use precise platform statistics if available: */ #ifdef CONFIG_VIRT_CPU_ACCOUNTING #ifndef __ARCH_HAS_VTIME_TASK_SWITCH
void vtime_common_task_switch(struct task_struct *prev) { if (is_idle_task(prev)) vtime_account_idle(prev); else vtime_account_system(prev); vtime_flush(prev); arch_vtime_task_switch(prev); }

Contributors

PersonTokensPropCommitsCommitProp
Frédéric Weisbecker38100.00%4100.00%
Total38100.00%4100.00%

#endif #endif /* CONFIG_VIRT_CPU_ACCOUNTING */ #ifdef CONFIG_VIRT_CPU_ACCOUNTING_NATIVE /* * Archs that account the whole time spent in the idle task * (outside irq) as idle time can rely on this and just implement * vtime_account_system() and vtime_account_idle(). Archs that * have other meaning of the idle time (s390 only includes the * time spent by the CPU when it's in low power mode) must override * vtime_account(). */ #ifndef __ARCH_HAS_VTIME_ACCOUNT
void vtime_account_irq_enter(struct task_struct *tsk) { if (!in_interrupt() && is_idle_task(tsk)) vtime_account_idle(tsk); else vtime_account_system(tsk); }

Contributors

PersonTokensPropCommitsCommitProp
Frédéric Weisbecker32100.00%3100.00%
Total32100.00%3100.00%

EXPORT_SYMBOL_GPL(vtime_account_irq_enter); #endif /* __ARCH_HAS_VTIME_ACCOUNT */
void task_cputime_adjusted(struct task_struct *p, u64 *ut, u64 *st) { *ut = p->utime; *st = p->stime; }

Contributors

PersonTokensPropCommitsCommitProp
Frédéric Weisbecker32100.00%3100.00%
Total32100.00%3100.00%

EXPORT_SYMBOL_GPL(task_cputime_adjusted);
void thread_group_cputime_adjusted(struct task_struct *p, u64 *ut, u64 *st) { struct task_cputime cputime; thread_group_cputime(p, &cputime); *ut = cputime.utime; *st = cputime.stime; }

Contributors

PersonTokensPropCommitsCommitProp
Frédéric Weisbecker44100.00%3100.00%
Total44100.00%3100.00%

#else /* !CONFIG_VIRT_CPU_ACCOUNTING_NATIVE */ /* * Account a single tick of cpu time. * @p: the process that the cpu time gets accounted to * @user_tick: indicates if the tick is a user or a system tick */
void account_process_tick(struct task_struct *p, int user_tick) { u64 cputime, steal; struct rq *rq = this_rq(); if (vtime_accounting_cpu_enabled()) return; if (sched_clock_irqtime) { irqtime_account_process_tick(p, user_tick, rq, 1); return; } cputime = TICK_NSEC; steal = steal_account_process_time(ULONG_MAX); if (steal >= cputime) return; cputime -= steal; if (user_tick) account_user_time(p, cputime); else if ((p != rq->idle) || (irq_count() != HARDIRQ_OFFSET)) account_system_time(p, HARDIRQ_OFFSET, cputime); else account_idle_time(cputime); }

Contributors

PersonTokensPropCommitsCommitProp
Frédéric Weisbecker9783.62%866.67%
Rik Van Riel1412.07%18.33%
Wanpeng Li32.59%216.67%
Thomas Gleixner21.72%18.33%
Total116100.00%12100.00%

/* * Account multiple ticks of idle time. * @ticks: number of stolen ticks */
void account_idle_ticks(unsigned long ticks) { u64 cputime, steal; if (sched_clock_irqtime) { irqtime_account_idle_ticks(ticks); return; } cputime = ticks * TICK_NSEC; steal = steal_account_process_time(ULONG_MAX); if (steal >= cputime) return; cputime -= steal; account_idle_time(cputime); }

Contributors

PersonTokensPropCommitsCommitProp
Frédéric Weisbecker2952.73%571.43%
Wanpeng Li2647.27%228.57%
Total55100.00%7100.00%

/* * Perform (stime * rtime) / total, but avoid multiplication overflow by * loosing precision when the numbers are big. */
static u64 scale_stime(u64 stime, u64 rtime, u64 total) { u64 scaled; for (;;) { /* Make sure "rtime" is the bigger of stime/rtime */ if (stime > rtime) swap(rtime, stime); /* Make sure 'total' fits in 32 bits */ if (total >> 32) goto drop_precision; /* Does rtime (and thus stime) fit in 32 bits? */ if (!(rtime >> 32)) break; /* Can we just balance rtime/stime rather than dropping bits? */ if (stime >> 31) goto drop_precision; /* We can grow stime and shrink rtime and try to make them both fit */ stime <<= 1; rtime >>= 1; continue; drop_precision: /* We drop from rtime, it has more bits than stime */ rtime >>= 1; total >>= 1; } /* * Make sure gcc understands that this is a 32x32->64 multiply, * followed by a 64/32->64 divide. */ scaled = div_u64((u64) (u32) stime * (u64) (u32) rtime, (u32)total); return scaled; }

Contributors

PersonTokensPropCommitsCommitProp
Stanislaw Gruszka8166.94%233.33%
Frédéric Weisbecker4033.06%466.67%
Total121100.00%6100.00%

/* * Adjust tick based cputime random precision against scheduler runtime * accounting. * * Tick based cputime accounting depend on random scheduling timeslices of a * task to be interrupted or not by the timer. Depending on these * circumstances, the number of these interrupts may be over or * under-optimistic, matching the real user and system cputime with a variable * precision. * * Fix this by scaling these tick based values against the total runtime * accounted by the CFS scheduler. * * This code provides the following guarantees: * * stime + utime == rtime * stime_i+1 >= stime_i, utime_i+1 >= utime_i * * Assuming that rtime_i+1 >= rtime_i. */
static void cputime_adjust(struct task_cputime *curr, struct prev_cputime *prev, u64 *ut, u64 *st) { u64 rtime, stime, utime; unsigned long flags; /* Serialize concurrent callers such that we can honour our guarantees */ raw_spin_lock_irqsave(&prev->lock, flags); rtime = curr->sum_exec_runtime; /* * This is possible under two circumstances: * - rtime isn't monotonic after all (a bug); * - we got reordered by the lock. * * In both cases this acts as a filter such that the rest of the code * can assume it is monotonic regardless of anything else. */ if (prev->stime + prev->utime >= rtime) goto out; stime = curr->stime; utime = curr->utime; /* * If either stime or both stime and utime are 0, assume all runtime is * userspace. Once a task gets some ticks, the monotonicy code at * 'update' will ensure things converge to the observed ratio. */ if (stime == 0) { utime = rtime; goto update; } if (utime == 0) { stime = rtime; goto update; } stime = scale_stime(stime, rtime, stime + utime); update: /* * Make sure stime doesn't go backwards; this preserves monotonicity * for utime because rtime is monotonic. * * utime_i+1 = rtime_i+1 - stime_i * = rtime_i+1 - (rtime_i - utime_i) * = (rtime_i+1 - rtime_i) + utime_i * >= utime_i */ if (stime < prev->stime) stime = prev->stime; utime = rtime - stime; /* * Make sure utime doesn't go backwards; this still preserves * monotonicity for stime, analogous argument to above. */ if (utime < prev->utime) { utime = prev->utime; stime = rtime - utime; } prev->stime = stime; prev->utime = utime; out: *ut = prev->utime; *st = prev->stime; raw_spin_unlock_irqrestore(&prev->lock, flags); }

Contributors

PersonTokensPropCommitsCommitProp
Peter Zijlstra8339.90%220.00%
Frédéric Weisbecker7536.06%550.00%
Stanislaw Gruszka5024.04%330.00%
Total208100.00%10100.00%


void task_cputime_adjusted(struct task_struct *p, u64 *ut, u64 *st) { struct task_cputime cputime = { .sum_exec_runtime = p->se.sum_exec_runtime, }; task_cputime(p, &cputime.utime, &cputime.stime); cputime_adjust(&cputime, &p->prev_cputime, ut, st); }

Contributors

PersonTokensPropCommitsCommitProp
Frédéric Weisbecker63100.00%4100.00%
Total63100.00%4100.00%

EXPORT_SYMBOL_GPL(task_cputime_adjusted);
void thread_group_cputime_adjusted(struct task_struct *p, u64 *ut, u64 *st) { struct task_cputime cputime; thread_group_cputime(p, &cputime); cputime_adjust(&cputime, &p->signal->prev_cputime, ut, st); }

Contributors

PersonTokensPropCommitsCommitProp
Frédéric Weisbecker47100.00%3100.00%
Total47100.00%3100.00%

#endif /* !CONFIG_VIRT_CPU_ACCOUNTING_NATIVE */ #ifdef CONFIG_VIRT_CPU_ACCOUNTING_GEN
static u64 vtime_delta(struct task_struct *tsk) { unsigned long now = READ_ONCE(jiffies); if (time_before(now, (unsigned long)tsk->vtime_snap)) return 0; return jiffies_to_nsecs(now - tsk->vtime_snap); }

Contributors

PersonTokensPropCommitsCommitProp
Frédéric Weisbecker3062.50%375.00%
Rik Van Riel1837.50%125.00%
Total48100.00%4100.00%


static u64 get_vtime_delta(struct task_struct *tsk) { unsigned long now = READ_ONCE(jiffies); u64 delta, other; /* * Unlike tick based timing, vtime based timing never has lost * ticks, and no need for steal time accounting to make up for * lost ticks. Vtime accounts a rounded version of actual * elapsed time. Limit account_other_time to prevent rounding * errors from causing elapsed vtime to go negative. */ delta = jiffies_to_nsecs(now - tsk->vtime_snap); other = account_other_time(delta); WARN_ON_ONCE(tsk->vtime_snap_whence == VTIME_INACTIVE); tsk->vtime_snap = now; return delta - other; }

Contributors

PersonTokensPropCommitsCommitProp
Frédéric Weisbecker3148.44%444.44%
Rik Van Riel2335.94%333.33%
Wanpeng Li1015.62%222.22%
Total64100.00%9100.00%


static void __vtime_account_system(struct task_struct *tsk) { account_system_time(tsk, irq_count(), get_vtime_delta(tsk)); }

Contributors

PersonTokensPropCommitsCommitProp
Frédéric Weisbecker24100.00%4100.00%
Total24100.00%4100.00%


void vtime_account_system(struct task_struct *tsk) { if (!vtime_delta(tsk)) return; write_seqcount_begin(&tsk->vtime_seqcount); __vtime_account_system(tsk); write_seqcount_end(&tsk->vtime_seqcount); }

Contributors

PersonTokensPropCommitsCommitProp
Frédéric Weisbecker3177.50%266.67%
Rik Van Riel922.50%133.33%
Total40100.00%3100.00%


void vtime_account_user(struct task_struct *tsk) { write_seqcount_begin(&tsk->vtime_seqcount); tsk->vtime_snap_whence = VTIME_SYS; if (vtime_delta(tsk)) account_user_time(tsk, get_vtime_delta(tsk)); write_seqcount_end(&tsk->vtime_seqcount); }

Contributors

PersonTokensPropCommitsCommitProp
Frédéric Weisbecker3673.47%583.33%
Rik Van Riel1326.53%116.67%
Total49100.00%6100.00%


void vtime_user_enter(struct task_struct *tsk) { write_seqcount_begin(&tsk->vtime_seqcount); if (vtime_delta(tsk)) __vtime_account_system(tsk); tsk->vtime_snap_whence = VTIME_USER; write_seqcount_end(&tsk->vtime_seqcount); }

Contributors

PersonTokensPropCommitsCommitProp
Frédéric Weisbecker3784.09%480.00%
Rik Van Riel715.91%120.00%
Total44100.00%5100.00%


void vtime_guest_enter(struct task_struct *tsk) { /* * The flags must be updated under the lock with * the vtime_snap flush and update. * That enforces a right ordering and update sequence * synchronization against the reader (task_gtime()) * that can thus safely catch up with a tickless delta. */ write_seqcount_begin(&tsk->vtime_seqcount); if (vtime_delta(tsk)) __vtime_account_system(tsk); current->flags |= PF_VCPU; write_seqcount_end(&tsk->vtime_seqcount); }

Contributors

PersonTokensPropCommitsCommitProp
Frédéric Weisbecker3884.44%480.00%
Rik Van Riel715.56%120.00%
Total45100.00%5100.00%

EXPORT_SYMBOL_GPL(vtime_guest_enter);
void vtime_guest_exit(struct task_struct *tsk) { write_seqcount_begin(&tsk->vtime_seqcount); __vtime_account_system(tsk); current->flags &= ~PF_VCPU; write_seqcount_end(&tsk->vtime_seqcount); }

Contributors

PersonTokensPropCommitsCommitProp
Frédéric Weisbecker38100.00%3100.00%
Total38100.00%3100.00%

EXPORT_SYMBOL_GPL(vtime_guest_exit);
void vtime_account_idle(struct task_struct *tsk) { account_idle_time(get_vtime_delta(tsk)); }

Contributors

PersonTokensPropCommitsCommitProp
Frédéric Weisbecker18100.00%3100.00%
Total18100.00%3100.00%


void arch_vtime_task_switch(struct task_struct *prev) { write_seqcount_begin(&prev->vtime_seqcount); prev->vtime_snap_whence = VTIME_INACTIVE; write_seqcount_end(&prev->vtime_seqcount); write_seqcount_begin(&current->vtime_seqcount); current->vtime_snap_whence = VTIME_SYS; current->vtime_snap = jiffies; write_seqcount_end(&current->vtime_seqcount); }

Contributors

PersonTokensPropCommitsCommitProp
Frédéric Weisbecker5998.33%375.00%
Rik Van Riel11.67%125.00%
Total60100.00%4100.00%


void vtime_init_idle(struct task_struct *t, int cpu) { unsigned long flags; local_irq_save(flags); write_seqcount_begin(&t->vtime_seqcount); t->vtime_snap_whence = VTIME_SYS; t->vtime_snap = jiffies; write_seqcount_end(&t->vtime_seqcount); local_irq_restore(flags); }

Contributors

PersonTokensPropCommitsCommitProp
Frédéric Weisbecker5498.18%375.00%
Rik Van Riel11.82%125.00%
Total55100.00%4100.00%


u64 task_gtime(struct task_struct *t) { unsigned int seq; u64 gtime; if (!vtime_accounting_enabled()) return t->gtime; do { seq = read_seqcount_begin(&t->vtime_seqcount); gtime = t->gtime; if (t->vtime_snap_whence == VTIME_SYS && t->flags & PF_VCPU) gtime += vtime_delta(t); } while (read_seqcount_retry(&t->vtime_seqcount, seq)); return gtime; }

Contributors

PersonTokensPropCommitsCommitProp
Frédéric Weisbecker7488.10%583.33%
Hiroshi Shimamoto1011.90%116.67%
Total84100.00%6100.00%

/* * Fetch cputime raw values from fields of task_struct and * add up the pending nohz execution time since the last * cputime snapshot. */
void task_cputime(struct task_struct *t, u64 *utime, u64 *stime) { u64 delta; unsigned int seq; if (!vtime_accounting_enabled()) { *utime = t->utime; *stime = t->stime; return; } do { seq = read_seqcount_begin(&t->vtime_seqcount); *utime = t->utime; *stime = t->stime; /* Task is sleeping, nothing to add */ if (t->vtime_snap_whence == VTIME_INACTIVE || is_idle_task(t)) continue; delta = vtime_delta(t); /* * Task runs either in user or kernel space, add pending nohz time to * the right place. */ if (t->vtime_snap_whence == VTIME_USER || t->flags & PF_VCPU) *utime += delta; else if (t->vtime_snap_whence == VTIME_SYS) *stime += delta; } while (read_seqcount_retry(&t->vtime_seqcount, seq)); }

Contributors

PersonTokensPropCommitsCommitProp
Frédéric Weisbecker10875.00%480.00%
Stanislaw Gruszka3625.00%120.00%
Total144100.00%5100.00%

#endif /* CONFIG_VIRT_CPU_ACCOUNTING_GEN */

Overall Contributors

PersonTokensPropCommitsCommitProp
Frédéric Weisbecker212075.80%4361.43%
Stanislaw Gruszka2729.72%912.86%
Rik Van Riel2268.08%57.14%
Peter Zijlstra843.00%22.86%
Wanpeng Li461.64%34.29%
Thomas Gleixner140.50%11.43%
Andrey Smetanin100.36%11.43%
Hiroshi Shimamoto100.36%11.43%
Oleg Nesterov60.21%11.43%
Christoph Lameter30.11%11.43%
Li Zefan30.11%11.43%
Dongsheng Yang20.07%11.43%
Ingo Molnar10.04%11.43%
Total2797100.00%70100.00%
Directory: kernel/sched
Information contained on this website is for historical information purposes only and does not indicate or represent copyright ownership.
Created with cregit.