xref: /aosp_15_r20/external/autotest/client/profilers/powertop/src/urbnum.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 #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