vserver 1.9.5.x5
[linux-2.6.git] / drivers / macintosh / therm_adt746x.c
index 8feba38..cce6b36 100644 (file)
 #include <linux/spinlock.h>
 #include <linux/smp_lock.h>
 #include <linux/wait.h>
+#include <linux/suspend.h>
+#include <linux/kthread.h>
+#include <linux/moduleparam.h>
+
 #include <asm/prom.h>
 #include <asm/machdep.h>
 #include <asm/io.h>
 #include <asm/system.h>
 #include <asm/sections.h>
 #include <asm/of_device.h>
-#include <linux/kthread.h>
 
 #undef DEBUG
 
@@ -50,21 +53,26 @@ static int limit_adjust = 0;
 static int fan_speed = -1;
 
 MODULE_AUTHOR("Colin Leroy <colin@colino.net>");
-MODULE_DESCRIPTION("Driver for ADT746x thermostat in iBook G4 and Powerbook G4 Alu");
+MODULE_DESCRIPTION("Driver for ADT746x thermostat in iBook G4 and "
+                  "Powerbook G4 Alu");
 MODULE_LICENSE("GPL");
 
-MODULE_PARM(limit_adjust,"i");
-MODULE_PARM_DESC(limit_adjust,"Adjust maximum temperatures (50 cpu, 70 gpu) by N degrees.");
-MODULE_PARM(fan_speed,"i");
-MODULE_PARM_DESC(fan_speed,"Specify fan speed (0-255) when lim < temp < lim+8 (default 128)");
+module_param(limit_adjust, int, 0644);
+MODULE_PARM_DESC(limit_adjust,"Adjust maximum temperatures (50 cpu, 70 gpu) "
+                "by N degrees.");
+
+module_param(fan_speed, int, 0644);
+MODULE_PARM_DESC(fan_speed,"Specify starting fan speed (0-255) "
+                "(default 64)");
 
 struct thermostat {
        struct i2c_client       clt;
+       u8                      temps[3];
        u8                      cached_temp[3];
        u8                      initial_limits[3];
        u8                      limits[3];
        int                     last_speed[2];
-       int                     overriding[2];
+       int                     last_var[2];
 };
 
 static enum {ADT7460, ADT7467} therm_type;
@@ -73,7 +81,9 @@ static struct of_device * of_dev;
 static struct thermostat* thermostat;
 static struct task_struct *thread_therm = NULL;
 
-static int attach_one_thermostat(struct i2c_adapter *adapter, int addr, int busno);
+static int attach_one_thermostat(struct i2c_adapter *adapter, int addr,
+                                int busno);
+
 static void write_both_fan_speed(struct thermostat *th, int speed);
 static void write_fan_speed(struct thermostat *th, int speed, int fan);
 
@@ -139,10 +149,11 @@ detach_thermostat(struct i2c_adapter *adapter)
                kthread_stop(thread_therm);
        }
                
-       printk(KERN_INFO "adt746x: Putting max temperatures back from %d, %d, %d,"
-               " to %d, %d, %d\n",
+       printk(KERN_INFO "adt746x: Putting max temperatures back from "
+                        "%d, %d, %d to %d, %d, %d\n",
                th->limits[0], th->limits[1], th->limits[2],
-               th->initial_limits[0], th->initial_limits[1], th->initial_limits[2]);
+               th->initial_limits[0], th->initial_limits[1],
+               th->initial_limits[2]);
        
        for (i = 0; i < 3; i++)
                write_reg(th, LIMIT_REG[i], th->initial_limits[i]);
@@ -159,11 +170,11 @@ detach_thermostat(struct i2c_adapter *adapter)
 }
 
 static struct i2c_driver thermostat_driver = {  
-       .name           ="Apple Thermostat ADT746x",
-       .id             =0xDEAD7467,
-       .flags          =I2C_DF_NOTIFY,
-       .attach_adapter =&attach_thermostat,
-       .detach_adapter =&detach_thermostat,
+       .owner          = THIS_MODULE,
+       .name           = "therm_adt746x",
+       .flags          = I2C_DF_NOTIFY,
+       .attach_adapter = attach_thermostat,
+       .detach_adapter = detach_thermostat,
 };
 
 static int read_fan_speed(struct thermostat *th, u8 addr)
@@ -201,11 +212,11 @@ static void write_fan_speed(struct thermostat *th, int speed, int fan)
        
        if (th->last_speed[fan] != speed) {
                if (speed == -1)
-                       printk(KERN_INFO "adt746x: Setting speed to: automatic for %s fan.\n",
-                               fan?"GPU":"CPU");
+                       printk(KERN_DEBUG "adt746x: Setting speed to automatic "
+                               "for %s fan.\n", fan?"GPU":"CPU");
                else
-                       printk(KERN_INFO "adt746x: Setting speed to: %d for %s fan.\n",
-                               speed, fan?"GPU":"CPU");
+                       printk(KERN_DEBUG "adt746x: Setting speed to %d "
+                               "for %s fan.\n", speed, fan?"GPU":"CPU");
        } else
                return;
        
@@ -216,8 +227,11 @@ static void write_fan_speed(struct thermostat *th, int speed, int fan)
        } else {
                /* back to automatic */
                if(therm_type == ADT7460) {
-                       manual = read_reg(th, MANUAL_MODE[fan]) & (~MANUAL_MASK);
-                       write_reg(th, MANUAL_MODE[fan], manual|REM_CONTROL[fan]);
+                       manual = read_reg(th,
+                               MANUAL_MODE[fan]) & (~MANUAL_MASK);
+
+                       write_reg(th,
+                               MANUAL_MODE[fan], manual|REM_CONTROL[fan]);
                } else {
                        manual = read_reg(th, MANUAL_MODE[fan]);
                        write_reg(th, MANUAL_MODE[fan], manual&(~AUTO_MASK));
@@ -227,97 +241,117 @@ static void write_fan_speed(struct thermostat *th, int speed, int fan)
        th->last_speed[fan] = speed;                    
 }
 
-static int monitor_task(void *arg)
+static void read_sensors(struct thermostat *th)
 {
-       struct thermostat* th = arg;
-       u8 temps[3];
-       u8 lims[3];
-       int i;
+       int i = 0;
+
+       for (i = 0; i < 3; i++)
+               th->temps[i]  = read_reg(th, TEMP_REG[i]);
+}
+
 #ifdef DEBUG
-       int mfan_speed;
+static void display_stats(struct thermostat *th)
+{
+       if (th->temps[0] != th->cached_temp[0]
+       ||  th->temps[1] != th->cached_temp[1]
+       ||  th->temps[2] != th->cached_temp[2]) {
+               printk(KERN_INFO "adt746x: Temperature infos:"
+                                " thermostats: %d,%d,%d;"
+                                " limits: %d,%d,%d;"
+                                " fan speed: %d RPM\n",
+                                th->temps[0], th->temps[1], th->temps[2],
+                                th->limits[0],  th->limits[1],  th->limits[2],
+                                read_fan_speed(th, FAN_SPEED[0]));
+       }
+       th->cached_temp[0] = th->temps[0];
+       th->cached_temp[1] = th->temps[1];
+       th->cached_temp[2] = th->temps[2];
+}
 #endif
-       while(!kthread_should_stop())
-       {
-               msleep_interruptible(2000);
 
-               /* Check status */
-               /* local   : chip */
-               /* remote 1: CPU ?*/
-               /* remote 2: GPU ?*/
-#ifndef DEBUG
-               if (fan_speed != -1) {
-#endif
-                       for (i = 0; i < 3; i++) {
-                               temps[i]  = read_reg(th, TEMP_REG[i]);
-                               lims[i]   = th->limits[i];
+static void update_fans_speed (struct thermostat *th)
+{
+       int lastvar = 0; /* last variation, for iBook */
+       int i = 0;
+
+       /* we don't care about local sensor, so we start at sensor 1 */
+       for (i = 1; i < 3; i++) {
+               int started = 0;
+               int fan_number = (therm_type == ADT7460 && i == 2);
+               int var = th->temps[i] - th->limits[i];
+
+               if (var > -1) {
+                       int step = (255 - fan_speed) / 7;
+                       int new_speed = 0;
+
+                       /* hysteresis : change fan speed only if variation is
+                        * more than two degrees */
+                       if (abs(var - th->last_var[fan_number]) < 2)
+                               continue;
+
+                       started = 1;
+                       new_speed = fan_speed + ((var-1)*step);
+
+                       if (new_speed < fan_speed)
+                               new_speed = fan_speed;
+                       if (new_speed > 255)
+                               new_speed = 255;
+
+                       printk(KERN_DEBUG "adt746x: setting fans speed to %d "
+                                        "(limit exceeded by %d on %s) \n",
+                                       new_speed, var,
+                                       fan_number?"GPU/pwr":"CPU");
+                       write_both_fan_speed(th, new_speed);
+                       th->last_var[fan_number] = var;
+               } else if (var < -2) {
+                       /* don't stop fan if GPU/power is cold and CPU is not
+                        * so cold (lastvar >= -1) */
+                       if (i == 2 && lastvar < -1) {
+                               if (th->last_speed[fan_number] != 0)
+                                       printk(KERN_DEBUG "adt746x: Stopping "
+                                               "fans.\n");
+                               write_both_fan_speed(th, 0);
                        }
-#ifndef DEBUG
                }
+
+               lastvar = var;
+
+               if (started)
+                       return; /* we don't want to re-stop the fan
+                               * if CPU is heating and GPU/power is not */
+       }
+}
+
+static int monitor_task(void *arg)
+{
+       struct thermostat* th = arg;
+
+       while(!kthread_should_stop()) {
+               if (current->flags & PF_FREEZE)
+                       refrigerator(PF_FREEZE);
+
+               msleep_interruptible(2000);
+
+#ifndef DEBUG
+               if (fan_speed != -1)
+                       read_sensors(th);
+#else
+               read_sensors(th);
 #endif         
-               if (fan_speed != -1) {
-                       int lastvar = 0;                /* for iBook */
-                       for (i = 1; i < 3; i++) {       /* we don't care about local sensor */
-                               int started = 0;
-                               int fan_number = (therm_type == ADT7460 && i == 2);
-                               int var = temps[i] - lims[i];
-                               if (var > 8) {
-                                       if (th->overriding[fan_number] == 0)
-                                               printk(KERN_INFO "adt746x: Limit exceeded by %d, overriding specified fan speed for %s.\n",
-                                                       var, fan_number?"GPU":"CPU");
-                                       th->overriding[fan_number] = 1;
-                                       write_fan_speed(th, 255, fan_number);
-                                       started = 1;
-                               } else if ((!th->overriding[fan_number] || var < 6) && var > 0) {
-                                       if (th->overriding[fan_number] == 1)
-                                               printk(KERN_INFO "adt746x: Limit exceeded by %d, setting speed to specified for %s.\n",
-                                                       var, fan_number?"GPU":"CPU");                                   
-                                       th->overriding[fan_number] = 0;
-                                       write_fan_speed(th, fan_speed, fan_number);
-                                       started = 1;
-                               } else if (var < -1) {
-                                       /* don't stop iBook fan if GPU is cold and CPU is not
-                                        * so cold (lastvar >= -1) */
-                                       if (therm_type == ADT7460 || lastvar < -1 || i == 1) {
-                                               if (th->last_speed[fan_number] != 0)
-                                                       printk(KERN_INFO "adt746x: Stopping %s fan.\n",
-                                                               fan_number?"GPU":"CPU");
-                                               write_fan_speed(th, 0, fan_number);
-                                       }
-                               }
-                               
-                               lastvar = var;
-                               
-                               if (started && therm_type == ADT7467)
-                                       break; /* we don't want to re-stop the fan
-                                               * if CPU is heating and GPU is not */
-                       }
-               }
+
+               if (fan_speed != -1)
+                       update_fans_speed(th);
+
 #ifdef DEBUG
-               mfan_speed = read_fan_speed(th, FAN_SPEED[0]);
-               /* only one fan in the iBook G4 */
-                               
-               if (temps[0] != th->cached_temp[0]
-               ||  temps[1] != th->cached_temp[1]
-               ||  temps[2] != th->cached_temp[2]) {
-                       printk(KERN_INFO "adt746x: Temperature infos:"
-                                        " thermostats: %d,%d,%d;"
-                                        " limits: %d,%d,%d;"
-                                        " fan speed: %d RPM\n",
-                               temps[0], temps[1], temps[2],
-                               lims[0],  lims[1],  lims[2],
-                               mfan_speed);
-               }
-               th->cached_temp[0] = temps[0];
-               th->cached_temp[1] = temps[1];
-               th->cached_temp[2] = temps[2];
-#endif         
+               display_stats(th);
+#endif
+
        }
 
        return 0;
 }
 
-static void
-set_limit(struct thermostat *th, int i)
+static void set_limit(struct thermostat *th, int i)
 {
                /* Set CPU limit higher to avoid powerdowns */ 
                th->limits[i] = default_limits_chip[i] + limit_adjust;
@@ -326,9 +360,9 @@ set_limit(struct thermostat *th, int i)
                /* set our limits to normal */
                th->limits[i] = default_limits_local[i] + limit_adjust;
 }
-       
-static int
-attach_one_thermostat(struct i2c_adapter *adapter, int addr, int busno)
+
+static int attach_one_thermostat(struct i2c_adapter *adapter, int addr,
+                                int busno)
 {
        struct thermostat* th;
        int rc;
@@ -336,27 +370,31 @@ attach_one_thermostat(struct i2c_adapter *adapter, int addr, int busno)
 
        if (thermostat)
                return 0;
-       th = (struct thermostat *)kmalloc(sizeof(struct thermostat), GFP_KERNEL);
+
+       th = (struct thermostat *)
+               kmalloc(sizeof(struct thermostat), GFP_KERNEL);
+
        if (!th)
                return -ENOMEM;
+
        memset(th, 0, sizeof(*th));
        th->clt.addr = addr;
        th->clt.adapter = adapter;
        th->clt.driver = &thermostat_driver;
-       th->clt.id = 0xDEAD7467;
        strcpy(th->clt.name, "thermostat");
 
        rc = read_reg(th, 0);
        if (rc < 0) {
-               printk(KERN_ERR "adt746x: Thermostat failed to read config from bus %d !\n",
-                       busno);
+               printk(KERN_ERR "adt746x: Thermostat failed to read config "
+                               "from bus %d !\n",
+                               busno);
                kfree(th);
                return -ENODEV;
        }
+
        /* force manual control to start the fan quieter */
-       
        if (fan_speed == -1)
-               fan_speed=128;
+               fan_speed = 64;
        
        if(therm_type == ADT7460) {
                printk(KERN_INFO "adt746x: ADT7460 initializing\n");
@@ -371,14 +409,16 @@ attach_one_thermostat(struct i2c_adapter *adapter, int addr, int busno)
        }
        
        printk(KERN_INFO "adt746x: Lowering max temperatures from %d, %d, %d"
-               " to %d, %d, %d\n",
-               th->initial_limits[0], th->initial_limits[1], th->initial_limits[2], 
-               th->limits[0], th->limits[1], th->limits[2]);
+                        " to %d, %d, %d\n",
+                        th->initial_limits[0], th->initial_limits[1],
+                        th->initial_limits[2], th->limits[0], th->limits[1],
+                        th->limits[2]);
 
        thermostat = th;
 
        if (i2c_attach_client(&th->clt)) {
-               printk(KERN_INFO "adt746x: Thermostat failed to attach client !\n");
+               printk(KERN_INFO "adt746x: Thermostat failed to attach "
+                                "client !\n");
                thermostat = NULL;
                kfree(th);
                return -ENODEV;
@@ -387,10 +427,14 @@ attach_one_thermostat(struct i2c_adapter *adapter, int addr, int busno)
        /* be sure to really write fan speed the first time */
        th->last_speed[0] = -2;
        th->last_speed[1] = -2;
-       
+       th->last_var[0] = -80;
+       th->last_var[1] = -80;
+
        if (fan_speed != -1) {
+               /* manual mode, stop fans */
                write_both_fan_speed(th, 0);
        } else {
+               /* automatic mode */
                write_both_fan_speed(th, -1);
        }
        
@@ -507,8 +551,9 @@ thermostat_init(void)
        therm_bus = ((*prop) >> 8) & 0x0f;
        therm_address = ((*prop) & 0xff) >> 1;
 
-       printk(KERN_INFO "adt746x: Thermostat bus: %d, address: 0x%02x, limit_adjust: %d, fan_speed: %d\n",
-               therm_bus, therm_address, limit_adjust, fan_speed);
+       printk(KERN_INFO "adt746x: Thermostat bus: %d, address: 0x%02x, "
+                        "limit_adjust: %d, fan_speed: %d\n",
+                        therm_bus, therm_address, limit_adjust, fan_speed);
 
        of_dev = of_platform_device_create(np, "temperatures");
        
@@ -545,8 +590,11 @@ thermostat_exit(void)
                device_remove_file(&of_dev->dev, &dev_attr_limit_adjust);
                device_remove_file(&of_dev->dev, &dev_attr_specified_fan_speed);
                device_remove_file(&of_dev->dev, &dev_attr_cpu_fan_speed);
+
                if(therm_type == ADT7460)
-                       device_remove_file(&of_dev->dev, &dev_attr_gpu_fan_speed);
+                       device_remove_file(&of_dev->dev,
+                                          &dev_attr_gpu_fan_speed);
+
                of_device_unregister(of_dev);
        }
        i2c_del_driver(&thermostat_driver);