Redis源码分析之内存检测memtest

redis的内存检测会和机器的CPU位数有关,32位或64位会影响后面的一些宏定义参数。首先给出memtest中的API:

void memtest_progress_start(char *title, int pass) /* 内存检测加载开始,输出开始的一些图线显示 */
void memtest_progress_end(void) /* progress bar加载完再次清屏操作 */
void memtest_progress_step(size_t curr, size_t size, char c) /* progress填充自己设置的字符串 */
void memtest_addressing(unsigned long *l, size_t bytes) /* 地址检测方法 */
void memtest_fill_random(unsigned long *l, size_t bytes) /* 随机填充内存 */
void memtest_fill_value(unsigned long *l, size_t bytes, unsigned long v1, unsigned long v2, char sym) /* 像上面的方法,只不过这是特定2种值的填充v1,v2 */
void memtest_compare(unsigned long *l, size_t bytes) /* 内存比较方法 */
void memtest_compare_times(unsigned long *m, size_t bytes, int pass, int times) /* 进行多次内存compare比较操作 */
void memtest_test(size_t megabytes, int passes) /* 整个内存检测类操作的测试方法,passes为目标的循环数 */
void memtest_non_destructive_invert(void *addr, size_t size) /* 将内存地址,进行了按位取反操作,不具有对数据的破坏性 */
void memtest_non_destructive_swap(void *addr, size_t size) /* 将内存地址,2个,2个内部之间做交换,同样不对数据具有破坏性 */
void memtest(size_t megabytes, int passes) /* 开发给整个系统使用的内存检测方法 */

CPU位数的限制:

#if (ULONG_MAX == 4294967295UL)
#define MEMTEST_32BIT
#elif (ULONG_MAX == 18446744073709551615ULL)
#define MEMTEST_64BIT
#else
#error "ULONG_MAX value not supported."
#endif

以其中的接口为例进行说明:

memtest_progress_start:是内存检测开始的接口

内存检测加载开始,输出开始时一些图线显示,一般的test都会有这种显示

void memtest_progress_start(char *title, int pass) {int j;printf("\x1b[H\x1b[2J");    /* Cursor home, clear screen. *//* Fill with dots. */// 填充.for (j = 0; j < ws.ws_col*(ws.ws_row-2); j++) printf(".");printf("Please keep the test running several minutes per GB of memory.\n");printf("Also check http://www.memtest86.com/ and http://pyropus.ca/software/memtester/");printf("\x1b[H\x1b[2K");          /* Cursor home, clear current line.  */// 输出标题printf("%s [%d]\n", title, pass); /* Print title. */progress_printed = 0;// 填满进度条的算法progress_full = ws.ws_col*(ws.ws_row-3);fflush(stdout);
}

memtest_addressing:内测检测的核心算法。

主要看里面的注释部分,其中为什么这样设置,本人也是不理解,请高手留言,在此谢过..........

int memtest_addressing(unsigned long *l, size_t bytes, int interactive) {// 算出地址的lenunsigned long words = bytes/sizeof(unsigned long);unsigned long j, *p;/* Fill */p = l;for (j = 0; j < words; j++) {// 取出p的地址再赋值给p*p = (unsigned long)p;p++;// 用'A'填充progress bar中符合该条件的位置if ((j & 0xffff) == 0 && interactive)memtest_progress_step(j,words*2,'A');}/* Test */p = l;for (j = 0; j < words; j++) {// address比较if (*p != (unsigned long)p) {if (interactive) {printf("\n*** MEMORY ADDRESSING ERROR: %p contains %lu\n",(void*) p, *p);exit(1);}return 1;}p++;if ((j & 0xffff) == 0 && interactive)memtest_progress_step(j+words,words*2,'A');}return 0;
}

还有一种是随机填充memtest_fill_random:在每次写操作的时候,在单页上填满整个字符,这样可以做到最快速的触及所有的页面, 减少了低效率的缓存使用,但是会让分区在转移页面时会比较困难

void memtest_fill_random(unsigned long *l, size_t bytes, int interactive) {// 每次移动的步长unsigned long step = 4096/sizeof(unsigned long);unsigned long words = bytes/sizeof(unsigned long)/2;unsigned long iwords = words/step;  /* words per iteration */unsigned long off, w, *l1, *l2;uint64_t rseed = UINT64_C(0xd13133de9afdb566); /* Just a random seed. */uint64_t rout = 0;assert((bytes & 4095) == 0);for (off = 0; off < step; off++) {l1 = l+off;l2 = l1+words;for (w = 0; w < iwords; w++) {xorshift64star_next();// 填充l1 l2*l1 = *l2 = (unsigned long) rout;l1 += step;l2 += step;// 剩余部分填充Rif ((w & 0xffff) == 0 && interactive)memtest_progress_step(w+iwords*off,words,'R');}}
}

开放给外界的方法:经过passes次的测试才能确定是否通过

int memtest_test(unsigned long *m, size_t bytes, int passes, int interactive) {int pass = 0;int errors = 0;// 运行passes 次while (pass != passes) {pass++;// address testif (interactive) memtest_progress_start("Addressing test",pass);errors += memtest_addressing(m,bytes,interactive);if (interactive) memtest_progress_end();// Random fillif (interactive) memtest_progress_start("Random fill",pass);memtest_fill_random(m,bytes,interactive);if (interactive) memtest_progress_end();errors += memtest_compare_times(m,bytes,pass,4,interactive);// 固定填充if (interactive) memtest_progress_start("Solid fill",pass);memtest_fill_value(m,bytes,0,(unsigned long)-1,'S',interactive);if (interactive) memtest_progress_end();errors += memtest_compare_times(m,bytes,pass,4,interactive);// 使用C字符填充测试if (interactive) memtest_progress_start("Checkerboard fill",pass);memtest_fill_value(m,bytes,ULONG_ONEZERO,ULONG_ZEROONE,'C',interactive);if (interactive) memtest_progress_end();errors += memtest_compare_times(m,bytes,pass,4,interactive);}return errors;
}

总之,内存测试分为4中情况,内存地址测试,其他都为填充测试,分为3种类型的填充Random fill, Solid fill, Checkboard fill 。填充之后再做内存比较,这里的内存比较是在内存内部做前半部分的内存和后半部分的内存比较操作

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/463448.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

Java Collections Framework - Java集合框架List,Map,Set等全面介绍之概要篇

deng 转载于:https://www.cnblogs.com/jacktu/archive/2009/05/15/1457316.html

C语言 数据结构 树和二叉树

树 1、树&#xff1a;是n节点的有限集。树是n(n>0)个节点的有限集。 n0时成为空树。 在任意一颗非空树中&#xff1a;&#xff08;1&#xff09;有且仅有一个称为根的节点&#xff1b;&#xff08;2&#xff09;当n>0时&#xff0c;其余节点可分为m(m>0)个互不相交的…

Oracle开启关闭归档日志

开启归档日志 shutdown immediate; --关闭数据库 startup mount; --打开数据库 alter database archivelog; --开启归档日志 alter database open; --开启数据库 archive log list; --查看归档日志是否开启 关闭归档日志 shutdown immediate; --关闭数据库 startup mount; …

Redis源码分析之anet网络通信的封装

anet是redis对tcp/ip网络中socket api接口的一个全面的封装&#xff0c;针对server/client端。封装的api的接口如下&#xff0c;注释了主要的接口&#xff1a; // tcp连接 int anetTcpConnect(char *err, char *addr, int port); // 非阻塞连接 int anetTcpNonBlockConnect(ch…

使用AJAX Toolkit创建新闻列表

我们很多站点上面都需要显示新闻列表&#xff0c;由标题和正文组成的。一般客户都希望实现这样的效果&#xff1a; 开始的时候只是显示标题&#xff0c;当点击标题的时候&#xff0c;再展开正文。再点击&#xff0c;又可缩回去。 这是典型的AJAX效果&#xff0c;或者说以前你也…

C语言 嵌入式 面试小知识点(一)

sizeof是C/C中的一个操作符&#xff08;operator&#xff09;&#xff0c;简单的说其作用就是返回一个对象或者类型所占的内存字节数。 与strlen的区别&#xff1a; 一、sizeof 是运算符&#xff0c;确切的说是一个编译时运算符&#xff0c;参数可以是数组、指针、类型、对象…

将数据渲染到页面的几种方式

将数据渲染到页面的几种方式&#xff1a; 1.字符串拼接&#xff1b; 2.dom回流 3.文档碎片&#xff08;文档碎片节点&#xff1a;documentFragment&#xff09; 4.模板 &#xff08;下章会详细介绍模板&#xff09;转载于:https://www.cnblogs.com/shangjun6/p/1039740…

论证是一门学问

本文的标题借用了安东尼.韦斯顿&#xff08;Anthony Weston&#xff09;的《论证是一门学问》一书的标题&#xff0c;向安东尼老爷子致敬的同时&#xff0c;也希望更多人能够真正了解“什么是论证”。 争论与论证从来都不是新鲜事物&#xff0c;作为软件行业的科技工作者&…

[翻译]SQL Server 工作集消息

Q&#xff1a;我发现有指向工作集&#xff08;SQL Server保留内存区域&#xff09;被分页出来相关的消息&#xff1a; 重要部分的 SQL 服务器进程内存已被分页。这可能导致性能下降。持续时间: 0 秒。 工作集 (KB)&#xff1a; 2484&#xff0c;已提交 (KB)&#xff1a; 48036&…

Redis源码分析之工具类util

在redis源码中的辅助工具类中&#xff0c;主要包括大小端转换、SHA算法以及util.h中对应的算法。 大小端转换&#xff1a; LittleEndian&#xff1a;低位字节数据存放于低地址&#xff0c;高位字节数据存放于高地址。 BigEndian&#xff1a;低位字节数据存放于高地址&#x…

Linux下如何安装软件

一、解析Linux应用软件安装包通常Linux应用软件的安装包有三种&#xff1a;1&#xff09; tar包&#xff0c;如software-1.2.3-1.tar.gz。它是使用UNIX系统的打包工具tar打包的。2&#xff09; rpm包&#xff0c;如software-1.2.3-1.i386.rpm。它是RedHat Linux提供的一种包封装…

Python深浅拷贝辨析

1 import copy2 3 list1 [11, 22, [33, 44]]4 list2 list15 list3 list1[:]6 list4 copy.copy(list1)7 list5 copy.deepcopy(list1)8 9 list1[0] 0 # 对列表的首层做增删改查操作 10 print("list1:",id(list1),list1) # list1: 1455502266696 [0, 22, […

生活规则

1.朋友请你吃饭&#xff0c;不要觉得理所当然&#xff0c;请礼尚往来&#xff0c;否则你的名声会越来越臭。 2.给自己定目标&#xff0c;一年&#xff0c;两年&#xff0c;五年&#xff0c;也许你出生不如别人好&#xff0c;通过努力&#xff0c;往往可以改变70%的命运。破罐子…

[AX]AX2012 SSRS报表使用Report Data Method

在AX2012的SSRS报表中可以使用c#或者Visual basic .net编写report data method来获取和操作数据&#xff0c;由report data method返回的数据可以用在报表的表达式中&#xff0c;也可以用作dataset的数据源。 使用Report data method首先需要创建AX model工程&#xff0c;在工程…

HIVE和HBASE区别

转载&#xff1a;https://www.cnblogs.com/justinzhang/p/4273470.html 1. 两者分别是什么&#xff1f; Apache Hive是一个构建在Hadoop基础设施之上的数据仓库。通过Hive可以使用HQL语言查询存放在HDFS上的数据。HQL是一种类SQL语言&#xff0c;这种语言最终被转化为Map/Re…

php调试

今天在使用php 的session 的时候&#xff0c;出现了以前就遇见但是又解决不了的问题&#xff0c;在页面上出现如下提示&#xff1a; Warning: session_start() [function.session-start]: Cannot send session cache limiter - headers already sent (output started at E:\php…

C 结构体

C 结构体C 数组允许定义可存储相同类型数据项的变量&#xff0c;结构是 C 编程中另一种用户自定义的可用的数据类型&#xff0c;它允许您存储不同类型的数据项。结构用于表示一条记录&#xff0c;假设您想要跟踪图书馆中书本的动态&#xff0c;您可能需要跟踪每本书的下列属性&…

lightoj1259 线性筛的另一种写法 v变成bool标记数组

也是用线性筛&#xff0c;但是v用int会爆&#xff0c;所以这个线性筛用的是另外一种写法 #include<cstdio> #include<cmath> #include<queue> #include<vector> #include<cstring> #include<iostream> #include<algorithm> using na…

Redis基数统计之HyperLogLog小内存大用处

转载&#xff1a;https://blog.csdn.net/azhegps/article/details/71158952 我们一直都知道&#xff0c;redis几大常用数据结构&#xff0c;字符串、散列、列表、集合、有序集合。其实后来Redis做了很多补充&#xff0c;其中之一就是HyperLogLog&#xff0c;另外的还有GEO&…

matlab中的qr函数

转自&#xff1a;https://blog.csdn.net/qq278672818/article/details/62038630 实数矩阵A的QR分解是把A分解为A QR这里的Q是正交矩阵&#xff08;意味着QTQ I&#xff09;而R是上三角矩阵。类似的&#xff0c;我们可以定义A的QL, RQ和LQ分解。更一般的说&#xff0c;我们可以…