IT数码 购物 网址 头条 软件 日历 阅读 图书馆
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
图片批量下载器
↓批量下载图片,美女图库↓
图片自动播放器
↓图片自动播放器↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁
 
   -> 嵌入式 -> 中断处理时间统计 cpu_irqtime -> 正文阅读

[嵌入式]中断处理时间统计 cpu_irqtime

1. 定义

cpu_irqtime为per-cpu变量,定义在kernel/sched/cputime.c文件中

DEFINE_PER_CPU(struct irqtime, cpu_irqtime);

本功能的开启由config?CONFIG_IRQ_TIME_ACCOUNTING控制,默认是开启状态

config IRQ_TIME_ACCOUNTING
        bool "Fine granularity task level IRQ time accounting"
        depends on HAVE_IRQ_TIME_ACCOUNTING && !VIRT_CPU_ACCOUNTING_NATIVE
        help
          Select this option to enable fine granularity task irq time
          accounting. This is done by reading a timestamp on each
          transitions between softirq and hardirq state, so there can be a
          small performance impact.

          If in doubt, say N here.

2. 调用

在系统中有三处调用位置:

2.1 irqtime_account_irq

首先看其实现:

/*
 * Called after incrementing preempt_count on {soft,}irq_enter
 * and before decrementing preempt_count on {soft,}irq_exit.
 */
void irqtime_account_irq(struct task_struct *curr, unsigned int offset)
{
        struct irqtime *irqtime = this_cpu_ptr(&cpu_irqtime);
        unsigned int pc;
        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;
        pc = irq_count() - offset; //减去本次正常处理的中断,将剩余中断赋值给临时变量pc

        /*
         * 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 (pc & HARDIRQ_MASK) //若还有硬中断未处理完
                irqtime_account_delta(irqtime, delta, CPUTIME_IRQ);
        else if ((pc & SOFTIRQ_OFFSET) && curr != this_cpu_ksoftirqd()) //仍有软中断,并且当前task不是ksoftirqd
                irqtime_account_delta(irqtime, delta, CPUTIME_SOFTIRQ);
}

判断sched_clock_irqtime变量值,该变量的定义含义是:是否使能了clock_irqtime功能。

该变量值定义如下,并通过以下两个接口调用。

static int sched_clock_irqtime;

void enable_sched_clock_irqtime(void)
{
        sched_clock_irqtime = 1;
}

void disable_sched_clock_irqtime(void)
{
        sched_clock_irqtime = 0;
}

再继续向下,记录本cpu的本次的中断处理时长,赋值给delta,并更新irq_start_time。

根据中断标志位判断是否还有剩余的HARDIRQ or SOFTIRQ未处理(不在ksoftirqd调用),若有就更新如下变量

a. per-cpu kernel_cpustat->cpustat 区分中断类型

b. irqtime->total

c. irqtime->tick_delta

struct kernel_cpustat {
        u64 cpustat[NR_STATS];
};

DECLARE_PER_CPU(struct kernel_cpustat, kernel_cpustat);

#define kcpustat_this_cpu this_cpu_ptr(&kernel_cpustat)

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; //加入对应类型中断的cpustat中
        irqtime->total += delta; //加入total
        irqtime->tick_delta += delta; //加入tick_delta
        u64_stats_update_end(&irqtime->sync);
}

2.1.1 enable_sched_clock_irqtime

enable_sched_clock_irqtime的调用,对于arm/arm64架构来说只有一处:

kernel/time/sched_clock.c

static int irqtime = -1;

core_param(irqtime, irqtime, int, 0400);

void __init
sched_clock_register(u64 (*read)(void), int bits, unsigned long rate)
{
        ...
        /* Enable IRQ time accounting if we have a fast enough sched_clock() */
        if (irqtime > 0 || (irqtime == -1 && rate >= 1000000))
                enable_sched_clock_irqtime();
        ...
}

调用需要两个判断条件:

a. irqtime>0

????????irqtime支持两种配置方式:

? ? ? ? cmdline

????????echo -n ${value} > /sys/module/${modulename}/parameters/${parm}

b.?(irqtime == -1 && rate >= 1000000)

? ? ? ? sched_clock_register在timer注册时调用,系统中最大rate为arch_timer为26M,>1000000。

2.1.2 disable_sched_clock_irqtime

在arm/arm64系统中无调用位置,忽略。

2.1.3 irq_count

统计此时处理的中断的数量。包括nmi中断,硬中断,软中断。

#define irq_count() (nmi_count() | hardirq_count() | softirq_count())

2.2 irqtime_tick_accounted

2.3 irq_time_read

  嵌入式 最新文章
基于高精度单片机开发红外测温仪方案
89C51单片机与DAC0832
基于51单片机宠物自动投料喂食器控制系统仿
《痞子衡嵌入式半月刊》 第 68 期
多思计组实验实验七 简单模型机实验
CSC7720
启明智显分享| ESP32学习笔记参考--PWM(脉冲
STM32初探
STM32 总结
【STM32】CubeMX例程四---定时器中断(附工
上一篇文章      下一篇文章      查看所有文章
加:2021-07-31 16:48:39  更:2021-07-31 16:49:19 
 
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁

360图书馆 购物 三丰科技 阅读网 日历 万年历 2025年1日历 -2025/1/28 11:42:38-

图片自动播放器
↓图片自动播放器↓
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
图片批量下载器
↓批量下载图片,美女图库↓
  网站联系: qq:121756557 email:121756557@qq.com  IT数码