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 <assert.h>
33*9c5db199SXin Li
34*9c5db199SXin Li #include "powertop.h"
35*9c5db199SXin Li
36*9c5db199SXin Li struct device_data;
37*9c5db199SXin Li
38*9c5db199SXin Li struct device_data {
39*9c5db199SXin Li struct device_data *next;
40*9c5db199SXin Li char pathname[4096];
41*9c5db199SXin Li char human_name[4096];
42*9c5db199SXin Li uint64_t urbs, active, connected;
43*9c5db199SXin Li uint64_t previous_urbs, previous_active, previous_connected;
44*9c5db199SXin Li int controller;
45*9c5db199SXin Li };
46*9c5db199SXin Li
47*9c5db199SXin Li
48*9c5db199SXin Li static struct device_data *devices;
49*9c5db199SXin Li
cachunk_urbs(void)50*9c5db199SXin Li static void cachunk_urbs(void)
51*9c5db199SXin Li {
52*9c5db199SXin Li struct device_data *ptr;
53*9c5db199SXin Li ptr = devices;
54*9c5db199SXin Li while (ptr) {
55*9c5db199SXin Li ptr->previous_urbs = ptr->urbs;
56*9c5db199SXin Li ptr->previous_active = ptr->active;
57*9c5db199SXin Li ptr->previous_connected = ptr->connected;
58*9c5db199SXin Li ptr = ptr->next;
59*9c5db199SXin Li }
60*9c5db199SXin Li }
61*9c5db199SXin Li
update_urbnum(char * path,uint64_t count,char * shortname)62*9c5db199SXin Li static void update_urbnum(char *path, uint64_t count, char *shortname)
63*9c5db199SXin Li {
64*9c5db199SXin Li struct device_data *ptr;
65*9c5db199SXin Li FILE *file;
66*9c5db199SXin Li char fullpath[4096];
67*9c5db199SXin Li char name[4096], vendor[4096];
68*9c5db199SXin Li ptr = devices;
69*9c5db199SXin Li
70*9c5db199SXin Li while (ptr) {
71*9c5db199SXin Li if (strcmp(ptr->pathname, path)==0) {
72*9c5db199SXin Li ptr->urbs = count;
73*9c5db199SXin Li sprintf(fullpath, "%s/power/active_duration", path);
74*9c5db199SXin Li file = fopen(fullpath, "r");
75*9c5db199SXin Li if (!file)
76*9c5db199SXin Li return;
77*9c5db199SXin Li fgets(name, 4096, file);
78*9c5db199SXin Li ptr->active = strtoull(name, NULL, 10);
79*9c5db199SXin Li fclose(file);
80*9c5db199SXin Li sprintf(fullpath, "%s/power/connected_duration", path);
81*9c5db199SXin Li file = fopen(fullpath, "r");
82*9c5db199SXin Li if (!file)
83*9c5db199SXin Li return;
84*9c5db199SXin Li fgets(name, 4096, file);
85*9c5db199SXin Li ptr->connected = strtoull(name, NULL, 10);
86*9c5db199SXin Li fclose(file);
87*9c5db199SXin Li
88*9c5db199SXin Li return;
89*9c5db199SXin Li }
90*9c5db199SXin Li ptr = ptr->next;
91*9c5db199SXin Li }
92*9c5db199SXin Li /* no luck, new one */
93*9c5db199SXin Li ptr = malloc(sizeof(struct device_data));
94*9c5db199SXin Li assert(ptr!=0);
95*9c5db199SXin Li memset(ptr, 0, sizeof(struct device_data));
96*9c5db199SXin Li ptr->next = devices;
97*9c5db199SXin Li devices = ptr;
98*9c5db199SXin Li strcpy(ptr->pathname, path);
99*9c5db199SXin Li ptr->urbs = ptr->previous_urbs = count;
100*9c5db199SXin Li sprintf(fullpath, "%s/product", path);
101*9c5db199SXin Li file = fopen(fullpath, "r");
102*9c5db199SXin Li memset(name, 0, 4096);
103*9c5db199SXin Li if (file) {
104*9c5db199SXin Li fgets(name, 4096, file);
105*9c5db199SXin Li fclose(file);
106*9c5db199SXin Li }
107*9c5db199SXin Li sprintf(fullpath, "%s/manufacturer", path);
108*9c5db199SXin Li file = fopen(fullpath, "r");
109*9c5db199SXin Li memset(vendor, 0, 4096);
110*9c5db199SXin Li if (file) {
111*9c5db199SXin Li fgets(vendor, 4096, file);
112*9c5db199SXin Li fclose(file);
113*9c5db199SXin Li }
114*9c5db199SXin Li
115*9c5db199SXin Li if (strlen(name)>0 && name[strlen(name)-1]=='\n')
116*9c5db199SXin Li name[strlen(name)-1]=0;
117*9c5db199SXin Li if (strlen(vendor)>0 && vendor[strlen(vendor)-1]=='\n')
118*9c5db199SXin Li vendor[strlen(vendor)-1]=0;
119*9c5db199SXin Li /* some devices have bogus names */
120*9c5db199SXin Li if (strlen(name)<4)
121*9c5db199SXin Li strcpy(ptr->human_name, path);
122*9c5db199SXin Li else
123*9c5db199SXin Li sprintf(ptr->human_name, _("USB device %4s : %s (%s)"), shortname, name, vendor);
124*9c5db199SXin Li
125*9c5db199SXin Li if (strstr(ptr->human_name, "Host Controller"))
126*9c5db199SXin Li ptr->controller = 1;
127*9c5db199SXin Li
128*9c5db199SXin Li }
129*9c5db199SXin Li
count_usb_urbs(void)130*9c5db199SXin Li void count_usb_urbs(void)
131*9c5db199SXin Li {
132*9c5db199SXin Li DIR *dir;
133*9c5db199SXin Li struct dirent *dirent;
134*9c5db199SXin Li FILE *file;
135*9c5db199SXin Li char filename[PATH_MAX];
136*9c5db199SXin Li char pathname[PATH_MAX];
137*9c5db199SXin Li char buffer[4096];
138*9c5db199SXin Li struct device_data *dev;
139*9c5db199SXin Li
140*9c5db199SXin Li dir = opendir("/sys/bus/usb/devices");
141*9c5db199SXin Li if (!dir)
142*9c5db199SXin Li return;
143*9c5db199SXin Li
144*9c5db199SXin Li cachunk_urbs();
145*9c5db199SXin Li while ((dirent = readdir(dir))) {
146*9c5db199SXin Li if (dirent->d_name[0]=='.')
147*9c5db199SXin Li continue;
148*9c5db199SXin Li sprintf(pathname, "/sys/bus/usb/devices/%s", dirent->d_name);
149*9c5db199SXin Li sprintf(filename, "%s/urbnum", pathname);
150*9c5db199SXin Li file = fopen(filename, "r");
151*9c5db199SXin Li if (!file)
152*9c5db199SXin Li continue;
153*9c5db199SXin Li memset(buffer, 0, 4096);
154*9c5db199SXin Li fgets(buffer, 4095, file);
155*9c5db199SXin Li update_urbnum(pathname, strtoull(buffer, NULL, 10), dirent->d_name);
156*9c5db199SXin Li fclose(file);
157*9c5db199SXin Li }
158*9c5db199SXin Li
159*9c5db199SXin Li closedir(dir);
160*9c5db199SXin Li
161*9c5db199SXin Li dev = devices;
162*9c5db199SXin Li while (dev) {
163*9c5db199SXin Li if (dev->urbs != dev->previous_urbs) {
164*9c5db199SXin Li push_line(dev->human_name, dev->urbs - dev->previous_urbs);
165*9c5db199SXin Li }
166*9c5db199SXin Li dev = dev->next;
167*9c5db199SXin Li }
168*9c5db199SXin Li }
169*9c5db199SXin Li
170*9c5db199SXin Li
display_usb_activity(void)171*9c5db199SXin Li void display_usb_activity(void)
172*9c5db199SXin Li {
173*9c5db199SXin Li struct device_data *dev;
174*9c5db199SXin Li printf("\n");
175*9c5db199SXin Li printf("%s\n", _("Recent USB suspend statistics"));
176*9c5db199SXin Li printf("%s\n", _("Active Device name"));
177*9c5db199SXin Li dev = devices;
178*9c5db199SXin Li while (dev) {
179*9c5db199SXin Li printf("%5.1f%%\t%s\n", 100.0*(dev->active - dev->previous_active) /
180*9c5db199SXin Li (0.00001 + dev->connected - dev->previous_connected), dev->human_name);
181*9c5db199SXin Li dev = dev->next;
182*9c5db199SXin Li }
183*9c5db199SXin Li
184*9c5db199SXin Li }
185*9c5db199SXin Li
usb_activity_hint(void)186*9c5db199SXin Li void usb_activity_hint(void)
187*9c5db199SXin Li {
188*9c5db199SXin Li int total_active = 0;
189*9c5db199SXin Li int pick;
190*9c5db199SXin Li struct device_data *dev;
191*9c5db199SXin Li dev = devices;
192*9c5db199SXin Li while (dev) {
193*9c5db199SXin Li if (dev->active-1 > dev->previous_active && !dev->controller)
194*9c5db199SXin Li total_active++;
195*9c5db199SXin Li dev = dev->next;
196*9c5db199SXin Li }
197*9c5db199SXin Li if (!total_active)
198*9c5db199SXin Li return;
199*9c5db199SXin Li
200*9c5db199SXin Li pick = rand() % total_active;
201*9c5db199SXin Li total_active = 0;
202*9c5db199SXin Li dev = devices;
203*9c5db199SXin Li while (dev) {
204*9c5db199SXin Li if (dev->active-1 > dev->previous_active && !dev->controller) {
205*9c5db199SXin Li if (total_active == pick) {
206*9c5db199SXin Li char usb_hint[8000];
207*9c5db199SXin Li sprintf(usb_hint, _("A USB device is active %4.1f%% of the time:\n%s"),
208*9c5db199SXin Li 100.0*(dev->active - dev->previous_active) /
209*9c5db199SXin Li (0.00001 + dev->connected - dev->previous_connected),
210*9c5db199SXin Li dev->human_name);
211*9c5db199SXin Li add_suggestion(usb_hint,
212*9c5db199SXin Li 1, 'U', _(" U - Enable USB suspend "), activate_usb_autosuspend);
213*9c5db199SXin Li }
214*9c5db199SXin Li total_active++;
215*9c5db199SXin Li }
216*9c5db199SXin Li dev = dev->next;
217*9c5db199SXin Li }
218*9c5db199SXin Li
219*9c5db199SXin Li }
220