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 #include <linux/types.h>
33*9c5db199SXin Li #include <net/if.h>
34*9c5db199SXin Li #include <linux/sockios.h>
35*9c5db199SXin Li #include <sys/ioctl.h>
36*9c5db199SXin Li
37*9c5db199SXin Li /* work around a bug in debian -- it exposes kernel internal types to userspace */
38*9c5db199SXin Li #define u64 __u64
39*9c5db199SXin Li #define u32 __u32
40*9c5db199SXin Li #define u16 __u16
41*9c5db199SXin Li #define u8 __u8
42*9c5db199SXin Li #include <linux/ethtool.h>
43*9c5db199SXin Li #undef u64
44*9c5db199SXin Li #undef u32
45*9c5db199SXin Li #undef u16
46*9c5db199SXin Li #undef u8
47*9c5db199SXin Li
48*9c5db199SXin Li
49*9c5db199SXin Li
50*9c5db199SXin Li #include "powertop.h"
51*9c5db199SXin Li
52*9c5db199SXin Li
53*9c5db199SXin Li static char wireless_nic[32];
54*9c5db199SXin Li static char rfkill_path[PATH_MAX];
55*9c5db199SXin Li static char powersave_path[PATH_MAX];
56*9c5db199SXin Li
rfkill_enabled(void)57*9c5db199SXin Li static int rfkill_enabled(void)
58*9c5db199SXin Li {
59*9c5db199SXin Li FILE *file;
60*9c5db199SXin Li char val;
61*9c5db199SXin Li if (strlen(rfkill_path)<2)
62*9c5db199SXin Li return 0;
63*9c5db199SXin Li if (access(rfkill_path, W_OK))
64*9c5db199SXin Li return 0;
65*9c5db199SXin Li
66*9c5db199SXin Li file = fopen(rfkill_path, "r");
67*9c5db199SXin Li if (!file)
68*9c5db199SXin Li return 0;
69*9c5db199SXin Li val = fgetc(file);
70*9c5db199SXin Li fclose(file);
71*9c5db199SXin Li if (val != '0') /* already rfkill'd */
72*9c5db199SXin Li return 1;
73*9c5db199SXin Li return 0;
74*9c5db199SXin Li }
75*9c5db199SXin Li
check_unused_wiresless_up(void)76*9c5db199SXin Li int check_unused_wiresless_up(void)
77*9c5db199SXin Li {
78*9c5db199SXin Li FILE *file;
79*9c5db199SXin Li char val;
80*9c5db199SXin Li char line[1024];
81*9c5db199SXin Li if (strlen(rfkill_path)<2)
82*9c5db199SXin Li return 0;
83*9c5db199SXin Li if (access(rfkill_path, W_OK))
84*9c5db199SXin Li return 0;
85*9c5db199SXin Li
86*9c5db199SXin Li file = fopen(rfkill_path, "r");
87*9c5db199SXin Li if (!file)
88*9c5db199SXin Li return 0;
89*9c5db199SXin Li val = fgetc(file);
90*9c5db199SXin Li fclose(file);
91*9c5db199SXin Li if (val != '0') /* already rfkill'd */
92*9c5db199SXin Li return -1;
93*9c5db199SXin Li
94*9c5db199SXin Li sprintf(line,"iwconfig %s 2> /dev/null", wireless_nic);
95*9c5db199SXin Li file = popen(line, "r");
96*9c5db199SXin Li if (!file)
97*9c5db199SXin Li return 0;
98*9c5db199SXin Li while (!feof(file)) {
99*9c5db199SXin Li memset(line, 0, 1024);
100*9c5db199SXin Li if (fgets(line, 1023, file) == 0)
101*9c5db199SXin Li break;
102*9c5db199SXin Li if (strstr(line, "Mode:Managed") && strstr(line,"Access Point: Not-Associated")) {
103*9c5db199SXin Li pclose(file);
104*9c5db199SXin Li return 1;
105*9c5db199SXin Li }
106*9c5db199SXin Li }
107*9c5db199SXin Li pclose(file);
108*9c5db199SXin Li return 0;
109*9c5db199SXin Li }
110*9c5db199SXin Li
111*9c5db199SXin Li
need_wireless_suggest(char * iface)112*9c5db199SXin Li static int need_wireless_suggest(char *iface)
113*9c5db199SXin Li {
114*9c5db199SXin Li FILE *file;
115*9c5db199SXin Li char line[1024];
116*9c5db199SXin Li int ret = 0;
117*9c5db199SXin Li
118*9c5db199SXin Li if (rfkill_enabled())
119*9c5db199SXin Li return 0;
120*9c5db199SXin Li
121*9c5db199SXin Li sprintf(line, "/sbin/iwpriv %s get_power 2> /dev/null", iface);
122*9c5db199SXin Li file = popen(line, "r");
123*9c5db199SXin Li if (!file)
124*9c5db199SXin Li return 0;
125*9c5db199SXin Li while (!feof(file)) {
126*9c5db199SXin Li memset(line, 0, 1024);
127*9c5db199SXin Li if (fgets(line, 1023, file)==NULL)
128*9c5db199SXin Li break;
129*9c5db199SXin Li if (strstr(line, "Power save level: 6 (AC)")) {
130*9c5db199SXin Li ret = 1;
131*9c5db199SXin Li break;
132*9c5db199SXin Li }
133*9c5db199SXin Li }
134*9c5db199SXin Li pclose(file);
135*9c5db199SXin Li return ret;
136*9c5db199SXin Li }
137*9c5db199SXin Li
138*9c5db199SXin Li
need_wireless_suggest_new(void)139*9c5db199SXin Li static int need_wireless_suggest_new(void)
140*9c5db199SXin Li {
141*9c5db199SXin Li FILE *file;
142*9c5db199SXin Li char val;
143*9c5db199SXin Li if (strlen(powersave_path)<2)
144*9c5db199SXin Li return 0;
145*9c5db199SXin Li if (access(powersave_path, W_OK))
146*9c5db199SXin Li return 0;
147*9c5db199SXin Li
148*9c5db199SXin Li if (rfkill_enabled())
149*9c5db199SXin Li return 0;
150*9c5db199SXin Li
151*9c5db199SXin Li file = fopen(powersave_path, "r");
152*9c5db199SXin Li if (!file)
153*9c5db199SXin Li return 0;
154*9c5db199SXin Li val = fgetc(file);
155*9c5db199SXin Li fclose(file);
156*9c5db199SXin Li if (val <= '5' && val >= '0') /* already in powersave */
157*9c5db199SXin Li return 0;
158*9c5db199SXin Li
159*9c5db199SXin Li return 1;
160*9c5db199SXin Li }
161*9c5db199SXin Li
find_4965(void)162*9c5db199SXin Li void find_4965(void)
163*9c5db199SXin Li {
164*9c5db199SXin Li static int tried_4965 = 0;
165*9c5db199SXin Li DIR *dir;
166*9c5db199SXin Li struct dirent *dirent;
167*9c5db199SXin Li char pathname[PATH_MAX];
168*9c5db199SXin Li
169*9c5db199SXin Li if (tried_4965++)
170*9c5db199SXin Li return;
171*9c5db199SXin Li
172*9c5db199SXin Li dir = opendir("/sys/bus/pci/drivers/iwl4965");
173*9c5db199SXin Li while (dir && (dirent = readdir(dir))) {
174*9c5db199SXin Li if (dirent->d_name[0]=='.')
175*9c5db199SXin Li continue;
176*9c5db199SXin Li sprintf(pathname, "/sys/bus/pci/drivers/iwl4965/%s/power_level", dirent->d_name);
177*9c5db199SXin Li if (!access(pathname, W_OK))
178*9c5db199SXin Li strcpy(powersave_path, pathname);
179*9c5db199SXin Li }
180*9c5db199SXin Li if (dir)
181*9c5db199SXin Li closedir(dir);
182*9c5db199SXin Li dir = opendir("/sys/bus/pci/drivers/iwl3945");
183*9c5db199SXin Li if (!dir)
184*9c5db199SXin Li return;
185*9c5db199SXin Li while ((dirent = readdir(dir))) {
186*9c5db199SXin Li if (dirent->d_name[0]=='.')
187*9c5db199SXin Li continue;
188*9c5db199SXin Li sprintf(pathname, "/sys/bus/pci/drivers/iwl3945/%s/power_level", dirent->d_name);
189*9c5db199SXin Li if (!access(pathname, W_OK))
190*9c5db199SXin Li strcpy(powersave_path, pathname);
191*9c5db199SXin Li }
192*9c5db199SXin Li
193*9c5db199SXin Li closedir(dir);
194*9c5db199SXin Li
195*9c5db199SXin Li }
196*9c5db199SXin Li
197*9c5db199SXin Li
find_wireless_nic(void)198*9c5db199SXin Li void find_wireless_nic(void)
199*9c5db199SXin Li {
200*9c5db199SXin Li static int found = 0;
201*9c5db199SXin Li FILE *file;
202*9c5db199SXin Li int sock;
203*9c5db199SXin Li struct ifreq ifr;
204*9c5db199SXin Li struct ethtool_value ethtool;
205*9c5db199SXin Li struct ethtool_drvinfo driver;
206*9c5db199SXin Li int ifaceup = 0;
207*9c5db199SXin Li int ret;
208*9c5db199SXin Li
209*9c5db199SXin Li if (found++)
210*9c5db199SXin Li return;
211*9c5db199SXin Li
212*9c5db199SXin Li wireless_nic[0] = 0;
213*9c5db199SXin Li rfkill_path[0] = 0;
214*9c5db199SXin Li powersave_path[0] = 0;
215*9c5db199SXin Li
216*9c5db199SXin Li strcpy(wireless_nic, "wlan0");
217*9c5db199SXin Li
218*9c5db199SXin Li file = popen("/sbin/iwpriv -a 2> /dev/null", "r");
219*9c5db199SXin Li if (!file)
220*9c5db199SXin Li return;
221*9c5db199SXin Li while (!feof(file)) {
222*9c5db199SXin Li char line[1024];
223*9c5db199SXin Li memset(line, 0, 1024);
224*9c5db199SXin Li if (fgets(line, 1023, file)==NULL)
225*9c5db199SXin Li break;
226*9c5db199SXin Li if (strstr(line, "get_power:Power save level")) {
227*9c5db199SXin Li char *c;
228*9c5db199SXin Li c = strchr(line, ' ');
229*9c5db199SXin Li if (c) *c = 0;
230*9c5db199SXin Li strcpy(wireless_nic, line);
231*9c5db199SXin Li }
232*9c5db199SXin Li if (strstr(line, "wlan0:"))
233*9c5db199SXin Li strcpy(wireless_nic, "wlan0");
234*9c5db199SXin Li }
235*9c5db199SXin Li pclose(file);
236*9c5db199SXin Li
237*9c5db199SXin Li
238*9c5db199SXin Li if (strlen(wireless_nic)==0)
239*9c5db199SXin Li return;
240*9c5db199SXin Li
241*9c5db199SXin Li
242*9c5db199SXin Li memset(&ifr, 0, sizeof(struct ifreq));
243*9c5db199SXin Li memset(ðtool, 0, sizeof(struct ethtool_value));
244*9c5db199SXin Li
245*9c5db199SXin Li sock = socket(AF_INET, SOCK_DGRAM, 0);
246*9c5db199SXin Li if (sock<0)
247*9c5db199SXin Li return;
248*9c5db199SXin Li
249*9c5db199SXin Li strcpy(ifr.ifr_name, wireless_nic);
250*9c5db199SXin Li
251*9c5db199SXin Li /* Check if the interface is up */
252*9c5db199SXin Li ret = ioctl(sock, SIOCGIFFLAGS, &ifr);
253*9c5db199SXin Li if (ret<0) {
254*9c5db199SXin Li close(sock);
255*9c5db199SXin Li return;
256*9c5db199SXin Li }
257*9c5db199SXin Li
258*9c5db199SXin Li ifaceup = 0;
259*9c5db199SXin Li if (ifr.ifr_flags & (IFF_UP | IFF_RUNNING))
260*9c5db199SXin Li ifaceup = 1;
261*9c5db199SXin Li
262*9c5db199SXin Li memset(&driver, 0, sizeof(driver));
263*9c5db199SXin Li driver.cmd = ETHTOOL_GDRVINFO;
264*9c5db199SXin Li ifr.ifr_data = (void*) &driver;
265*9c5db199SXin Li ret = ioctl(sock, SIOCETHTOOL, &ifr);
266*9c5db199SXin Li
267*9c5db199SXin Li sprintf(rfkill_path,"/sys/bus/pci/devices/%s/rfkill/rfkill0/state", driver.bus_info);
268*9c5db199SXin Li sprintf(powersave_path,"/sys/bus/pci/devices/%s/power_level", driver.bus_info);
269*9c5db199SXin Li close(sock);
270*9c5db199SXin Li }
271*9c5db199SXin Li
activate_wireless_suggestion(void)272*9c5db199SXin Li void activate_wireless_suggestion(void)
273*9c5db199SXin Li {
274*9c5db199SXin Li char line[1024];
275*9c5db199SXin Li sprintf(line, "/sbin/iwpriv %s set_power 5 2> /dev/null", wireless_nic);
276*9c5db199SXin Li system(line);
277*9c5db199SXin Li }
activate_wireless_suggestion_new(void)278*9c5db199SXin Li void activate_wireless_suggestion_new(void)
279*9c5db199SXin Li {
280*9c5db199SXin Li FILE *file;
281*9c5db199SXin Li file = fopen(powersave_path, "w");
282*9c5db199SXin Li if (!file)
283*9c5db199SXin Li return;
284*9c5db199SXin Li fprintf(file,"1\n");
285*9c5db199SXin Li fclose(file);
286*9c5db199SXin Li }
287*9c5db199SXin Li
activate_rfkill_suggestion(void)288*9c5db199SXin Li void activate_rfkill_suggestion(void)
289*9c5db199SXin Li {
290*9c5db199SXin Li FILE *file;
291*9c5db199SXin Li file = fopen(rfkill_path, "w");
292*9c5db199SXin Li if (!file)
293*9c5db199SXin Li return;
294*9c5db199SXin Li fprintf(file,"1\n");
295*9c5db199SXin Li fclose(file);
296*9c5db199SXin Li }
suggest_wireless_powersave(void)297*9c5db199SXin Li void suggest_wireless_powersave(void)
298*9c5db199SXin Li {
299*9c5db199SXin Li char sug[1024];
300*9c5db199SXin Li int ret;
301*9c5db199SXin Li
302*9c5db199SXin Li if (strlen(wireless_nic)==0)
303*9c5db199SXin Li find_wireless_nic();
304*9c5db199SXin Li find_4965();
305*9c5db199SXin Li ret = check_unused_wiresless_up();
306*9c5db199SXin Li
307*9c5db199SXin Li if (ret >= 0 && need_wireless_suggest(wireless_nic)) {
308*9c5db199SXin Li sprintf(sug, _("Suggestion: Enable wireless power saving mode by executing the following command:\n "
309*9c5db199SXin Li " iwpriv %s set_power 5 \n"
310*9c5db199SXin Li "This will sacrifice network performance slightly to save power."), wireless_nic);
311*9c5db199SXin Li add_suggestion(sug, 20, 'W', _(" W - Enable wireless power saving "), activate_wireless_suggestion);
312*9c5db199SXin Li }
313*9c5db199SXin Li if (ret >= 0 && need_wireless_suggest_new()) {
314*9c5db199SXin Li sprintf(sug, _("Suggestion: Enable wireless power saving mode by executing the following command:\n "
315*9c5db199SXin Li " echo 5 > %s \n"
316*9c5db199SXin Li "This will sacrifice network performance slightly to save power."), powersave_path);
317*9c5db199SXin Li add_suggestion(sug, 20, 'W', _(" W - Enable wireless power saving "), activate_wireless_suggestion_new);
318*9c5db199SXin Li }
319*9c5db199SXin Li if (ret>0) {
320*9c5db199SXin Li sprintf(sug, _("Suggestion: Disable the unused WIFI radio by executing the following command:\n "
321*9c5db199SXin Li " echo 1 > %s \n"), rfkill_path);
322*9c5db199SXin Li add_suggestion(sug, 60, 'I', _(" I - disable WIFI Radio "), activate_rfkill_suggestion);
323*9c5db199SXin Li
324*9c5db199SXin Li }
325*9c5db199SXin Li }
326