mbpfan/src/mbpfan.c

350 lines
8.6 KiB
C
Raw Normal View History

2012-06-09 14:25:28 +00:00
/**
* mbpfan.c - automatically control fan for MacBook Pro
* Copyright (C) 2010 Allan McRae <allan@archlinux.org>
* Modifications by Rafael Vega <rvega@elsoftwarehamuerto.org>
* Modifications (2012) by Daniel Graziotin <dgraziotin@task3.cc>
*
* This program 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.
*
* This program 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.
*
* 2012-06-09 - v1.2
*
* Notes:
* Assumes any number of processors and fans (max. 10)
2012-06-15 19:07:11 +00:00
* It uses only the temperatures from the processors as input.
2012-06-09 14:25:28 +00:00
* Requires coretemp and applesmc kernel modules to be loaded.
* Requires root use
*
* Tested models:
* MacBook Pro 8.1 13" (Intel i7 - Linux 3.2)
* Macbook Pro 6,2 15" (Intel i7 - Linux 3.2)
*/
#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <string.h>
2012-06-11 19:43:53 +00:00
#include <math.h>
#include <syslog.h>
2012-06-09 14:25:28 +00:00
#include "mbpfan.h"
2012-06-11 19:43:53 +00:00
#include "global.h"
2012-06-09 14:25:28 +00:00
/* lazy min/max... */
#define min(a,b) a < b ? a : b
#define max(a,b) a > b ? a : b
int min_fan_speed = 2000;
int max_fan_speed = 6200;
2012-06-09 14:25:28 +00:00
/* temperature thresholds
* low_temp - temperature below which fan speed will be at minimum
* high_temp - fan will increase speed when higher than this temperature
* max_temp - fan will run at full speed above this temperature */
int low_temp = 63; // try ranges 55-63
int high_temp = 66; // try ranges 58-66
int max_temp = 86; // do not set it > 90
2012-06-09 14:25:28 +00:00
2012-06-15 19:47:23 +00:00
int polling_interval = 7;
2012-06-09 14:25:28 +00:00
2012-06-16 09:02:48 +00:00
struct s_sensors
{
char* path;
char* fan_min_path;
char* fan_man_path;
unsigned int temperature;
struct s_sensors *next;
2012-06-09 14:25:28 +00:00
};
2012-06-11 19:43:53 +00:00
t_sensors *retrieve_sensors()
2012-06-09 14:25:28 +00:00
{
2012-06-16 09:02:48 +00:00
t_sensors *sensors_head = NULL;
t_sensors *s = NULL;
char *path = NULL;
const char *path_begin = "/sys/devices/platform/coretemp.0/temp";
const char *path_end = "_input";
int path_size = strlen(path_begin) + strlen(path_end) + 2;
char number[1];
sprintf(number,"%d",0);
int i = 0;
for(i = 0; i<10; i++)
{
path = (char*) malloc(sizeof( char ) * path_size);
sprintf(number,"%d",i);
path[0] = '\0';
strncat( path, path_begin, strlen(path_begin) );
strncat( path, number, strlen(number) );
strncat( path, path_end, strlen(path_begin) );
FILE *file = fopen(path, "r");
if(file != NULL)
{
s = (t_sensors *) malloc( sizeof( t_sensors ) );
s->path = (char *) malloc(sizeof( char ) * path_size);
strcpy(s->path, path);
fscanf(file, "%d", &s->temperature);
if (sensors_head == NULL)
{
sensors_head = s;
sensors_head->next = NULL;
}
else
{
t_sensors *tmp = sensors_head;
while (tmp->next != NULL)
{
tmp = tmp->next;
2012-06-09 14:25:28 +00:00
}
2012-06-16 09:02:48 +00:00
tmp->next = s;
tmp->next->next = NULL;
}
fclose(file);
2012-06-09 14:25:28 +00:00
}
2012-06-16 09:02:48 +00:00
free(path);
path = NULL;
}
if(sensors_head != NULL)
find_fans(sensors_head);
return sensors_head;
2012-06-09 14:25:28 +00:00
}
void find_fans(t_sensors* sensors)
{
2012-06-16 09:02:48 +00:00
t_sensors *tmp = sensors;
char *path_min = NULL;
char *path_man = NULL;
const char *path_begin = "/sys/devices/platform/applesmc.768/fan";
const char *path_min_end = "_min";
const char *path_man_end = "_manual";
int path_min_size = strlen(path_begin) + strlen(path_min_end) + 2;
int path_man_size = strlen(path_begin) + strlen(path_man_end) + 2;
char number[1];
sprintf(number,"%d",0);
int n_sensors = 0;
int n_fans = 0;
for(n_sensors = 0; n_sensors<10; n_sensors++)
{
path_min = (char*) malloc(sizeof( char ) * path_min_size);
path_min[0] = '\0';
path_man = (char*) malloc(sizeof( char ) * path_man_size);
path_man[0] = '\0';
sprintf(number,"%d",n_sensors);
strncat( path_min, path_begin, strlen(path_begin) );
strncat( path_min, number, strlen(number) );
strncat( path_min, path_min_end, strlen(path_begin) );
strncat( path_man, path_begin, strlen(path_begin) );
strncat( path_man, number, strlen(number) );
strncat( path_man, path_man_end, strlen(path_begin) );
FILE *file = fopen(path_min, "r");
if(file != NULL)
{
if (tmp->path != NULL)
{
tmp->fan_min_path = (char *) malloc(sizeof( char ) * path_min_size);
tmp->fan_man_path = (char *) malloc(sizeof( char ) * path_man_size);
}
strcpy(tmp->fan_min_path, path_min);
strcpy(tmp->fan_man_path, path_man);
tmp = tmp->next;
n_fans++;
fclose(file);
}
2012-06-16 09:02:48 +00:00
}
if(verbose)
{
printf("Found %d: sensors and %d fans\n", n_sensors, n_fans);
if(daemonize)
{
syslog(LOG_INFO, "Found %d: sensors and %d fans", n_sensors, n_fans);
2012-06-11 19:43:53 +00:00
}
2012-06-16 09:02:48 +00:00
}
2012-06-11 19:43:53 +00:00
2012-06-16 09:02:48 +00:00
free(path_min);
path_min = NULL;
free(path_man);
path_man = NULL;
}
void set_fans_man(t_sensors *sensors)
{
2012-06-16 09:02:48 +00:00
t_sensors *tmp = sensors;
FILE *file;
while(tmp != NULL)
{
file = fopen(tmp->fan_man_path, "rw+");
if(file != NULL)
{
fprintf(file, "%d", 0);
fclose(file);
}
2012-06-16 09:02:48 +00:00
tmp = tmp->next;
}
}
2012-06-09 14:25:28 +00:00
t_sensors *refresh_sensors(t_sensors *sensors)
{
2012-06-16 09:02:48 +00:00
t_sensors *tmp = sensors;
2012-06-09 14:25:28 +00:00
2012-06-16 09:02:48 +00:00
while(tmp != NULL)
{
FILE *file = fopen(tmp->path, "r");
2012-06-09 14:25:28 +00:00
2012-06-16 09:02:48 +00:00
if(file != NULL)
{
fscanf(file, "%d", &tmp->temperature);
fclose(file);
2012-06-09 14:25:28 +00:00
}
2012-06-16 09:02:48 +00:00
tmp = tmp->next;
}
return sensors;
2012-06-09 14:25:28 +00:00
}
/* Controls the speed of the fan */
void set_fan_speed(t_sensors* sensors, int speed)
2012-06-09 14:25:28 +00:00
{
2012-06-16 09:02:48 +00:00
t_sensors *tmp = sensors;
FILE *file;
while(tmp != NULL)
{
file = fopen(tmp->fan_min_path, "rw+");
if(file != NULL)
{
fprintf(file, "%d", speed);
fclose(file);
2012-06-09 14:25:28 +00:00
}
2012-06-16 09:02:48 +00:00
tmp = tmp->next;
}
2012-06-09 14:25:28 +00:00
}
unsigned short get_temp(t_sensors* sensors)
{
2012-06-16 09:02:48 +00:00
sensors = refresh_sensors(sensors);
int sum_temp = 0;
unsigned short temp = 0;
t_sensors* tmp = sensors;
while(tmp != NULL)
{
sum_temp += tmp->temperature;
tmp = tmp->next;
}
temp = (unsigned short)( ceil( (float)( sum_temp ) / 2000. ) );
return temp;
2012-06-09 14:25:28 +00:00
}
2012-06-11 19:43:53 +00:00
void mbpfan()
2012-06-09 14:25:28 +00:00
{
2012-06-16 09:02:48 +00:00
int old_temp, new_temp, fan_speed, steps;
int temp_change;
int step_up, step_down;
t_sensors* sensors = retrieve_sensors();
set_fans_man(sensors);
new_temp = get_temp(sensors);
fan_speed = 2000;
set_fan_speed(sensors, fan_speed);
if(verbose)
{
printf("Sleeping for %d seconds\n", polling_interval);
if(daemonize)
{
syslog(LOG_INFO, "Sleeping for %d seconds\n", polling_interval);
2012-06-11 19:43:53 +00:00
}
2012-06-16 09:02:48 +00:00
}
sleep(polling_interval);
2012-06-09 14:25:28 +00:00
2012-06-16 09:02:48 +00:00
step_up = (float)( max_fan_speed - min_fan_speed ) /
(float)( ( max_temp - high_temp ) * ( max_temp - high_temp + 1 ) / 2 );
2012-06-09 14:25:28 +00:00
2012-06-16 09:02:48 +00:00
step_down = (float)( max_fan_speed - min_fan_speed ) /
(float)( ( max_temp - low_temp ) * ( max_temp - low_temp + 1 ) / 2 );
2012-06-09 14:25:28 +00:00
2012-06-16 09:02:48 +00:00
while(1)
{
old_temp = new_temp;
new_temp = get_temp(sensors);
2012-06-09 14:25:28 +00:00
2012-06-16 09:02:48 +00:00
if(new_temp >= max_temp && fan_speed != max_fan_speed)
{
fan_speed = max_fan_speed;
}
2012-06-09 14:25:28 +00:00
2012-06-16 09:02:48 +00:00
if(new_temp <= low_temp && fan_speed != min_fan_speed)
{
fan_speed = min_fan_speed;
}
2012-06-09 14:25:28 +00:00
2012-06-16 09:02:48 +00:00
temp_change = new_temp - old_temp;
2012-06-09 14:25:28 +00:00
2012-06-16 09:02:48 +00:00
if(temp_change > 0 && new_temp > high_temp && new_temp < max_temp)
{
steps = ( new_temp - high_temp ) * ( new_temp - high_temp + 1 ) / 2;
fan_speed = max( fan_speed, ceil(min_fan_speed + steps * step_up) );
}
2012-06-09 14:25:28 +00:00
2012-06-16 09:02:48 +00:00
if(temp_change < 0 && new_temp > low_temp && new_temp < max_temp)
{
steps = ( max_temp - new_temp ) * ( max_temp - new_temp + 1 ) / 2;
fan_speed = min( fan_speed, floor(max_fan_speed - steps * step_down) );
}
2012-06-09 14:25:28 +00:00
2012-06-16 09:02:48 +00:00
if(verbose)
{
printf("Old Temp %d: New Temp: %d, Fan Speed: %d\n", old_temp, new_temp, fan_speed);
if(daemonize)
{
syslog(LOG_INFO, "Old Temp %d: New Temp: %d, Fan Speed: %d\n", old_temp, new_temp, fan_speed);
}
}
2012-06-11 19:43:53 +00:00
2012-06-16 09:02:48 +00:00
set_fan_speed(sensors, fan_speed);
2012-06-09 14:25:28 +00:00
2012-06-16 09:02:48 +00:00
if(verbose)
{
printf("Sleeping for %d seconds\n", polling_interval);
if(daemonize)
{
syslog(LOG_INFO, "Sleeping for %d seconds\n", polling_interval);
}
2012-06-09 14:25:28 +00:00
}
2012-06-16 09:02:48 +00:00
sleep(polling_interval);
}
2012-06-09 14:25:28 +00:00
}