Linux内核源码详解——命令篇之iostat

转自:http://www.cnblogs.com/york-hust/p/4846497.html

本文主要分析了Linux的iostat命令的源码,iostat的主要功能见博客:性能测试进阶指南——基础篇之磁盘IO

iostat源码共563行,应该算是Linux系统命令代码比较少的了。源代码中主要涉及到如下几个Linux的内核文件:

  • 1、/proc/diskstats——该文件是内核2.6以上的系统中的,记录了从Linux系统启动之后,所有磁盘的相关信息,该文件中每个参数代表的意义可以自行google或者baidu,或者见博客:/proc/diskstats参数含义
  • 2、/proc/partitions——partitions是2.4版本的系统中的,其含义基本与diskstats一样。
  • 3、/proc/stat——stat记录了自系统启动之后,CPU的信息,具体含义可以参考博客:性能测试进阶指南——基础篇一(系统资源的讲解)
  • 4、/proc/cpuinfo——iostat主要是从该内核文件中获取cpu的核心数的。

iostat源码解析

第一步,从/proc/cpuinfo中获取系统的cpu核心数,通过计算该文件中processor出现的次数便可以得到cpu的核心数;

第二步,通过判断文件/proc/diskstats和/proc/partitions是否存在,从而判断linux的内核是2.4版本还是2.6版本:如果/proc/diskstats文件存在,则为2.6版本;否则判断/proc/partitions是否存在,若存在,则为2.4版本;

第三部,分析iostat命令输入的参数,每个参数的功能可以在上一篇博客中找到:性能测试进阶指南——基础篇之磁盘IO

第四步,初始化,获取磁盘名称。以内核2.6为例,读取文件/proc/diskstats

  1. 104 0 cciss/c0d0 49787 19805 1597284 159946 20172754 28596938 390157514 1583532 0 1352168 1737502

第一个参数104和第二个参数0分别代表了major和minor,major是8的倍数,minor是16的倍数,只要同时符合这两个的条件,其对应的第三个参数cciss/c0d0便是所需要获取的磁盘名称;

第五步,进入主循环:

(1) 获取/proc/diskstats中每个磁盘的数据,例如:

  1. 104 0 cciss/c0d0 49787 19805 1597284 159946 20172754 28596938 390157514 1583532 0 1352168 1737502

每个参数对应的值为

  1. 104——major
  2. 0——minor
  3. 49787——rd_ios
  4. 19805——rd_merges
  5. 1597284——rd_sectors
  6. 159946——rd_ticks
  7. 20172754——wr_ios
  8. 28596938——wr_merges
  9. 390157514——wr_sectors
  10. 1583532——wr_ticks
  11. 1352168——ticks
  12. 1737502——aveq

(2) 获取/proc/stat中的数据,计算cpu的平均时间:分别获取cpu的user时间,nice时间,system时间,idle时间,iowait时间。计算中将nice时间并入user时间,将irq时间和softirq时间并入system时间。此处只计算cpu的平均和状态,不计算每隔核单独的状态。

(3)计算deltams时间,其中HZ是Linux的系统频率。

  1. deltams = 1000.0 * ((new_cpu.user + new_cpu.system + new_cpu.idle + new_cpu.iowait) - (old_cpu.user + old_cpu.system + old_cpu.idle + old_cpu.iowait)) / ncpu / HZ; `

(4)计算IO

  1. blkio.rd_ios = new_blkio[p].rd_ios - old_blkio[p].rd_ios;
  2. blkio.rd_merges = new_blkio[p].rd_merges - old_blkio[p].rd_merges;
  3. blkio.rd_sectors = new_blkio[p].rd_sectors - old_blkio[p].rd_sectors;
  4. blkio.rd_ticks = new_blkio[p].rd_ticks - old_blkio[p].rd_ticks;
  5. blkio.wr_ios = new_blkio[p].wr_ios - old_blkio[p].wr_ios;
  6. blkio.wr_merges = new_blkio[p].wr_merges - old_blkio[p].wr_merges;
  7. blkio.wr_sectors = new_blkio[p].wr_sectors - old_blkio[p].wr_sectors;
  8. blkio.wr_ticks = new_blkio[p].wr_ticks - old_blkio[p].wr_ticks;
  9. blkio.ticks = new_blkio[p].ticks - old_blkio[p].ticks;
  10. blkio.aveq = new_blkio[p].aveq - old_blkio[p].aveq;
  11. n_ios = blkio.rd_ios + blkio.wr_ios;
  12. n_ticks = blkio.rd_ticks + blkio.wr_ticks;
  13. n_kbytes = (blkio.rd_sectors + blkio.wr_sectors) / 2.0;
  14. queue = blkio.aveq / deltams;
  15. size = n_ios ? n_kbytes / n_ios : 0.0;
  16. wait = n_ios ? n_ticks / n_ios : 0.0;
  17. svc_t = n_ios ? blkio.ticks / n_ios : 0.0;
  18. busy = 100.0 * blkio.ticks / deltams;
  19. if (busy > 100.0) busy = 100.0;

rd_sectors和wr_sectors是扇区数,如果需要换算成KB等单位,需要除以2,1KB=2*512Bytes。512Bytes为1个扇区数。

(5)计算CPU

  1. cpu.user = new_cpu.user - old_cpu.user;
  2. cpu.system = new_cpu.system - old_cpu.system;
  3. cpu.idle = new_cpu.idle - old_cpu.idle;
  4. cpu.iowait = new_cpu.iowait - old_cpu.iowait;
  5. total = (cpu.user + cpu.system + cpu.idle + cpu.iowait) / 100.0;
  6. printf("%3.0f %3.0f ", cpu.user / total, cpu.system / total);
  7. if (kernel == 6) printf("%3.0f ", cpu.iowait / total);
  8. printf("%3.0f", cpu.idle / total);

(6) Save old stats:

  1. old_blkio[p] = new_blkio[p];
  2. old_cpu = new_cpu;

每隔采样时间循环执行第五步。

从源码中可以看出,第一次获取的时候,是没有old stats的,所有的old stats值均为0,即iostat在第一次输出的值为Linux启动之后至当前时间的一个平均状态值,在之后的输出值则为系统当前的实时磁盘I/O信息和CPU信息。

[转]Linux内核源码详解--iostat的更多相关文章

  1. Linux内核源码详解——命令篇之iostat[zz]

    本文主要分析了Linux的iostat命令的源码,iostat的主要功能见博客:性能测试进阶指南——基础篇之磁盘IO iostat源码共563行,应该算是Linux系统命令代码比较少的了.源代码中主要 ...

  2. 【Nginx】epoll及内核源码详解

    内核源码: https://www.nowcoder.com/discuss/26226?type=0&order=0&pos=21&page=1 epoll流程: 首先调用e ...

  3. epoll内核源码详解(转 作者:赛罗·奥特曼 来源:牛客网)

    发现自己发的一篇面经后,很多小伙伴向我索要epoll的内核源码实现,那我就在牛客网发下这源码还有自己总结的流程. 另外 网上很多博客说epoll使用了共享内存,这个是完全错误的 ,可以阅读源码,会发现 ...

  4. Linux内核异常处理体系结构详解(一)【转】

    转自:http://www.techbulo.com/1841.html 2015年11月30日 ⁄ 基础知识 ⁄ 共 6653字 ⁄ 字号 小 中 大 ⁄ Linux内核异常处理体系结构详解(一)已 ...

  5. Linux内核源码分析--内核启动之(6)Image内核启动(do_basic_setup函数)(Linux-3.0 ARMv7)【转】

    原文地址:Linux内核源码分析--内核启动之(6)Image内核启动(do_basic_setup函数)(Linux-3.0 ARMv7) 作者:tekkamanninja 转自:http://bl ...

  6. RocketMQ源码详解 | Broker篇 · 其一:线程模型与接收链路

    概述 在上一节 RocketMQ源码详解 | Producer篇 · 其二:消息组成.发送链路 中,我们终于将消息发送出了 Producer,在短暂的 tcp 握手后,很快它就会进入目的 Broker ...

  7. ubuntu下linux内核源码阅读工具和调试方法总结

    http://blog.chinaunix.net/uid-20940095-id-66148.html 一 linux内核源码阅读工具 windows下当然首选source insight, 但是l ...

  8. RocketMQ源码详解 | Broker篇 · 其五:高可用之主从架构

    概述 对于一个消息中间件来讲,高可用功能是极其重要的,RocketMQ 当然也具有其对应的高可用方案. 在 RocketMQ 中,有主从架构和 Dledger 两种高可用方案: 第一种通过主 Brok ...

  9. linux 内核 RCU机制详解

    RCU(Read-Copy Update)是数据同步的一种方式,在当前的Linux内核中发挥着重要的作用.RCU主要针对的数据对象是链表,目的是提高遍历读取数据的效率,为了达到目的使用RCU机制读取数 ...

随机推荐

  1. python pandas ---Series,DataFrame 创建方法,操作运算操作(赋值,sort,get,del,pop,insert,+,-,*,/)

    pandas 是基于 Numpy 构建的含有更高级数据结构和工具的数据分析包 pandas 也是围绕着 Series 和 DataFrame 两个核心数据结构展开的, 导入如下: from panda ...

  2. kafka 怎么保证的exactly once

    Kafka auto.offset.reset值详解 发表于2017/7/6 11:25:22  1010人阅读 分类: Kafka 昨天在写一个java消费kafka数据的实例,明明设置auto.o ...

  3. Javascript类型转换的规则

    Javascript的变量是松散类型的,它可以存储Javascript支持的任何数据类型,其变量的类型可以在运行时被动态改变.请看示例: 1 2 3 var n = 10; n = "hel ...

  4. JVM相关命题的博客整理及总结

    JVM垃圾回收基础介绍 http://www.jianshu.com/p/57457a351b8a 减少JVM中逃逸对象的使用 http://www.importnew.com/23150.html ...

  5. Android——Android Sutido:[2]导入eclipse项目篇

    说明:直接导入,不用先将eclipse先转出再导入 原文地址:http://jingyan.baidu.com/article/bea41d43698ca3b4c51be68e.html

  6. mosquitto简单应用

    1. 简述 一款实现了消息推送协议 MQTT v3.1 的开源消息代理软件,提供轻量级的,支持可发布/可订阅的的消息推送模式,使设备对设备之间的短消息通信变得简单,比如现在应用广泛的低功耗传感器,手机 ...

  7. mongoDB自动杀执行时间的连接

    ####shell 执行 mongo -u'root' -p'密码' admin mg.js [root@localhost ~]# cat mg.js var currOp = db.current ...

  8. Python - 连接msql数据库

    可能出现的异常:python中出现SyntaxError: Non-UTF-8 code 解决方法 刚开始一直有用idle写代码,今天用了一下PyDev结果发现中文不支持,在网上搜了一下,结果发现解决 ...

  9. 升级R语言

    Linux (RedHat, CentOS)上升级R语言: sudo yum install R Windows上升级R语言: install.packages("installr" ...

  10. Android Custom View系列《圆形菜单一》

    前言 自定义view能够做出很多不同寻常的效果,圆形菜单交互效果不错,目前网上有两个版本,虽然比较庞大,但非常值得研究与学习. radial-menu-widget: https://code.goo ...