aboutsummaryrefslogtreecommitdiff
path: root/getfreq.c
blob: ac989fdcb5214cf781afa12712f343828bf8183b (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
/************************************************************************
 * This file is part of paramano.                                       *
 *                                                                      *
 * paramano is free software; you can redistribute it and/or            *
 * modify it under the terms of the GNU General Public License as       *
 * published by the Free Software Foundation; either version 3 of the   *
 * License, or (at your option) any later version.                      *
 *                                                                      *
 * paramano is distributed in the hope that it will be useful,          *
 * but WITHOUT ANY WARRANTY; without even the implied warranty of       *
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the        *
 * GNU General Public License for more details.                         *
 *                                                                      *
 * You should have received a copy of the GNU General Public License    *
 * along with paramano. If not, see                                     *
 * <http://www.gnu.org/licenses/>.                                      *
 ************************************************************************/

#include "paramano.h"

#define MAX_CORES 1000
#define MAX_FREQS 50
#define FREQ_LENGTH 14

char freqs[MAX_CORES][MAX_FREQS][FREQ_LENGTH];
int total_freqs; // Number of freqs for core 0


/***********************************************************************
 * Initialise surrounding variables, get available freqs etc
 **********************************************************************/
void gf_init()
{
	char freq_string[4096]; // POSIX suggested line length. Source of error for CPUs with a huge number of freqs
	char *next_token;
	unsigned int i;

	memset(freqs, '\0', sizeof(freqs));


	for(i = 0; (i < gc_number() && i < MAX_CORES); i++)
	{
		memset(freq_string, '\0', sizeof(freq_string) );

		// Get available governor freqs. If no governor, try next cpu
		if (gf_available(i, freq_string, sizeof(freq_string) ) == -1)
		{
			debug("Couldn't find freq scaling on core %d\n",i);
			continue;
		}

		*strchrnul(freq_string, '\n') = '\0';

		// freq_string is a space separated list of freqs
		// Use strtok to find each
		next_token = strtok(freq_string, " \n");
		total_freqs = 0;
		do
		{
			//chomp(next_token);
			debug("Found frequency #%d (%s KHz)\n",total_freqs,next_token);
			strncpy(freqs[i][total_freqs], next_token, FREQ_LENGTH);
			total_freqs++;
		} while((next_token = strtok(NULL, " ")) != NULL);
	}

	// Hit the limit of storage of cores' frequencies
	if (i == MAX_CORES)
		info("Unable to add more than %d cores\n", MAX_CORES);

	debug("Found %d frequencies\n",total_freqs);
}

/***********************************************************************
 * Return current frequency for core
 **********************************************************************/
int gf_current(int core)
{
	FILE* fd;
	char buff[13]; // TO DO : magic constant
	char* path;
	int freq;

	asprintf(&path, "/sys/devices/system/cpu/cpu%d/cpufreq/scaling_cur_freq", core);

	if(!(fd = fopen(path, "r")))
	{
		debug("Couldn't open '%s'\n",path);
		free(path);
		return -1;
	}

	fgets(buff, 13, fd);

	freq = atoi(buff);
	fclose(fd);
	debug("Found freq %d on core %d\n",freq,core);

	free(path);
	return freq;
}


/***********************************************************************
 * Populate `out` with available frequencies for core
 **********************************************************************/
int gf_available(int core, char* out, int size)
{
	FILE* fd;
	char* path;

	asprintf(&path, "/sys/devices/system/cpu/cpu%d/cpufreq/scaling_available_frequencies", core);

	if(!(fd = fopen(path, "r")))
	{
		debug("Couldn't open '%s'\n",path);
		free(path);
		return -1;
	}

	fgets(out, size, fd);

	fclose(fd);
	free(path);
	return 0;
}

/***********************************************************************
 * Populate `out` with a formatted, units-added freq label for `freq`
 **********************************************************************/
char* gf_get_frequency_label(int freq)
{
	char *string;
	if(freq >= 1000000000) // >= 1 billion KHz (1 THz) This, ladies and gentlement, is future-proofing ;)
		asprintf(&string, "%.2f THz", (double)freq/1000000000 );
	else if(freq >= 1000000) // >= 1 million KHz (1 GHz)
		asprintf(&string, "%.2f GHz", (double)freq/1000000 );
	else if (freq >= 1000) // >= 1 thousand KHz (1 MHz)
		asprintf(&string, "%.2f MHz", (double)freq/1000 );
	else // < 1000 KHz (1 MHz)
		asprintf(&string, "%.2f KHz", (double)freq);

	debug("Prepared freq label '%s' for freq %d\n",string,freq);

	return string;
}

/***********************************************************************
 * Return freq value at index for core, as a string
 **********************************************************************/
char* gf_freqa(int core, int index)
{
	return freqs[core][index];
}

/***********************************************************************
 * Return freq value at index for core, as an int
 **********************************************************************/
int gf_freqi(int core, int index)
{
	return atoi(gf_freqa(core, index));
}

/***********************************************************************
 * Return total number of frequencies
 **********************************************************************/
unsigned int gf_number()
{
	return total_freqs;
}