xref: /OK3568_Linux_fs/kernel/tools/power/cpupower/lib/cpupower.c (revision 4882a59341e53eb6f0b4789bf948001014eff981)
1*4882a593Smuzhiyun // SPDX-License-Identifier: GPL-2.0-only
2*4882a593Smuzhiyun /*
3*4882a593Smuzhiyun  *  (C) 2004-2009  Dominik Brodowski <linux@dominikbrodowski.de>
4*4882a593Smuzhiyun  */
5*4882a593Smuzhiyun 
6*4882a593Smuzhiyun #include <sys/types.h>
7*4882a593Smuzhiyun #include <sys/stat.h>
8*4882a593Smuzhiyun #include <fcntl.h>
9*4882a593Smuzhiyun #include <unistd.h>
10*4882a593Smuzhiyun #include <stdio.h>
11*4882a593Smuzhiyun #include <errno.h>
12*4882a593Smuzhiyun #include <stdlib.h>
13*4882a593Smuzhiyun 
14*4882a593Smuzhiyun #include "cpupower.h"
15*4882a593Smuzhiyun #include "cpupower_intern.h"
16*4882a593Smuzhiyun 
cpupower_read_sysfs(const char * path,char * buf,size_t buflen)17*4882a593Smuzhiyun unsigned int cpupower_read_sysfs(const char *path, char *buf, size_t buflen)
18*4882a593Smuzhiyun {
19*4882a593Smuzhiyun 	int fd;
20*4882a593Smuzhiyun 	ssize_t numread;
21*4882a593Smuzhiyun 
22*4882a593Smuzhiyun 	fd = open(path, O_RDONLY);
23*4882a593Smuzhiyun 	if (fd == -1)
24*4882a593Smuzhiyun 		return 0;
25*4882a593Smuzhiyun 
26*4882a593Smuzhiyun 	numread = read(fd, buf, buflen - 1);
27*4882a593Smuzhiyun 	if (numread < 1) {
28*4882a593Smuzhiyun 		close(fd);
29*4882a593Smuzhiyun 		return 0;
30*4882a593Smuzhiyun 	}
31*4882a593Smuzhiyun 
32*4882a593Smuzhiyun 	buf[numread] = '\0';
33*4882a593Smuzhiyun 	close(fd);
34*4882a593Smuzhiyun 
35*4882a593Smuzhiyun 	return (unsigned int) numread;
36*4882a593Smuzhiyun }
37*4882a593Smuzhiyun 
38*4882a593Smuzhiyun /*
39*4882a593Smuzhiyun  * Detect whether a CPU is online
40*4882a593Smuzhiyun  *
41*4882a593Smuzhiyun  * Returns:
42*4882a593Smuzhiyun  *     1 -> if CPU is online
43*4882a593Smuzhiyun  *     0 -> if CPU is offline
44*4882a593Smuzhiyun  *     negative errno values in error case
45*4882a593Smuzhiyun  */
cpupower_is_cpu_online(unsigned int cpu)46*4882a593Smuzhiyun int cpupower_is_cpu_online(unsigned int cpu)
47*4882a593Smuzhiyun {
48*4882a593Smuzhiyun 	char path[SYSFS_PATH_MAX];
49*4882a593Smuzhiyun 	int fd;
50*4882a593Smuzhiyun 	ssize_t numread;
51*4882a593Smuzhiyun 	unsigned long long value;
52*4882a593Smuzhiyun 	char linebuf[MAX_LINE_LEN];
53*4882a593Smuzhiyun 	char *endp;
54*4882a593Smuzhiyun 	struct stat statbuf;
55*4882a593Smuzhiyun 
56*4882a593Smuzhiyun 	snprintf(path, sizeof(path), PATH_TO_CPU "cpu%u", cpu);
57*4882a593Smuzhiyun 
58*4882a593Smuzhiyun 	if (stat(path, &statbuf) != 0)
59*4882a593Smuzhiyun 		return 0;
60*4882a593Smuzhiyun 
61*4882a593Smuzhiyun 	/*
62*4882a593Smuzhiyun 	 * kernel without CONFIG_HOTPLUG_CPU
63*4882a593Smuzhiyun 	 * -> cpuX directory exists, but not cpuX/online file
64*4882a593Smuzhiyun 	 */
65*4882a593Smuzhiyun 	snprintf(path, sizeof(path), PATH_TO_CPU "cpu%u/online", cpu);
66*4882a593Smuzhiyun 	if (stat(path, &statbuf) != 0)
67*4882a593Smuzhiyun 		return 1;
68*4882a593Smuzhiyun 
69*4882a593Smuzhiyun 	fd = open(path, O_RDONLY);
70*4882a593Smuzhiyun 	if (fd == -1)
71*4882a593Smuzhiyun 		return -errno;
72*4882a593Smuzhiyun 
73*4882a593Smuzhiyun 	numread = read(fd, linebuf, MAX_LINE_LEN - 1);
74*4882a593Smuzhiyun 	if (numread < 1) {
75*4882a593Smuzhiyun 		close(fd);
76*4882a593Smuzhiyun 		return -EIO;
77*4882a593Smuzhiyun 	}
78*4882a593Smuzhiyun 	linebuf[numread] = '\0';
79*4882a593Smuzhiyun 	close(fd);
80*4882a593Smuzhiyun 
81*4882a593Smuzhiyun 	value = strtoull(linebuf, &endp, 0);
82*4882a593Smuzhiyun 	if (value > 1)
83*4882a593Smuzhiyun 		return -EINVAL;
84*4882a593Smuzhiyun 
85*4882a593Smuzhiyun 	return value;
86*4882a593Smuzhiyun }
87*4882a593Smuzhiyun 
88*4882a593Smuzhiyun /* returns -1 on failure, 0 on success */
sysfs_topology_read_file(unsigned int cpu,const char * fname,int * result)89*4882a593Smuzhiyun static int sysfs_topology_read_file(unsigned int cpu, const char *fname, int *result)
90*4882a593Smuzhiyun {
91*4882a593Smuzhiyun 	char linebuf[MAX_LINE_LEN];
92*4882a593Smuzhiyun 	char *endp;
93*4882a593Smuzhiyun 	char path[SYSFS_PATH_MAX];
94*4882a593Smuzhiyun 
95*4882a593Smuzhiyun 	snprintf(path, sizeof(path), PATH_TO_CPU "cpu%u/topology/%s",
96*4882a593Smuzhiyun 			 cpu, fname);
97*4882a593Smuzhiyun 	if (cpupower_read_sysfs(path, linebuf, MAX_LINE_LEN) == 0)
98*4882a593Smuzhiyun 		return -1;
99*4882a593Smuzhiyun 	*result = strtol(linebuf, &endp, 0);
100*4882a593Smuzhiyun 	if (endp == linebuf || errno == ERANGE)
101*4882a593Smuzhiyun 		return -1;
102*4882a593Smuzhiyun 	return 0;
103*4882a593Smuzhiyun }
104*4882a593Smuzhiyun 
__compare(const void * t1,const void * t2)105*4882a593Smuzhiyun static int __compare(const void *t1, const void *t2)
106*4882a593Smuzhiyun {
107*4882a593Smuzhiyun 	struct cpuid_core_info *top1 = (struct cpuid_core_info *)t1;
108*4882a593Smuzhiyun 	struct cpuid_core_info *top2 = (struct cpuid_core_info *)t2;
109*4882a593Smuzhiyun 	if (top1->pkg < top2->pkg)
110*4882a593Smuzhiyun 		return -1;
111*4882a593Smuzhiyun 	else if (top1->pkg > top2->pkg)
112*4882a593Smuzhiyun 		return 1;
113*4882a593Smuzhiyun 	else if (top1->core < top2->core)
114*4882a593Smuzhiyun 		return -1;
115*4882a593Smuzhiyun 	else if (top1->core > top2->core)
116*4882a593Smuzhiyun 		return 1;
117*4882a593Smuzhiyun 	else if (top1->cpu < top2->cpu)
118*4882a593Smuzhiyun 		return -1;
119*4882a593Smuzhiyun 	else if (top1->cpu > top2->cpu)
120*4882a593Smuzhiyun 		return 1;
121*4882a593Smuzhiyun 	else
122*4882a593Smuzhiyun 		return 0;
123*4882a593Smuzhiyun }
124*4882a593Smuzhiyun 
125*4882a593Smuzhiyun /*
126*4882a593Smuzhiyun  * Returns amount of cpus, negative on error, cpu_top must be
127*4882a593Smuzhiyun  * passed to cpu_topology_release to free resources
128*4882a593Smuzhiyun  *
129*4882a593Smuzhiyun  * Array is sorted after ->pkg, ->core, then ->cpu
130*4882a593Smuzhiyun  */
get_cpu_topology(struct cpupower_topology * cpu_top)131*4882a593Smuzhiyun int get_cpu_topology(struct cpupower_topology *cpu_top)
132*4882a593Smuzhiyun {
133*4882a593Smuzhiyun 	int cpu, last_pkg, cpus = sysconf(_SC_NPROCESSORS_CONF);
134*4882a593Smuzhiyun 
135*4882a593Smuzhiyun 	cpu_top->core_info = malloc(sizeof(struct cpuid_core_info) * cpus);
136*4882a593Smuzhiyun 	if (cpu_top->core_info == NULL)
137*4882a593Smuzhiyun 		return -ENOMEM;
138*4882a593Smuzhiyun 	cpu_top->pkgs = cpu_top->cores = 0;
139*4882a593Smuzhiyun 	for (cpu = 0; cpu < cpus; cpu++) {
140*4882a593Smuzhiyun 		cpu_top->core_info[cpu].cpu = cpu;
141*4882a593Smuzhiyun 		cpu_top->core_info[cpu].is_online = cpupower_is_cpu_online(cpu);
142*4882a593Smuzhiyun 		if(sysfs_topology_read_file(
143*4882a593Smuzhiyun 			cpu,
144*4882a593Smuzhiyun 			"physical_package_id",
145*4882a593Smuzhiyun 			&(cpu_top->core_info[cpu].pkg)) < 0) {
146*4882a593Smuzhiyun 			cpu_top->core_info[cpu].pkg = -1;
147*4882a593Smuzhiyun 			cpu_top->core_info[cpu].core = -1;
148*4882a593Smuzhiyun 			continue;
149*4882a593Smuzhiyun 		}
150*4882a593Smuzhiyun 		if(sysfs_topology_read_file(
151*4882a593Smuzhiyun 			cpu,
152*4882a593Smuzhiyun 			"core_id",
153*4882a593Smuzhiyun 			&(cpu_top->core_info[cpu].core)) < 0) {
154*4882a593Smuzhiyun 			cpu_top->core_info[cpu].pkg = -1;
155*4882a593Smuzhiyun 			cpu_top->core_info[cpu].core = -1;
156*4882a593Smuzhiyun 			continue;
157*4882a593Smuzhiyun 		}
158*4882a593Smuzhiyun 	}
159*4882a593Smuzhiyun 
160*4882a593Smuzhiyun 	qsort(cpu_top->core_info, cpus, sizeof(struct cpuid_core_info),
161*4882a593Smuzhiyun 	      __compare);
162*4882a593Smuzhiyun 
163*4882a593Smuzhiyun 	/* Count the number of distinct pkgs values. This works
164*4882a593Smuzhiyun 	   because the primary sort of the core_info struct was just
165*4882a593Smuzhiyun 	   done by pkg value. */
166*4882a593Smuzhiyun 	last_pkg = cpu_top->core_info[0].pkg;
167*4882a593Smuzhiyun 	for(cpu = 1; cpu < cpus; cpu++) {
168*4882a593Smuzhiyun 		if (cpu_top->core_info[cpu].pkg != last_pkg &&
169*4882a593Smuzhiyun 				cpu_top->core_info[cpu].pkg != -1) {
170*4882a593Smuzhiyun 
171*4882a593Smuzhiyun 			last_pkg = cpu_top->core_info[cpu].pkg;
172*4882a593Smuzhiyun 			cpu_top->pkgs++;
173*4882a593Smuzhiyun 		}
174*4882a593Smuzhiyun 	}
175*4882a593Smuzhiyun 	if (!(cpu_top->core_info[0].pkg == -1))
176*4882a593Smuzhiyun 		cpu_top->pkgs++;
177*4882a593Smuzhiyun 
178*4882a593Smuzhiyun 	/* Intel's cores count is not consecutively numbered, there may
179*4882a593Smuzhiyun 	 * be a core_id of 3, but none of 2. Assume there always is 0
180*4882a593Smuzhiyun 	 * Get amount of cores by counting duplicates in a package
181*4882a593Smuzhiyun 	for (cpu = 0; cpu_top->core_info[cpu].pkg = 0 && cpu < cpus; cpu++) {
182*4882a593Smuzhiyun 		if (cpu_top->core_info[cpu].core == 0)
183*4882a593Smuzhiyun 	cpu_top->cores++;
184*4882a593Smuzhiyun 	*/
185*4882a593Smuzhiyun 	return cpus;
186*4882a593Smuzhiyun }
187*4882a593Smuzhiyun 
cpu_topology_release(struct cpupower_topology cpu_top)188*4882a593Smuzhiyun void cpu_topology_release(struct cpupower_topology cpu_top)
189*4882a593Smuzhiyun {
190*4882a593Smuzhiyun 	free(cpu_top.core_info);
191*4882a593Smuzhiyun }
192