xref: /aosp_15_r20/external/autotest/client/profilers/powertop/src/misctips.c (revision 9c5db1993ded3edbeafc8092d69fe5de2ee02df7)
1*9c5db199SXin Li /*
2*9c5db199SXin Li  * Copyright 2007, Intel Corporation
3*9c5db199SXin Li  *
4*9c5db199SXin Li  * This file is part of PowerTOP
5*9c5db199SXin Li  *
6*9c5db199SXin Li  * This program file is free software; you can redistribute it and/or modify it
7*9c5db199SXin Li  * under the terms of the GNU General Public License as published by the
8*9c5db199SXin Li  * Free Software Foundation; version 2 of the License.
9*9c5db199SXin Li  *
10*9c5db199SXin Li  * This program is distributed in the hope that it will be useful, but WITHOUT
11*9c5db199SXin Li  * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
12*9c5db199SXin Li  * FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General Public License
13*9c5db199SXin Li  * for more details.
14*9c5db199SXin Li  *
15*9c5db199SXin Li  * You should have received a copy of the GNU General Public License
16*9c5db199SXin Li  * along with this program in a file named COPYING; if not, write to the
17*9c5db199SXin Li  * Free Software Foundation, Inc.,
18*9c5db199SXin Li  * 51 Franklin Street, Fifth Floor,
19*9c5db199SXin Li  * Boston, MA 02110-1301 USA
20*9c5db199SXin Li  *
21*9c5db199SXin Li  * Authors:
22*9c5db199SXin Li  * 	Arjan van de Ven <[email protected]>
23*9c5db199SXin Li  */
24*9c5db199SXin Li 
25*9c5db199SXin Li #include <unistd.h>
26*9c5db199SXin Li #include <stdio.h>
27*9c5db199SXin Li #include <stdlib.h>
28*9c5db199SXin Li #include <string.h>
29*9c5db199SXin Li #include <stdint.h>
30*9c5db199SXin Li #include <sys/types.h>
31*9c5db199SXin Li #include <dirent.h>
32*9c5db199SXin Li 
33*9c5db199SXin Li #include "powertop.h"
34*9c5db199SXin Li 
set_laptop_mode(void)35*9c5db199SXin Li void set_laptop_mode(void)
36*9c5db199SXin Li {
37*9c5db199SXin Li 	FILE *file;
38*9c5db199SXin Li 	file = fopen("/proc/sys/vm/laptop_mode", "w");
39*9c5db199SXin Li 	if (!file)
40*9c5db199SXin Li 		return;
41*9c5db199SXin Li 	fprintf(file,"5\n");
42*9c5db199SXin Li 	fclose(file);
43*9c5db199SXin Li }
44*9c5db199SXin Li 
suggest_laptop_mode(void)45*9c5db199SXin Li void suggest_laptop_mode(void)
46*9c5db199SXin Li {
47*9c5db199SXin Li 	FILE *file;
48*9c5db199SXin Li 	int i;
49*9c5db199SXin Li 	char buffer[1024];
50*9c5db199SXin Li 	/*
51*9c5db199SXin Li 	 * Check to see if we are on AC - lots of distros have
52*9c5db199SXin Li 	 * annoying scripts to turn laptop mode off when on AC, which
53*9c5db199SXin Li 	 * results in annoying distracting return of set laptop mode
54*9c5db199SXin Li 	 * hint.
55*9c5db199SXin Li 	 */
56*9c5db199SXin Li 	file = fopen("/proc/acpi/ac_adapter/AC/state", "r");
57*9c5db199SXin Li 	if (!file)
58*9c5db199SXin Li 		return;
59*9c5db199SXin Li 	memset(buffer, 0, 1024);
60*9c5db199SXin Li 	if (!fgets(buffer, 1023, file)) {
61*9c5db199SXin Li 		fclose(file);
62*9c5db199SXin Li 		return;
63*9c5db199SXin Li 	}
64*9c5db199SXin Li 	fclose(file);
65*9c5db199SXin Li 	if (strstr(buffer, "on-line") != NULL)
66*9c5db199SXin Li 		return;
67*9c5db199SXin Li 
68*9c5db199SXin Li 	/* Now check for laptop mode */
69*9c5db199SXin Li 	file = fopen("/proc/sys/vm/laptop_mode", "r");
70*9c5db199SXin Li 	if (!file)
71*9c5db199SXin Li 		return;
72*9c5db199SXin Li 	memset(buffer, 0, 1024);
73*9c5db199SXin Li 	if (!fgets(buffer, 1023, file)) {
74*9c5db199SXin Li 		fclose(file);
75*9c5db199SXin Li 		return;
76*9c5db199SXin Li 	}
77*9c5db199SXin Li 	i = strtoul(buffer, NULL, 10);
78*9c5db199SXin Li 	if (i<1) {
79*9c5db199SXin Li 		add_suggestion( _("Suggestion: Enable laptop-mode by executing the following command:\n"
80*9c5db199SXin Li 		 	"   echo 5 > /proc/sys/vm/laptop_mode \n"), 15, 'L', _(" L - enable Laptop mode "), set_laptop_mode);
81*9c5db199SXin Li 	}
82*9c5db199SXin Li 	fclose(file);
83*9c5db199SXin Li }
84*9c5db199SXin Li 
nmi_watchdog_off(void)85*9c5db199SXin Li void nmi_watchdog_off(void)
86*9c5db199SXin Li {
87*9c5db199SXin Li 	FILE *file;
88*9c5db199SXin Li 	file = fopen("/proc/sys/kernel/nmi_watchdog", "w");
89*9c5db199SXin Li 	if (!file)
90*9c5db199SXin Li 		return;
91*9c5db199SXin Li 	fprintf(file,"0\n");
92*9c5db199SXin Li 	fclose(file);
93*9c5db199SXin Li }
suggest_nmi_watchdog(void)94*9c5db199SXin Li void suggest_nmi_watchdog(void)
95*9c5db199SXin Li {
96*9c5db199SXin Li 	FILE *file;
97*9c5db199SXin Li 	int i;
98*9c5db199SXin Li 	char buffer[1024];
99*9c5db199SXin Li 	file = fopen("/proc/sys/kernel/nmi_watchdog", "r");
100*9c5db199SXin Li 	if (!file)
101*9c5db199SXin Li 		return;
102*9c5db199SXin Li 	memset(buffer, 0, 1024);
103*9c5db199SXin Li 	if (!fgets(buffer, 1023, file)) {
104*9c5db199SXin Li 		fclose(file);
105*9c5db199SXin Li 		return;
106*9c5db199SXin Li 	}
107*9c5db199SXin Li 	i = strtoul(buffer, NULL, 10);
108*9c5db199SXin Li 	if (i!=0) {
109*9c5db199SXin Li 		add_suggestion( _("Suggestion: disable the NMI watchdog by executing the following command:\n"
110*9c5db199SXin Li 		 	"   echo 0 > /proc/sys/kernel/nmi_watchdog \n"
111*9c5db199SXin Li 			"The NMI watchdog is a kernel debug mechanism to detect deadlocks"), 25, 'N', _(" N - Turn NMI watchdog off "), nmi_watchdog_off);
112*9c5db199SXin Li 	}
113*9c5db199SXin Li 	fclose(file);
114*9c5db199SXin Li }
115*9c5db199SXin Li 
suggest_hpet(void)116*9c5db199SXin Li void suggest_hpet(void)
117*9c5db199SXin Li {
118*9c5db199SXin Li 	FILE *file;
119*9c5db199SXin Li 	char buffer[1024];
120*9c5db199SXin Li 	file = fopen("/sys/devices/system/clocksource/clocksource0/available_clocksource", "r");
121*9c5db199SXin Li 	if (!file)
122*9c5db199SXin Li 		return;
123*9c5db199SXin Li 	memset(buffer, 0, 1024);
124*9c5db199SXin Li 
125*9c5db199SXin Li 	if (!fgets(buffer, 1023, file)) {
126*9c5db199SXin Li 		fclose(file);
127*9c5db199SXin Li 		return;
128*9c5db199SXin Li 	}
129*9c5db199SXin Li 
130*9c5db199SXin Li 	if (strstr(buffer, "hpet")) {
131*9c5db199SXin Li 		fclose(file);
132*9c5db199SXin Li 		return;
133*9c5db199SXin Li 	}
134*9c5db199SXin Li 
135*9c5db199SXin Li 	fclose(file);
136*9c5db199SXin Li 
137*9c5db199SXin Li 	add_suggestion( _("Suggestion: enable the HPET (Multimedia Timer) in your BIOS or add \n"
138*9c5db199SXin Li 		          "the kernel patch to force-enable HPET. HPET support allows Linux to \n"
139*9c5db199SXin Li 			  "have much longer sleep intervals."), 7, 0, NULL, NULL);
140*9c5db199SXin Li }
141*9c5db199SXin Li 
ac97_power_on(void)142*9c5db199SXin Li void ac97_power_on(void)
143*9c5db199SXin Li {
144*9c5db199SXin Li 	FILE *file;
145*9c5db199SXin Li 	file = fopen("/sys/module/snd_ac97_codec/parameters/power_save", "w");
146*9c5db199SXin Li 	if (!file)
147*9c5db199SXin Li 		return;
148*9c5db199SXin Li 	fprintf(file,"1");
149*9c5db199SXin Li 	fclose(file);
150*9c5db199SXin Li 	if (access("/dev/dsp", F_OK))
151*9c5db199SXin Li 		return;
152*9c5db199SXin Li 	file = fopen("/dev/dsp", "w");
153*9c5db199SXin Li 	if (file) {
154*9c5db199SXin Li 		fprintf(file,"1");
155*9c5db199SXin Li 		fclose(file);
156*9c5db199SXin Li 	}
157*9c5db199SXin Li }
158*9c5db199SXin Li 
suggest_ac97_powersave(void)159*9c5db199SXin Li void suggest_ac97_powersave(void)
160*9c5db199SXin Li {
161*9c5db199SXin Li 	FILE *file;
162*9c5db199SXin Li 	char buffer[1024];
163*9c5db199SXin Li 	file = fopen("/sys/module/snd_ac97_codec/parameters/power_save", "r");
164*9c5db199SXin Li 	if (!file)
165*9c5db199SXin Li 		return;
166*9c5db199SXin Li 	memset(buffer, 0, 1024);
167*9c5db199SXin Li 	if (!fgets(buffer, 1023, file)) {
168*9c5db199SXin Li 		fclose(file);
169*9c5db199SXin Li 		return;
170*9c5db199SXin Li 	}
171*9c5db199SXin Li 	if (buffer[0]=='N') {
172*9c5db199SXin Li 		add_suggestion( _("Suggestion: enable AC97 powersave mode by executing the following command:\n"
173*9c5db199SXin Li 		 	"   echo 1 > /sys/module/snd_ac97_codec/parameters/power_save \n"
174*9c5db199SXin Li 			"or by passing power_save=1 as module parameter."), 25, 'A', _(" A - Turn AC97 powersave on "), ac97_power_on);
175*9c5db199SXin Li 	}
176*9c5db199SXin Li 	fclose(file);
177*9c5db199SXin Li }
178*9c5db199SXin Li 
noatime_on(void)179*9c5db199SXin Li void noatime_on(void)
180*9c5db199SXin Li {
181*9c5db199SXin Li 	system("/bin/mount -o remount,noatime,nodiratime /");
182*9c5db199SXin Li }
183*9c5db199SXin Li 
suggest_noatime(void)184*9c5db199SXin Li void suggest_noatime(void)
185*9c5db199SXin Li {
186*9c5db199SXin Li 	FILE *file;
187*9c5db199SXin Li 	char buffer[1024];
188*9c5db199SXin Li 	int suggest = 0;
189*9c5db199SXin Li 	file = fopen("/proc/mounts","r");
190*9c5db199SXin Li 	if (!file)
191*9c5db199SXin Li 		return;
192*9c5db199SXin Li 	while (!feof(file)) {
193*9c5db199SXin Li 		memset(buffer, 0, 1024);
194*9c5db199SXin Li 		if (!fgets(buffer, 1023, file))
195*9c5db199SXin Li 			break;
196*9c5db199SXin Li 		if (strstr(buffer, " / ext3") && !strstr(buffer, "noatime") && !strstr(buffer, "relatime"))
197*9c5db199SXin Li 			suggest = 1;
198*9c5db199SXin Li 
199*9c5db199SXin Li 	}
200*9c5db199SXin Li 	if (suggest) {
201*9c5db199SXin Li 		add_suggestion( _("Suggestion: enable the noatime filesystem option by executing the following command:\n"
202*9c5db199SXin Li 		 	"   mount -o remount,noatime /          or by pressing the T key \n"
203*9c5db199SXin Li 			"noatime disables persistent access time of file accesses, which causes lots of disk IO."), 5, 'T', _(" T - enable noatime "), noatime_on);
204*9c5db199SXin Li 	}
205*9c5db199SXin Li 	fclose(file);
206*9c5db199SXin Li }
207*9c5db199SXin Li 
powersched_on(void)208*9c5db199SXin Li void powersched_on(void)
209*9c5db199SXin Li {
210*9c5db199SXin Li 	FILE *file;
211*9c5db199SXin Li 	file = fopen("/sys/devices/system/cpu/sched_mc_power_savings", "w");
212*9c5db199SXin Li 	if (!file)
213*9c5db199SXin Li 		return;
214*9c5db199SXin Li 	fprintf(file,"1");
215*9c5db199SXin Li 	fclose(file);
216*9c5db199SXin Li }
217*9c5db199SXin Li 
suggest_powersched(void)218*9c5db199SXin Li void suggest_powersched(void)
219*9c5db199SXin Li {
220*9c5db199SXin Li 	FILE *file;
221*9c5db199SXin Li 	char buffer[1024];
222*9c5db199SXin Li 	int suggest = 0;
223*9c5db199SXin Li 	int cpu;
224*9c5db199SXin Li 
225*9c5db199SXin Li 	file = fopen("/sys/devices/system/cpu/sched_mc_power_savings","r");
226*9c5db199SXin Li 	if (!file)
227*9c5db199SXin Li 		return;
228*9c5db199SXin Li 	memset(buffer, 0, 1024);
229*9c5db199SXin Li 	if (!fgets(buffer, 1023, file)) {
230*9c5db199SXin Li 		fclose(file);
231*9c5db199SXin Li 		return;
232*9c5db199SXin Li 	}
233*9c5db199SXin Li 	fclose(file);
234*9c5db199SXin Li 	if (buffer[0]!='0')
235*9c5db199SXin Li 		return;
236*9c5db199SXin Li 	/* ok so power saving scheduler is off; now to see if we actually have a multi-package system */
237*9c5db199SXin Li 	cpu =  sysconf(_SC_NPROCESSORS_ONLN);
238*9c5db199SXin Li 
239*9c5db199SXin Li 	if (cpu<2)
240*9c5db199SXin Li 		return; /* UP system */
241*9c5db199SXin Li 
242*9c5db199SXin Li 	file = fopen("/proc/cpuinfo", "r");
243*9c5db199SXin Li 	suggest = 1;
244*9c5db199SXin Li 	if (!file)
245*9c5db199SXin Li 		return;
246*9c5db199SXin Li 	while (!feof(file)) {
247*9c5db199SXin Li 		memset(buffer, 0, 1024);
248*9c5db199SXin Li 		char *c;
249*9c5db199SXin Li 		if (!fgets(buffer, 1023, file))
250*9c5db199SXin Li 			break;
251*9c5db199SXin Li 		if (strstr(buffer, "cpu cores")) {
252*9c5db199SXin Li 			c = strchr(buffer, ':');
253*9c5db199SXin Li 			if (!c)
254*9c5db199SXin Li 				continue;
255*9c5db199SXin Li 			c++;
256*9c5db199SXin Li 			if (strtoll(c, NULL, 10) >= cpu)
257*9c5db199SXin Li 				suggest = 0;
258*9c5db199SXin Li 		}
259*9c5db199SXin Li 	}
260*9c5db199SXin Li 	fclose(file);
261*9c5db199SXin Li 
262*9c5db199SXin Li 
263*9c5db199SXin Li 	if (suggest) {
264*9c5db199SXin Li 		add_suggestion( _("Suggestion: enable the power aware CPU scheduler with the following command:\n"
265*9c5db199SXin Li 		 	"  echo 1 > /sys/devices/system/cpu/sched_mc_power_savings\n"
266*9c5db199SXin Li 			"or by pressing the C key."), 5, 'C', _(" C - Power aware CPU scheduler "), powersched_on);
267*9c5db199SXin Li 	}
268*9c5db199SXin Li }
269*9c5db199SXin Li 
270*9c5db199SXin Li 
writeback_long(void)271*9c5db199SXin Li void writeback_long(void)
272*9c5db199SXin Li {
273*9c5db199SXin Li 	FILE *file;
274*9c5db199SXin Li 	file = fopen("/proc/sys/vm/dirty_writeback_centisecs", "w");
275*9c5db199SXin Li 	if (!file)
276*9c5db199SXin Li 		return;
277*9c5db199SXin Li 	fprintf(file,"1500");
278*9c5db199SXin Li 	fclose(file);
279*9c5db199SXin Li }
280*9c5db199SXin Li 
suggest_writeback_time(void)281*9c5db199SXin Li void suggest_writeback_time(void)
282*9c5db199SXin Li {
283*9c5db199SXin Li 	FILE *file;
284*9c5db199SXin Li 	char buffer[1024];
285*9c5db199SXin Li 	int i;
286*9c5db199SXin Li 	file = fopen("/proc/sys/vm/dirty_writeback_centisecs", "r");
287*9c5db199SXin Li 	if (!file)
288*9c5db199SXin Li 		return;
289*9c5db199SXin Li 	memset(buffer, 0, 1024);
290*9c5db199SXin Li 	if (!fgets(buffer, 1023, file)) {
291*9c5db199SXin Li 		fclose(file);
292*9c5db199SXin Li 		return;
293*9c5db199SXin Li 	}
294*9c5db199SXin Li 	i = strtoull(buffer, NULL, 10);
295*9c5db199SXin Li 	if (i<1400) {
296*9c5db199SXin Li 		char line[1024];
297*9c5db199SXin Li 		sprintf(line,_("Suggestion: increase the VM dirty writeback time from %1.2f to 15 seconds with:\n"
298*9c5db199SXin Li 			 	"  echo 1500 > /proc/sys/vm/dirty_writeback_centisecs \n"
299*9c5db199SXin Li 				"This wakes the disk up less frequently for background VM activity"),
300*9c5db199SXin Li 			i/100.0);
301*9c5db199SXin Li 		add_suggestion(line, 15, 'W', _(" W - Increase Writeback time "), writeback_long);
302*9c5db199SXin Li 	}
303*9c5db199SXin Li 	fclose(file);
304*9c5db199SXin Li }
305