aboutsummaryrefslogtreecommitdiffstats
path: root/cputemp2maxfreq.c
blob: 163917c29f296064dcad197386df63fbaff87cfa (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
#include <stdio.h>
#include <unistd.h>
#include <stdlib.h>
#include <errno.h>
#include <string.h>
#include "sysfs.h"
#include "cpufreq.h"
#include "debug.h"
#include "cputemp2maxfreq.h"
#include "failsafe.h"
#include "version.h"

struct s_cpudata cpudata;
struct s_config config={"conservative",70000,"/sys/devices/virtual/thermal/thermal_zone0/temp",100000,2000000,10};

int main()
{
 long int diff;
 long int newfreq;

 printf("Version %s, buildtime %s %s\n",VERSION_FULL,__DATE__,__TIME__);

// Print configuration
 printf("Configuration:\n");
 printf("Governor: %s\n",config.governor);
 printf("Temperature: %ld\n",config.max_temp);
 printf("Temp input: %s\n",config.temp_input);
 printf("Frequency step: %ld\n",config.freq_step);
 printf("Fallback frquency: %ld\n",config.fallback_freq);
 printf("Interval: %d\n",config.interval);

// Get and validate CPU data
 cpudata.min_freq=sysfs_read_long_int("/sys/devices/system/cpu/cpufreq/policy0/cpuinfo_min_freq");
 cpudata.max_freq=sysfs_read_long_int("/sys/devices/system/cpu/cpufreq/policy0/cpuinfo_max_freq");
 cpudata.scale_max=sysfs_read_long_int("/sys/devices/system/cpu/cpufreq/policy0/scaling_max_freq");

 printf("\nCPU data:\n");
 printf("Minimum frequency: %ld\n",cpudata.min_freq);
 printf("Maximum frequency: %ld\n",cpudata.max_freq);
 printf("Scaling maximum frequency: %ld\n",cpudata.scale_max);

 if ((cpudata.min_freq<100000) || (cpudata.min_freq>10000000) ||
     (cpudata.max_freq<100000) || (cpudata.max_freq>10000000) ||
     (cpudata.scale_max<100000) || (cpudata.scale_max>10000000))
 {
// If we have to fail now, there's not much we can do because we have no data
  printf("Invalid CPU data, exiting.\n");
  exit(1);
 }

// Set the governor
 if (cpufreq_set_str("scaling_governor",config.governor,0)!=0)
 {
  printf("Failed to set governor, error %d (%s).\n",errno,strerror(errno));

// We failed to set the governor, call the failsafe
  failsafe(1);
 }

 while(1)
 {
  cpudata.cur_freq=sysfs_read_long_int("/sys/devices/system/cpu/cpufreq/policy0/scaling_cur_freq");
  cpudata.cur_temp=sysfs_read_long_int(config.temp_input);


  DEBUG1_MAIN("Data: %ld %ld %ld %ld %ld\n",cpudata.cur_temp,config.max_temp,cpudata.max_freq,cpudata.scale_max,cpudata.cur_freq);

  if ((cpudata.cur_temp<config.max_temp) && (cpudata.scale_max<cpudata.max_freq))
  {
   diff=(config.max_temp-cpudata.cur_temp)/1000;
   newfreq=cpudata.scale_max+(config.freq_step*diff);
   if (newfreq>cpudata.max_freq) newfreq=cpudata.max_freq;
   DEBUG1_MAIN("Increase to %ld\n",newfreq);
   cpufreq_set_long_int("scaling_max_freq",newfreq,100);
   cpudata.scale_max=sysfs_read_long_int("/sys/devices/system/cpu/cpufreq/policy0/scaling_max_freq");
  }

  if ((cpudata.cur_temp>config.max_temp) && (cpudata.scale_max>cpudata.min_freq))
  {
   diff=(cpudata.cur_temp-config.max_temp)/1000;
   newfreq=cpudata.scale_max-(config.freq_step*diff);
   if (newfreq<cpudata.min_freq) newfreq=cpudata.min_freq;
   DEBUG1_MAIN("Decrease to %ld\n",newfreq);
   cpufreq_set_long_int("scaling_max_freq",newfreq,100);
   cpudata.scale_max=sysfs_read_long_int("/sys/devices/system/cpu/cpufreq/policy0/scaling_max_freq");
  }

  sleep(config.interval);
 }
}