1. 文件介绍
cpuinfo.h和cpuinfo.c,同样是文件(不含其他头文件)
包括cpuinfo结构体,属性为cpu信息
get_cpuinfo函数,从linux根目录proc/cpuinfo中获取cpu信息,存储在结构体中
2. cpuinfo结构体属性
int processor
逻辑CPU数
int physical_id
物理CPU数
int siblings
单个CPU逻辑物理核数
int core_id
当前物理核在其所处CPU中的编号,这个编号不一定连续
int cpu_cores
该逻辑核所处CPU的物理核数
可以执行cat proc/cpuinfo查看文件,详情见blog
3. ltrim与rtrim
这两个函数定义在cpuinfo.c中,对指针进行偏移操作,在get_cpuinfo中使用以寻招键值信息
它们都使用isspace库函数
ltrim函数自左向右找到第一个非空字符('\n' '\t' ' ' '\r' ...)
// str[] = "cpu_id = 0"
// 返回str指向非空,str不变
static const char *ltrim(const char *str)
{
while (isspace(*str)) // str是否为空白符,如'\n' '\t' ' ' '\r'
str++;
return str;
}
rtrim略有不同,先找到字符串末尾,然后向前移动直到非空字符出现
// str[] = "cpu_id : 0"
// 返回p = "0"
static void rtrim(char *str)
{
char *p = str + strlen(str) - 1;
while (p >= str && isspace(*p))
*p-- = 0;
}
3.get_cpuinfo函数
函数定义原型:int get_cpuinfo(struct cpuinfo *cpus, int max_cpus)
max_cpus : cpu最大数量,决定cpus数组长度
正常结束则返回实际/proc/cpuinfo中cpu数
函数比较长,只注解关键部分
① 打开文件f = fopen("/proc/cpuinfo", "r");
② 找到每一个cpu的信息,cpu信息包括processor,physical_id,siblings,cpu_id,cpu_cores
可以看到有两层for循环,第一层for循环控制各个cpu,第二层控制cpu详细信息的存储
while (n < max_cpus) {
/*对每一行进行*/
while (fscanf(f, "%m[^:]:%m[^\n]\n", &key, &value) == 2) {
rtrim(key);
if (strcmp(ltrim(key), "processor") == 0)
sscanf(value, "%d", &cpus[n].processor);
else if (strcmp(ltrim(key), "physical id") == 0)
sscanf(value, "%d", &cpus[n].physical_id);
else if (strcmp(ltrim(key), "siblings") == 0)
sscanf(value, "%d", &cpus[n].siblings);
else if (strcmp(ltrim(key), "core id") == 0)
sscanf(value, "%d", &cpus[n].core_id);
else if (strcmp(ltrim(key), "cpu cores") == 0)
sscanf(value, "%d", &cpus[n].cpu_cores);
free(key);
free(value);
}
if (ferror(f))
return -1;
if (feof(f))
break;
n++;
}
对于fscanf(f, "%m[^:]:%m[^\n]\n", &key, &value) == 2
的理解
proc/cpuinfo格式:第一行为 processor : 0
第二个参数是正则表达式匹配
①[^:]
匹配不是:
的字符即processor,将其赋值给key
②下一个:
匹配processor和0之间:
③[^\n]
匹配所有非换行符即0,并赋值给value
对每一行进行读取直到正则表达式不生效,即fscanf返回不是2
跳出循环,n+1,直到n>=max_cpu或文件出错或读写完毕