转自:http://blog.csdn.net/lkkey80/article/details/45190095

版权声明:博文地址 http://blog.csdn.net/lkkey80?viewmode=list 转载请注明出处

在Linux内核代码里当需要限制printk打印频率时会用到__ratelimit或printk_ratelimit(封装了__ratelimit)。

  1. /*
  2. * printk rate limiting, lifted from the networking subsystem.
  3. *
  4. * This enforces a rate limit: not more than 10 kernel messages
  5. * every 5s to make a denial-of-service attack impossible.
  6. */
  7. DEFINE_RATELIMIT_STATE(printk_ratelimit_state, 5 * HZ, 10);
  8. int printk_ratelimit(void)
  9. {
  10. return __ratelimit(&printk_ratelimit_state);
  11. }
  12. EXPORT_SYMBOL(printk_ratelimit);

/var/log/messages里一个可能的输出如下:

kernel: end_request: I/O error, dev sdr, sector 2048                            
kernel: end_request: I/O error, dev sdr, sector 2048                            
kernel: end_request: I/O error, dev sdr, sector 2104                            
kernel: end_request: I/O error, dev sdr, sector 2048                            
kernel: end_request: I/O error, dev sdr, sector 2048                            
kernel: __ratelimit: 125 callbacks suppressed
                                  
kernel: sd 21:0:0:0: [sdr] Unhandled error code                                 
kernel: sd 21:0:0:0: [sdr] Result: hostbyte=DID_BAD_TARGET driverbyte=DRIVER_OK 
kernel: sd 21:0:0:0: [sdr] CDB: Read(10): 28 00 00 00 00 00 00 00 20 00         
kernel: __ratelimit: 125 callbacks suppressed                                   
kernel: __ratelimit: 128 callbacks suppressed                                   
kernel: Buffer I/O error on device sdr, logical block 0                         
kernel: Buffer I/O error on device sdr, logical block 1                         
kernel: Buffer I/O error on device sdr, logical block 2                         
kernel: Buffer I/O error on device sdr, logical block 3

printk_ratelimit默认允许在5s内最多打印10条消息出来。可通过 /proc/sys/kernel/printk_ratelimit
(多长时间)和 /proc/sys/kernel/printk_ratelimit_burst
(在printk_ratelimit时间段内最多允许的消息数量)。当然如果内核代码里没有调用printk_ratelimit就不受这两个值的限制。

内核块设备代码中的一个例子如下:

  1. bool blk_update_request(struct request *req, int error, unsigned int nr_bytes)
  2. {
  3. int total_bytes, bio_nbytes, next_idx = 0;
  4. struct bio *bio;
  5. if (!req->bio)
  6. return false;
  7. trace_block_rq_complete(req->q, req);
  8. /*
  9. * For fs requests, rq is just carrier of independent bio's
  10. * and each partial completion should be handled separately.
  11. * Reset per-request error on each partial completion.
  12. *
  13. * TODO: tj: This is too subtle.  It would be better to let
  14. * low level drivers do what they see fit.
  15. */
  16. if (req->cmd_type == REQ_TYPE_FS)
  17. req->errors = 0;
  18. if (error && req->cmd_type == REQ_TYPE_FS &&
  19. !(req->cmd_flags & REQ_QUIET)) {
  20. char *error_type;
  21. switch (error) {
  22. case -ENOLINK:
  23. error_type = "recoverable transport";
  24. break;
  25. case -EREMOTEIO:
  26. error_type = "critical target";
  27. break;
  28. case -EBADE:
  29. error_type = "critical nexus";
  30. break;
  31. case -EIO:
  32. default:
  33. error_type = "I/O";
  34. break;
  35. }
  36. printk_ratelimited(KERN_ERR "end_request: %s error, dev %s, "
  37. "sector %llu\n", error_type, req->rq_disk ?
  38. req->rq_disk->disk_name : "?",
  39. (unsigned long long)blk_rq_pos(req));
  40. }
  41. blk_account_io_completion(req, nr_bytes);
  42. ...
  43. }

printk_ratelimited是一个宏,并直接调用了 __ratelimit

    1. #define printk_ratelimited(fmt, ...)  ({        \
    2. static struct ratelimit_state _rs = {       \
    3. .interval = DEFAULT_RATELIMIT_INTERVAL, \
    4. .burst = DEFAULT_RATELIMIT_BURST,       \
    5. };                                              \
    6. \
    7. if (!__ratelimit(&_rs))                         \
    8. printk(fmt, ##__VA_ARGS__);     \
    9. })

限制printk打印频率函数printk_ratelimit【转】的更多相关文章

  1. 更改printk打印级别

    1.查看当前控制台的打印级别 cat /proc/sys/kernel/printk 4    4    1    7 其中第一个"4"表示内核打印函数printk的打印级别,只有 ...

  2. 内核printk打印等级

    为了确认内核打印等级以及prink 参数对打印的分级,在led驱动初始化代码[以及exit出口]加入如下代码. 每次insmod .rmmod led模块时,根据打印等级的设置,得到不同的打印结果: ...

  3. 更改printk打印级别【转】

    本文转载自:http://blog.csdn.net/weed_hz/article/details/8949140 1.查看当前控制台的打印级别 cat /proc/sys/kernel/print ...

  4. 终端下更改printk打印级别

    如何去更改printk的打印级别? 1.查看当前控制台的打印级别 # cat /proc/sys/kernel/printk 该文件有4个数字值,它们根据日志记录消息的重要性,定义将其发送到何处,上面 ...

  5. printk打印级别

    默认级别 # cat /proc/sys/kernel/printk 4 4 1 7 分别是:控制台日志级别.默认的消息日志级别.最低的控制台日志级别和默认的控制台日志级别 举例 # echo 0 & ...

  6. 二分查找方法和printk打印级别

    人生就是一个茶几,上面摆满了杯具.内核也是一个大茶几,不过它上面的杯具是一个个的bug.确定bug什么时候被引入是一个很关键的步骤,在这个定位bug的过程中,不论有意或无意,都会很自然地用到二分查找的 ...

  7. linux内核打印数据到串口控制台,printk数据不打印问题

    linux内核打印数据到串口控制台问题 原文来源:http://i.cnblogs.com/EditPosts.aspx?opt=1 1.查看当前控制台的打印级别 cat /proc/sys/kern ...

  8. ubuntu14.04 printk()默认打印的位置

    tail /var/log/syslog 即可显示printk打印的信息

  9. [linux-内核][转]内核日志及printk结构浅析

    这段时间复习了一下内核调试系统,注意看了一下printk的实现以及内核日志的相关知识,这里做一下总结. 1.问题的引出: 做DPDK项目时,调试rte_kni.ko时,发现printk并不会向我们想想 ...

随机推荐

  1. BZOJ4754 & 洛谷4323 & LOJ2072:[JSOI2016]独特的树叶——题解

    https://www.lydsy.com/JudgeOnline/problem.php?id=4754 https://www.luogu.org/problemnew/show/P4323 ht ...

  2. BZOJ1043:[HAOI2008]下落的圆盘——题解(配图片)

    http://www.lydsy.com/JudgeOnline/problem.php?id=1043 Description 有n个圆盘从天而降,后面落下的可以盖住前面的.求最后形成的封闭区域的周 ...

  3. ContestHunter暑假欢乐赛 SRM 09(TJM大傻逼选手再创佳绩)

    T1 f[i]为前i页最少被撕几页,用二分转移就行了,答案为ans=min(f[i]+(n-i)); 不知道为什么写挂了嗯 二分的l初始应该是0 T2 数位DP f[i][1/0][1/0][1/0] ...

  4. CF993E Nikita and Order Statistics

    小于x的赋值为1,否则为0 区间等于k的个数 求0~n连续的n+1个k? N<=1e5? FFT! 考虑卷积建模:用下标相加实现转移到位,数值相乘类比乘法原理! 法一: 分治,然后FFT没了 法 ...

  5. JQuery仿淘宝滚动加载图片

    用 JQuery 制作随着显示页面的滚动条的滚动动态加载图片,适用于图片太多的页面,在访问网页时,可以先只加载第一屏要显示的图片,当用户进行向下滚动查看页面的时候,动态去加载这些图片,好处是减少页面第 ...

  6. JavaScript转换与解析JSON的方法

    在JavaScript中将JSON的字符串解析成JSON数据格式,一般有两种方式: 一种为使用eval()函数. 使用Function对象来进行返回解析. 使用eval函数来解析,jquery的eac ...

  7. Spring源码解析-Advice中的Adapter模式

    在spring中与通知相关的类有: 以Advice结尾的通知接口    MethodBeforeAdvice    AfterReturningAdvice   ThrowsAdvice 以Inter ...

  8. nfs挂载权限问题

    问题: 服务器A:192.168.10.230 服务器B:192.168.10.231 由于服务器A空间不足,打算将服务器A产生的数据库日志挂载到服务器B上,刚开始设定的anonuid和anongid ...

  9. HDU 5646

    DZY Loves Partition Time Limit: 4000/2000 MS (Java/Others)    Memory Limit: 262144/262144 K (Java/Ot ...

  10. 使用springcloud的feign调用服务时出现的错误:关于实体转换成json错误的介绍

    http://blog.csdn.net/java_huashan/article/details/46428971 原因:实体中没有添加无参的构造函数 fastjson的解释: http://www ...