processor_thermal.c 9.3 KB
Newer Older
L
Linus Torvalds 已提交
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
/*
 * processor_thermal.c - Passive cooling submodule of the ACPI processor driver
 *
 *  Copyright (C) 2001, 2002 Andy Grover <andrew.grover@intel.com>
 *  Copyright (C) 2001, 2002 Paul Diefenbaugh <paul.s.diefenbaugh@intel.com>
 *  Copyright (C) 2004       Dominik Brodowski <linux@brodo.de>
 *  Copyright (C) 2004  Anil S Keshavamurthy <anil.s.keshavamurthy@intel.com>
 *  			- Added processor hotplug support
 *
 * ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
 *
 *  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 2 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.
 *
 *  You should have received a copy of the GNU General Public License along
 *  with this program; if not, write to the Free Software Foundation, Inc.,
 *  59 Temple Place, Suite 330, Boston, MA 02111-1307 USA.
 *
 * ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
 */

#include <linux/kernel.h>
#include <linux/module.h>
#include <linux/init.h>
#include <linux/cpufreq.h>
#include <linux/proc_fs.h>
#include <linux/seq_file.h>

#include <asm/uaccess.h>

#include <acpi/acpi_bus.h>
#include <acpi/processor.h>
#include <acpi/acpi_drivers.h>

#define ACPI_PROCESSOR_COMPONENT        0x01000000
#define ACPI_PROCESSOR_CLASS            "processor"
#define ACPI_PROCESSOR_DRIVER_NAME      "ACPI Processor Driver"
#define _COMPONENT              ACPI_PROCESSOR_COMPONENT
L
Len Brown 已提交
46
ACPI_MODULE_NAME("acpi_processor")
L
Linus Torvalds 已提交
47 48 49 50

/* --------------------------------------------------------------------------
                                 Limit Interface
   -------------------------------------------------------------------------- */
L
Len Brown 已提交
51
static int acpi_processor_apply_limit(struct acpi_processor *pr)
L
Linus Torvalds 已提交
52
{
L
Len Brown 已提交
53 54 55
	int result = 0;
	u16 px = 0;
	u16 tx = 0;
L
Linus Torvalds 已提交
56 57 58


	if (!pr)
59
		return -EINVAL;
L
Linus Torvalds 已提交
60 61

	if (!pr->flags.limit)
62
		return -ENODEV;
L
Linus Torvalds 已提交
63 64 65 66 67 68 69 70 71 72 73 74 75 76 77

	if (pr->flags.throttling) {
		if (pr->limit.user.tx > tx)
			tx = pr->limit.user.tx;
		if (pr->limit.thermal.tx > tx)
			tx = pr->limit.thermal.tx;

		result = acpi_processor_set_throttling(pr, tx);
		if (result)
			goto end;
	}

	pr->limit.state.px = px;
	pr->limit.state.tx = tx;

L
Len Brown 已提交
78 79 80
	ACPI_DEBUG_PRINT((ACPI_DB_INFO,
			  "Processor [%d] limit set to (P%d:T%d)\n", pr->id,
			  pr->limit.state.px, pr->limit.state.tx));
L
Linus Torvalds 已提交
81

L
Len Brown 已提交
82
      end:
L
Linus Torvalds 已提交
83
	if (result)
84
		printk(KERN_ERR PREFIX "Unable to set limit\n");
L
Linus Torvalds 已提交
85

86
	return result;
L
Linus Torvalds 已提交
87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102
}

#ifdef CONFIG_CPU_FREQ

/* If a passive cooling situation is detected, primarily CPUfreq is used, as it
 * offers (in most cases) voltage scaling in addition to frequency scaling, and
 * thus a cubic (instead of linear) reduction of energy. Also, we allow for
 * _any_ cpufreq driver and not only the acpi-cpufreq driver.
 */

static unsigned int cpufreq_thermal_reduction_pctg[NR_CPUS];
static unsigned int acpi_thermal_cpufreq_is_init = 0;

static int cpu_has_cpufreq(unsigned int cpu)
{
	struct cpufreq_policy policy;
103
	if (!acpi_thermal_cpufreq_is_init || cpufreq_get_policy(&policy, cpu))
104 105
		return 0;
	return 1;
L
Linus Torvalds 已提交
106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126
}

static int acpi_thermal_cpufreq_increase(unsigned int cpu)
{
	if (!cpu_has_cpufreq(cpu))
		return -ENODEV;

	if (cpufreq_thermal_reduction_pctg[cpu] < 60) {
		cpufreq_thermal_reduction_pctg[cpu] += 20;
		cpufreq_update_policy(cpu);
		return 0;
	}

	return -ERANGE;
}

static int acpi_thermal_cpufreq_decrease(unsigned int cpu)
{
	if (!cpu_has_cpufreq(cpu))
		return -ENODEV;

127
	if (cpufreq_thermal_reduction_pctg[cpu] > 20)
L
Linus Torvalds 已提交
128
		cpufreq_thermal_reduction_pctg[cpu] -= 20;
129 130 131 132 133
	else
		cpufreq_thermal_reduction_pctg[cpu] = 0;
	cpufreq_update_policy(cpu);
	/* We reached max freq again and can leave passive mode */
	return !cpufreq_thermal_reduction_pctg[cpu];
L
Linus Torvalds 已提交
134 135
}

L
Len Brown 已提交
136 137
static int acpi_thermal_cpufreq_notifier(struct notifier_block *nb,
					 unsigned long event, void *data)
L
Linus Torvalds 已提交
138 139 140 141 142 143 144
{
	struct cpufreq_policy *policy = data;
	unsigned long max_freq = 0;

	if (event != CPUFREQ_ADJUST)
		goto out;

L
Len Brown 已提交
145 146 147
	max_freq =
	    (policy->cpuinfo.max_freq *
	     (100 - cpufreq_thermal_reduction_pctg[policy->cpu])) / 100;
L
Linus Torvalds 已提交
148 149 150

	cpufreq_verify_within_limits(policy, 0, max_freq);

L
Len Brown 已提交
151
      out:
L
Linus Torvalds 已提交
152 153 154 155 156 157 158
	return 0;
}

static struct notifier_block acpi_thermal_cpufreq_notifier_block = {
	.notifier_call = acpi_thermal_cpufreq_notifier,
};

L
Len Brown 已提交
159 160
void acpi_thermal_cpufreq_init(void)
{
L
Linus Torvalds 已提交
161 162
	int i;

L
Len Brown 已提交
163
	for (i = 0; i < NR_CPUS; i++)
L
Linus Torvalds 已提交
164 165
		cpufreq_thermal_reduction_pctg[i] = 0;

L
Len Brown 已提交
166 167
	i = cpufreq_register_notifier(&acpi_thermal_cpufreq_notifier_block,
				      CPUFREQ_POLICY_NOTIFIER);
L
Linus Torvalds 已提交
168 169 170 171
	if (!i)
		acpi_thermal_cpufreq_is_init = 1;
}

L
Len Brown 已提交
172 173
void acpi_thermal_cpufreq_exit(void)
{
L
Linus Torvalds 已提交
174
	if (acpi_thermal_cpufreq_is_init)
L
Len Brown 已提交
175 176 177
		cpufreq_unregister_notifier
		    (&acpi_thermal_cpufreq_notifier_block,
		     CPUFREQ_POLICY_NOTIFIER);
L
Linus Torvalds 已提交
178 179 180 181

	acpi_thermal_cpufreq_is_init = 0;
}

L
Len Brown 已提交
182
#else				/* ! CONFIG_CPU_FREQ */
L
Linus Torvalds 已提交
183

L
Len Brown 已提交
184 185 186 187 188 189 190 191
static int acpi_thermal_cpufreq_increase(unsigned int cpu)
{
	return -ENODEV;
}
static int acpi_thermal_cpufreq_decrease(unsigned int cpu)
{
	return -ENODEV;
}
L
Linus Torvalds 已提交
192 193 194

#endif

L
Len Brown 已提交
195
int acpi_processor_set_thermal_limit(acpi_handle handle, int type)
L
Linus Torvalds 已提交
196
{
L
Len Brown 已提交
197 198 199
	int result = 0;
	struct acpi_processor *pr = NULL;
	struct acpi_device *device = NULL;
200
	int tx = 0, max_tx_px = 0;
L
Linus Torvalds 已提交
201 202 203


	if ((type < ACPI_PROCESSOR_LIMIT_NONE)
L
Len Brown 已提交
204
	    || (type > ACPI_PROCESSOR_LIMIT_DECREMENT))
205
		return -EINVAL;
L
Linus Torvalds 已提交
206 207 208

	result = acpi_bus_get_device(handle, &device);
	if (result)
209
		return result;
L
Linus Torvalds 已提交
210

211
	pr = acpi_driver_data(device);
L
Linus Torvalds 已提交
212
	if (!pr)
213
		return -ENODEV;
L
Linus Torvalds 已提交
214 215 216 217 218 219 220 221 222 223 224 225 226 227 228 229 230 231 232 233 234 235 236 237 238 239 240 241 242

	/* Thermal limits are always relative to the current Px/Tx state. */
	if (pr->flags.throttling)
		pr->limit.thermal.tx = pr->throttling.state;

	/*
	 * Our default policy is to only use throttling at the lowest
	 * performance state.
	 */

	tx = pr->limit.thermal.tx;

	switch (type) {

	case ACPI_PROCESSOR_LIMIT_NONE:
		do {
			result = acpi_thermal_cpufreq_decrease(pr->id);
		} while (!result);
		tx = 0;
		break;

	case ACPI_PROCESSOR_LIMIT_INCREMENT:
		/* if going up: P-states first, T-states later */

		result = acpi_thermal_cpufreq_increase(pr->id);
		if (!result)
			goto end;
		else if (result == -ERANGE)
			ACPI_DEBUG_PRINT((ACPI_DB_INFO,
L
Len Brown 已提交
243
					  "At maximum performance state\n"));
L
Linus Torvalds 已提交
244 245 246 247

		if (pr->flags.throttling) {
			if (tx == (pr->throttling.state_count - 1))
				ACPI_DEBUG_PRINT((ACPI_DB_INFO,
L
Len Brown 已提交
248
						  "At maximum throttling state\n"));
L
Linus Torvalds 已提交
249 250 251 252 253 254 255 256 257
			else
				tx++;
		}
		break;

	case ACPI_PROCESSOR_LIMIT_DECREMENT:
		/* if going down: T-states first, P-states later */

		if (pr->flags.throttling) {
258 259
			if (tx == 0) {
				max_tx_px = 1;
L
Linus Torvalds 已提交
260
				ACPI_DEBUG_PRINT((ACPI_DB_INFO,
L
Len Brown 已提交
261
						  "At minimum throttling state\n"));
262
			} else {
L
Linus Torvalds 已提交
263 264 265 266 267 268
				tx--;
				goto end;
			}
		}

		result = acpi_thermal_cpufreq_decrease(pr->id);
269 270 271 272 273
		if (result) {
			/*
			 * We only could get -ERANGE, 1 or 0.
			 * In the first two cases we reached max freq again.
			 */
L
Linus Torvalds 已提交
274
			ACPI_DEBUG_PRINT((ACPI_DB_INFO,
L
Len Brown 已提交
275
					  "At minimum performance state\n"));
276 277 278
			max_tx_px = 1;
		} else
			max_tx_px = 0;
L
Linus Torvalds 已提交
279 280 281 282

		break;
	}

L
Len Brown 已提交
283
      end:
L
Linus Torvalds 已提交
284 285 286 287 288 289
	if (pr->flags.throttling) {
		pr->limit.thermal.px = 0;
		pr->limit.thermal.tx = tx;

		result = acpi_processor_apply_limit(pr);
		if (result)
290
			printk(KERN_ERR PREFIX "Unable to set thermal limit\n");
L
Linus Torvalds 已提交
291 292

		ACPI_DEBUG_PRINT((ACPI_DB_INFO, "Thermal limit now (P%d:T%d)\n",
L
Len Brown 已提交
293
				  pr->limit.thermal.px, pr->limit.thermal.tx));
L
Linus Torvalds 已提交
294 295
	} else
		result = 0;
296
	if (max_tx_px)
297
		return 1;
298
	else
299
		return result;
L
Linus Torvalds 已提交
300 301
}

L
Len Brown 已提交
302
int acpi_processor_get_limit_info(struct acpi_processor *pr)
L
Linus Torvalds 已提交
303 304 305
{

	if (!pr)
306
		return -EINVAL;
L
Linus Torvalds 已提交
307 308 309 310

	if (pr->flags.throttling)
		pr->flags.limit = 1;

311
	return 0;
L
Linus Torvalds 已提交
312 313 314 315 316 317
}

/* /proc interface */

static int acpi_processor_limit_seq_show(struct seq_file *seq, void *offset)
{
L
Len Brown 已提交
318
	struct acpi_processor *pr = (struct acpi_processor *)seq->private;
L
Linus Torvalds 已提交
319 320 321 322 323 324 325 326 327 328 329


	if (!pr)
		goto end;

	if (!pr->flags.limit) {
		seq_puts(seq, "<not supported>\n");
		goto end;
	}

	seq_printf(seq, "active limit:            P%d:T%d\n"
L
Len Brown 已提交
330 331 332 333 334
		   "user limit:              P%d:T%d\n"
		   "thermal limit:           P%d:T%d\n",
		   pr->limit.state.px, pr->limit.state.tx,
		   pr->limit.user.px, pr->limit.user.tx,
		   pr->limit.thermal.px, pr->limit.thermal.tx);
L
Linus Torvalds 已提交
335

L
Len Brown 已提交
336
      end:
337
	return 0;
L
Linus Torvalds 已提交
338 339 340 341 342
}

static int acpi_processor_limit_open_fs(struct inode *inode, struct file *file)
{
	return single_open(file, acpi_processor_limit_seq_show,
L
Len Brown 已提交
343
			   PDE(inode)->data);
L
Linus Torvalds 已提交
344 345
}

346 347 348
static ssize_t acpi_processor_write_limit(struct file * file,
					  const char __user * buffer,
					  size_t count, loff_t * data)
L
Linus Torvalds 已提交
349
{
L
Len Brown 已提交
350
	int result = 0;
351 352
	struct seq_file *m = file->private_data;
	struct acpi_processor *pr = m->private;
L
Len Brown 已提交
353 354 355
	char limit_string[25] = { '\0' };
	int px = 0;
	int tx = 0;
L
Linus Torvalds 已提交
356 357 358


	if (!pr || (count > sizeof(limit_string) - 1)) {
359
		return -EINVAL;
L
Linus Torvalds 已提交
360 361 362
	}

	if (copy_from_user(limit_string, buffer, count)) {
363
		return -EFAULT;
L
Linus Torvalds 已提交
364 365 366 367 368
	}

	limit_string[count] = '\0';

	if (sscanf(limit_string, "%d:%d", &px, &tx) != 2) {
369
		printk(KERN_ERR PREFIX "Invalid data format\n");
370
		return -EINVAL;
L
Linus Torvalds 已提交
371 372 373 374
	}

	if (pr->flags.throttling) {
		if ((tx < 0) || (tx > (pr->throttling.state_count - 1))) {
375
			printk(KERN_ERR PREFIX "Invalid tx\n");
376
			return -EINVAL;
L
Linus Torvalds 已提交
377 378 379 380 381 382
		}
		pr->limit.user.tx = tx;
	}

	result = acpi_processor_apply_limit(pr);

383
	return count;
L
Linus Torvalds 已提交
384 385 386
}

struct file_operations acpi_processor_limit_fops = {
L
Len Brown 已提交
387 388
	.open = acpi_processor_limit_open_fs,
	.read = seq_read,
389
	.write = acpi_processor_write_limit,
L
Len Brown 已提交
390 391
	.llseek = seq_lseek,
	.release = single_release,
L
Linus Torvalds 已提交
392
};