video.c 49.5 KB
Newer Older
L
Linus Torvalds 已提交
1
/*
2
 *  video.c - ACPI Video Driver
L
Linus Torvalds 已提交
3 4 5
 *
 *  Copyright (C) 2004 Luming Yu <luming.yu@intel.com>
 *  Copyright (C) 2004 Bruno Ducrot <ducrot@poupinou.org>
6
 *  Copyright (C) 2006 Thomas Tuttle <linux-kernel@ttuttle.net>
L
Linus Torvalds 已提交
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
 *
 * ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
 *
 *  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/types.h>
#include <linux/list.h>
32
#include <linux/mutex.h>
33
#include <linux/input.h>
34
#include <linux/backlight.h>
35
#include <linux/thermal.h>
36
#include <linux/sort.h>
37 38
#include <linux/pci.h>
#include <linux/pci_ids.h>
39
#include <linux/slab.h>
L
Linus Torvalds 已提交
40
#include <asm/uaccess.h>
41
#include <linux/dmi.h>
L
Linus Torvalds 已提交
42 43
#include <acpi/acpi_bus.h>
#include <acpi/acpi_drivers.h>
44
#include <linux/suspend.h>
45
#include <acpi/video.h>
L
Linus Torvalds 已提交
46

47 48
#include "internal.h"

49 50
#define PREFIX "ACPI: "

L
Linus Torvalds 已提交
51 52 53 54 55 56 57 58
#define ACPI_VIDEO_BUS_NAME		"Video Bus"
#define ACPI_VIDEO_DEVICE_NAME		"Video Device"
#define ACPI_VIDEO_NOTIFY_SWITCH	0x80
#define ACPI_VIDEO_NOTIFY_PROBE		0x81
#define ACPI_VIDEO_NOTIFY_CYCLE		0x82
#define ACPI_VIDEO_NOTIFY_NEXT_OUTPUT	0x83
#define ACPI_VIDEO_NOTIFY_PREV_OUTPUT	0x84

59 60 61 62 63
#define ACPI_VIDEO_NOTIFY_CYCLE_BRIGHTNESS	0x85
#define	ACPI_VIDEO_NOTIFY_INC_BRIGHTNESS	0x86
#define ACPI_VIDEO_NOTIFY_DEC_BRIGHTNESS	0x87
#define ACPI_VIDEO_NOTIFY_ZERO_BRIGHTNESS	0x88
#define ACPI_VIDEO_NOTIFY_DISPLAY_OFF		0x89
L
Linus Torvalds 已提交
64

65
#define MAX_NAME_LEN	20
L
Linus Torvalds 已提交
66 67

#define _COMPONENT		ACPI_VIDEO_COMPONENT
68
ACPI_MODULE_NAME("video");
L
Linus Torvalds 已提交
69

70
MODULE_AUTHOR("Bruno Ducrot");
71
MODULE_DESCRIPTION("ACPI Video Driver");
L
Linus Torvalds 已提交
72 73
MODULE_LICENSE("GPL");

74
static bool brightness_switch_enabled = 1;
75 76
module_param(brightness_switch_enabled, bool, 0644);

77 78 79 80
/*
 * By default, we don't allow duplicate ACPI video bus devices
 * under the same VGA controller
 */
81
static bool allow_duplicates;
82 83
module_param(allow_duplicates, bool, 0644);

84 85 86 87
/*
 * Some BIOSes claim they use minimum backlight at boot,
 * and this may bring dimming screen after boot
 */
88
static bool use_bios_initial_backlight = 1;
89 90
module_param(use_bios_initial_backlight, bool, 0644);

91
static int register_count = 0;
L
Len Brown 已提交
92
static int acpi_video_bus_add(struct acpi_device *device);
93
static int acpi_video_bus_remove(struct acpi_device *device);
94
static void acpi_video_bus_notify(struct acpi_device *device, u32 event);
L
Linus Torvalds 已提交
95

96 97 98 99 100 101
static const struct acpi_device_id video_device_ids[] = {
	{ACPI_VIDEO_HID, 0},
	{"", 0},
};
MODULE_DEVICE_TABLE(acpi, video_device_ids);

L
Linus Torvalds 已提交
102
static struct acpi_driver acpi_video_bus = {
L
Len Brown 已提交
103
	.name = "video",
L
Linus Torvalds 已提交
104
	.class = ACPI_VIDEO_CLASS,
105
	.ids = video_device_ids,
L
Linus Torvalds 已提交
106 107 108
	.ops = {
		.add = acpi_video_bus_add,
		.remove = acpi_video_bus_remove,
109
		.notify = acpi_video_bus_notify,
L
Len Brown 已提交
110
		},
L
Linus Torvalds 已提交
111 112 113
};

struct acpi_video_bus_flags {
L
Len Brown 已提交
114 115 116 117
	u8 multihead:1;		/* can switch video heads */
	u8 rom:1;		/* can retrieve a video rom */
	u8 post:1;		/* can configure the head to */
	u8 reserved:5;
L
Linus Torvalds 已提交
118 119 120
};

struct acpi_video_bus_cap {
121 122 123 124 125 126
	u8 _DOS:1;		/* Enable/Disable output switching */
	u8 _DOD:1;		/* Enumerate all devices attached to display adapter */
	u8 _ROM:1;		/* Get ROM Data */
	u8 _GPD:1;		/* Get POST Device */
	u8 _SPD:1;		/* Set POST Device */
	u8 _VPO:1;		/* Video POST Options */
L
Len Brown 已提交
127
	u8 reserved:2;
L
Linus Torvalds 已提交
128 129
};

L
Len Brown 已提交
130 131
struct acpi_video_device_attrib {
	u32 display_index:4;	/* A zero-based instance of the Display */
132 133 134 135 136
	u32 display_port_attachment:4;	/* This field differentiates the display type */
	u32 display_type:4;	/* Describe the specific type in use */
	u32 vendor_specific:4;	/* Chipset Vendor Specific */
	u32 bios_can_detect:1;	/* BIOS can detect the device */
	u32 depend_on_vga:1;	/* Non-VGA output device whose power is related to
L
Len Brown 已提交
137
				   the VGA device. */
138 139 140
	u32 pipe_id:3;		/* For VGA multiple-head devices. */
	u32 reserved:10;	/* Must be 0 */
	u32 device_id_scheme:1;	/* Device ID Scheme */
L
Linus Torvalds 已提交
141 142 143 144 145
};

struct acpi_video_enumerated_device {
	union {
		u32 int_val;
L
Len Brown 已提交
146
		struct acpi_video_device_attrib attrib;
L
Linus Torvalds 已提交
147 148 149 150 151
	} value;
	struct acpi_video_device *bind_info;
};

struct acpi_video_bus {
152
	struct acpi_device *device;
L
Len Brown 已提交
153
	u8 dos_setting;
L
Linus Torvalds 已提交
154
	struct acpi_video_enumerated_device *attached_array;
L
Len Brown 已提交
155 156
	u8 attached_count;
	struct acpi_video_bus_cap cap;
L
Linus Torvalds 已提交
157
	struct acpi_video_bus_flags flags;
L
Len Brown 已提交
158
	struct list_head video_device_list;
159
	struct mutex device_list_lock;	/* protects video_device_list */
160 161
	struct input_dev *input;
	char phys[32];	/* for input device */
162
	struct notifier_block pm_nb;
L
Linus Torvalds 已提交
163 164 165
};

struct acpi_video_device_flags {
L
Len Brown 已提交
166 167 168
	u8 crt:1;
	u8 lcd:1;
	u8 tvout:1;
169
	u8 dvi:1;
L
Len Brown 已提交
170 171
	u8 bios:1;
	u8 unknown:1;
172 173
	u8 notify:1;
	u8 reserved:1;
L
Linus Torvalds 已提交
174 175 176
};

struct acpi_video_device_cap {
177 178 179
	u8 _ADR:1;		/* Return the unique ID */
	u8 _BCL:1;		/* Query list of brightness control levels supported */
	u8 _BCM:1;		/* Set the brightness level */
180
	u8 _BQC:1;		/* Get current brightness level */
181
	u8 _BCQ:1;		/* Some buggy BIOS uses _BCQ instead of _BQC */
182
	u8 _DDC:1;		/* Return the EDID for this device */
L
Linus Torvalds 已提交
183 184
};

185 186
struct acpi_video_brightness_flags {
	u8 _BCL_no_ac_battery_levels:1;	/* no AC/Battery levels in _BCL */
187
	u8 _BCL_reversed:1;		/* _BCL package is in a reversed order */
188 189 190
	u8 _BCL_use_index:1;		/* levels in _BCL are index values */
	u8 _BCM_use_index:1;		/* input of _BCM is an index value */
	u8 _BQC_use_index:1;		/* _BQC returns an index value */
191 192
};

L
Linus Torvalds 已提交
193
struct acpi_video_device_brightness {
L
Len Brown 已提交
194 195 196
	int curr;
	int count;
	int *levels;
197
	struct acpi_video_brightness_flags flags;
L
Linus Torvalds 已提交
198 199 200
};

struct acpi_video_device {
L
Len Brown 已提交
201 202 203 204 205 206
	unsigned long device_id;
	struct acpi_video_device_flags flags;
	struct acpi_video_device_cap cap;
	struct list_head entry;
	struct acpi_video_bus *video;
	struct acpi_device *dev;
L
Linus Torvalds 已提交
207
	struct acpi_video_device_brightness *brightness;
208
	struct backlight_device *backlight;
209
	struct thermal_cooling_device *cooling_dev;
L
Linus Torvalds 已提交
210 211
};

J
Jan Engelhardt 已提交
212
static const char device_decode[][30] = {
L
Linus Torvalds 已提交
213 214 215 216 217 218
	"motherboard VGA device",
	"PCI VGA device",
	"AGP VGA device",
	"UNKNOWN",
};

L
Len Brown 已提交
219 220 221 222
static void acpi_video_device_notify(acpi_handle handle, u32 event, void *data);
static void acpi_video_device_rebind(struct acpi_video_bus *video);
static void acpi_video_device_bind(struct acpi_video_bus *video,
				   struct acpi_video_device *device);
L
Linus Torvalds 已提交
223
static int acpi_video_device_enumerate(struct acpi_video_bus *video);
224 225 226 227
static int acpi_video_device_lcd_set_level(struct acpi_video_device *device,
			int level);
static int acpi_video_device_lcd_get_level_current(
			struct acpi_video_device *device,
228
			unsigned long long *level, bool raw);
L
Len Brown 已提交
229 230
static int acpi_video_get_next_level(struct acpi_video_device *device,
				     u32 level_current, u32 event);
231
static int acpi_video_switch_brightness(struct acpi_video_device *device,
L
Len Brown 已提交
232
					 int event);
L
Linus Torvalds 已提交
233

234
/* backlight device sysfs support */
235 236
static int acpi_video_get_brightness(struct backlight_device *bd)
{
237
	unsigned long long cur_level;
238
	int i;
239
	struct acpi_video_device *vd =
240
		(struct acpi_video_device *)bl_get_data(bd);
241

242
	if (acpi_video_device_lcd_get_level_current(vd, &cur_level, false))
243
		return -EINVAL;
244 245
	for (i = 2; i < vd->brightness->count; i++) {
		if (vd->brightness->levels[i] == cur_level)
246 247 248 249
			/*
			 * The first two entries are special - see page 575
			 * of the ACPI spec 3.0
			 */
250 251 252
			return i-2;
	}
	return 0;
253 254 255 256
}

static int acpi_video_set_brightness(struct backlight_device *bd)
{
257
	int request_level = bd->props.brightness + 2;
258
	struct acpi_video_device *vd =
259
		(struct acpi_video_device *)bl_get_data(bd);
260 261 262

	return acpi_video_device_lcd_set_level(vd,
				vd->brightness->levels[request_level]);
263 264
}

L
Lionel Debroux 已提交
265
static const struct backlight_ops acpi_backlight_ops = {
266 267 268 269
	.get_brightness = acpi_video_get_brightness,
	.update_status  = acpi_video_set_brightness,
};

270
/* thermal cooling device callbacks */
271
static int video_get_max_state(struct thermal_cooling_device *cooling_dev, unsigned
272
			       long *state)
273
{
274
	struct acpi_device *device = cooling_dev->devdata;
275 276
	struct acpi_video_device *video = acpi_driver_data(device);

277 278
	*state = video->brightness->count - 3;
	return 0;
279 280
}

281
static int video_get_cur_state(struct thermal_cooling_device *cooling_dev, unsigned
282
			       long *state)
283
{
284
	struct acpi_device *device = cooling_dev->devdata;
285
	struct acpi_video_device *video = acpi_driver_data(device);
286
	unsigned long long level;
287
	int offset;
288

289
	if (acpi_video_device_lcd_get_level_current(video, &level, false))
290
		return -EINVAL;
291 292 293 294 295
	for (offset = 2; offset < video->brightness->count; offset++)
		if (level == video->brightness->levels[offset]) {
			*state = video->brightness->count - offset - 1;
			return 0;
		}
296 297 298 299 300

	return -EINVAL;
}

static int
301
video_set_cur_state(struct thermal_cooling_device *cooling_dev, unsigned long state)
302
{
303
	struct acpi_device *device = cooling_dev->devdata;
304 305 306 307 308 309 310 311 312 313 314
	struct acpi_video_device *video = acpi_driver_data(device);
	int level;

	if ( state >= video->brightness->count - 2)
		return -EINVAL;

	state = video->brightness->count - state;
	level = video->brightness->levels[state -1];
	return acpi_video_device_lcd_set_level(video, level);
}

V
Vasiliy Kulikov 已提交
315
static const struct thermal_cooling_device_ops video_cooling_ops = {
316 317 318 319 320
	.get_max_state = video_get_max_state,
	.get_cur_state = video_get_cur_state,
	.set_cur_state = video_set_cur_state,
};

321 322 323 324 325
/*
 * --------------------------------------------------------------------------
 *                             Video Management
 * --------------------------------------------------------------------------
 */
L
Linus Torvalds 已提交
326 327

static int
L
Len Brown 已提交
328 329
acpi_video_device_lcd_query_levels(struct acpi_video_device *device,
				   union acpi_object **levels)
L
Linus Torvalds 已提交
330
{
L
Len Brown 已提交
331 332 333
	int status;
	struct acpi_buffer buffer = { ACPI_ALLOCATE_BUFFER, NULL };
	union acpi_object *obj;
L
Linus Torvalds 已提交
334 335 336 337


	*levels = NULL;

338
	status = acpi_evaluate_object(device->dev->handle, "_BCL", NULL, &buffer);
L
Linus Torvalds 已提交
339
	if (!ACPI_SUCCESS(status))
340
		return status;
L
Len Brown 已提交
341
	obj = (union acpi_object *)buffer.pointer;
342
	if (!obj || (obj->type != ACPI_TYPE_PACKAGE)) {
343
		printk(KERN_ERR PREFIX "Invalid _BCL data\n");
L
Linus Torvalds 已提交
344 345 346 347 348 349
		status = -EFAULT;
		goto err;
	}

	*levels = obj;

350
	return 0;
L
Linus Torvalds 已提交
351

L
Len Brown 已提交
352
      err:
353
	kfree(buffer.pointer);
L
Linus Torvalds 已提交
354

355
	return status;
L
Linus Torvalds 已提交
356 357 358
}

static int
L
Len Brown 已提交
359
acpi_video_device_lcd_set_level(struct acpi_video_device *device, int level)
L
Linus Torvalds 已提交
360
{
361
	int status;
L
Len Brown 已提交
362 363
	union acpi_object arg0 = { ACPI_TYPE_INTEGER };
	struct acpi_object_list args = { 1, &arg0 };
364
	int state;
L
Linus Torvalds 已提交
365 366 367

	arg0.integer.value = level;

368 369 370 371 372 373 374
	status = acpi_evaluate_object(device->dev->handle, "_BCM",
				      &args, NULL);
	if (ACPI_FAILURE(status)) {
		ACPI_ERROR((AE_INFO, "Evaluating _BCM failed"));
		return -EIO;
	}

375
	device->brightness->curr = level;
376
	for (state = 2; state < device->brightness->count; state++)
377
		if (level == device->brightness->levels[state]) {
378 379
			if (device->backlight)
				device->backlight->props.brightness = state - 2;
380 381
			return 0;
		}
382

383 384
	ACPI_ERROR((AE_INFO, "Current brightness invalid"));
	return -EINVAL;
L
Linus Torvalds 已提交
385 386
}

387 388 389 390 391 392 393 394 395 396 397 398
/*
 * For some buggy _BQC methods, we need to add a constant value to
 * the _BQC return value to get the actual current brightness level
 */

static int bqc_offset_aml_bug_workaround;
static int __init video_set_bqc_offset(const struct dmi_system_id *d)
{
	bqc_offset_aml_bug_workaround = 9;
	return 0;
}

399 400 401 402 403 404
static int video_ignore_initial_backlight(const struct dmi_system_id *d)
{
	use_bios_initial_backlight = 0;
	return 0;
}

405 406 407 408 409 410 411 412 413 414 415 416
static struct dmi_system_id video_dmi_table[] __initdata = {
	/*
	 * Broken _BQC workaround http://bugzilla.kernel.org/show_bug.cgi?id=13121
	 */
	{
	 .callback = video_set_bqc_offset,
	 .ident = "Acer Aspire 5720",
	 .matches = {
		DMI_MATCH(DMI_BOARD_VENDOR, "Acer"),
		DMI_MATCH(DMI_PRODUCT_NAME, "Aspire 5720"),
		},
	},
417 418 419 420 421 422 423 424
	{
	 .callback = video_set_bqc_offset,
	 .ident = "Acer Aspire 5710Z",
	 .matches = {
		DMI_MATCH(DMI_BOARD_VENDOR, "Acer"),
		DMI_MATCH(DMI_PRODUCT_NAME, "Aspire 5710Z"),
		},
	},
425 426 427 428 429 430 431 432
	{
	 .callback = video_set_bqc_offset,
	 .ident = "eMachines E510",
	 .matches = {
		DMI_MATCH(DMI_BOARD_VENDOR, "EMACHINES"),
		DMI_MATCH(DMI_PRODUCT_NAME, "eMachines E510"),
		},
	},
433 434 435 436 437 438 439 440
	{
	 .callback = video_set_bqc_offset,
	 .ident = "Acer Aspire 5315",
	 .matches = {
		DMI_MATCH(DMI_BOARD_VENDOR, "Acer"),
		DMI_MATCH(DMI_PRODUCT_NAME, "Aspire 5315"),
		},
	},
441 442 443 444 445 446 447 448
	{
	 .callback = video_set_bqc_offset,
	 .ident = "Acer Aspire 7720",
	 .matches = {
		DMI_MATCH(DMI_BOARD_VENDOR, "Acer"),
		DMI_MATCH(DMI_PRODUCT_NAME, "Aspire 7720"),
		},
	},
449 450 451 452 453 454 455 456
	{
	 .callback = video_ignore_initial_backlight,
	 .ident = "HP Folio 13-2000",
	 .matches = {
		DMI_MATCH(DMI_BOARD_VENDOR, "Hewlett-Packard"),
		DMI_MATCH(DMI_PRODUCT_NAME, "HP Folio 13 - 2000 Notebook PC"),
		},
	},
457 458 459 460 461 462 463 464
	{
	 .callback = video_ignore_initial_backlight,
	 .ident = "Fujitsu E753",
	 .matches = {
		DMI_MATCH(DMI_BOARD_VENDOR, "FUJITSU"),
		DMI_MATCH(DMI_PRODUCT_NAME, "LIFEBOOK E753"),
		},
	},
465 466 467 468 469 470 471 472
	{
	 .callback = video_ignore_initial_backlight,
	 .ident = "HP Pavilion dm4",
	 .matches = {
		DMI_MATCH(DMI_BOARD_VENDOR, "Hewlett-Packard"),
		DMI_MATCH(DMI_PRODUCT_NAME, "HP Pavilion dm4 Notebook PC"),
		},
	},
473 474 475 476 477 478 479 480
	{
	 .callback = video_ignore_initial_backlight,
	 .ident = "HP Pavilion g6 Notebook PC",
	 .matches = {
		 DMI_MATCH(DMI_BOARD_VENDOR, "Hewlett-Packard"),
		 DMI_MATCH(DMI_PRODUCT_NAME, "HP Pavilion g6 Notebook PC"),
		},
	},
481 482 483 484 485 486 487 488
	{
	 .callback = video_ignore_initial_backlight,
	 .ident = "HP 1000 Notebook PC",
	 .matches = {
		DMI_MATCH(DMI_BOARD_VENDOR, "Hewlett-Packard"),
		DMI_MATCH(DMI_PRODUCT_NAME, "HP 1000 Notebook PC"),
		},
	},
489 490 491 492 493 494 495 496
	{
	 .callback = video_ignore_initial_backlight,
	 .ident = "HP Pavilion m4",
	 .matches = {
		DMI_MATCH(DMI_BOARD_VENDOR, "Hewlett-Packard"),
		DMI_MATCH(DMI_PRODUCT_NAME, "HP Pavilion m4 Notebook PC"),
		},
	},
497 498 499
	{}
};

500 501 502 503 504 505 506 507 508 509 510 511 512 513 514 515 516 517 518 519 520 521 522 523 524
static unsigned long long
acpi_video_bqc_value_to_level(struct acpi_video_device *device,
			      unsigned long long bqc_value)
{
	unsigned long long level;

	if (device->brightness->flags._BQC_use_index) {
		/*
		 * _BQC returns an index that doesn't account for
		 * the first 2 items with special meaning, so we need
		 * to compensate for that by offsetting ourselves
		 */
		if (device->brightness->flags._BCL_reversed)
			bqc_value = device->brightness->count - 3 - bqc_value;

		level = device->brightness->levels[bqc_value + 2];
	} else {
		level = bqc_value;
	}

	level += bqc_offset_aml_bug_workaround;

	return level;
}

L
Linus Torvalds 已提交
525
static int
L
Len Brown 已提交
526
acpi_video_device_lcd_get_level_current(struct acpi_video_device *device,
527
					unsigned long long *level, bool raw)
L
Linus Torvalds 已提交
528
{
529
	acpi_status status = AE_OK;
530
	int i;
531

532 533 534 535
	if (device->cap._BQC || device->cap._BCQ) {
		char *buf = device->cap._BQC ? "_BQC" : "_BCQ";

		status = acpi_evaluate_integer(device->dev->handle, buf,
536 537
						NULL, level);
		if (ACPI_SUCCESS(status)) {
538 539 540 541 542 543 544 545 546
			if (raw) {
				/*
				 * Caller has indicated he wants the raw
				 * value returned by _BQC, so don't furtherly
				 * mess with the value.
				 */
				return 0;
			}

547
			*level = acpi_video_bqc_value_to_level(device, *level);
548

549 550 551 552 553
			for (i = 2; i < device->brightness->count; i++)
				if (device->brightness->levels[i] == *level) {
					device->brightness->curr = *level;
					return 0;
			}
554 555 556 557 558 559 560 561
			/*
			 * BQC returned an invalid level.
			 * Stop using it.
			 */
			ACPI_WARNING((AE_INFO,
				      "%s returned an invalid level",
				      buf));
			device->cap._BQC = device->cap._BCQ = 0;
562
		} else {
563 564
			/*
			 * Fixme:
565 566 567 568 569 570
			 * should we return an error or ignore this failure?
			 * dev->brightness->curr is a cached value which stores
			 * the correct current backlight level in most cases.
			 * ACPI video backlight still works w/ buggy _BQC.
			 * http://bugzilla.kernel.org/show_bug.cgi?id=12233
			 */
571 572
			ACPI_WARNING((AE_INFO, "Evaluating %s failed", buf));
			device->cap._BQC = device->cap._BCQ = 0;
573 574 575
		}
	}

576
	*level = device->brightness->curr;
577
	return 0;
L
Linus Torvalds 已提交
578 579 580
}

static int
L
Len Brown 已提交
581 582
acpi_video_device_EDID(struct acpi_video_device *device,
		       union acpi_object **edid, ssize_t length)
L
Linus Torvalds 已提交
583
{
L
Len Brown 已提交
584 585 586 587 588
	int status;
	struct acpi_buffer buffer = { ACPI_ALLOCATE_BUFFER, NULL };
	union acpi_object *obj;
	union acpi_object arg0 = { ACPI_TYPE_INTEGER };
	struct acpi_object_list args = { 1, &arg0 };
L
Linus Torvalds 已提交
589 590 591 592 593


	*edid = NULL;

	if (!device)
594
		return -ENODEV;
L
Linus Torvalds 已提交
595 596 597 598 599
	if (length == 128)
		arg0.integer.value = 1;
	else if (length == 256)
		arg0.integer.value = 2;
	else
600
		return -EINVAL;
L
Linus Torvalds 已提交
601

602
	status = acpi_evaluate_object(device->dev->handle, "_DDC", &args, &buffer);
L
Linus Torvalds 已提交
603
	if (ACPI_FAILURE(status))
604
		return -ENODEV;
L
Linus Torvalds 已提交
605

606
	obj = buffer.pointer;
L
Linus Torvalds 已提交
607 608 609 610

	if (obj && obj->type == ACPI_TYPE_BUFFER)
		*edid = obj;
	else {
611
		printk(KERN_ERR PREFIX "Invalid _DDC data\n");
L
Linus Torvalds 已提交
612 613 614 615
		status = -EFAULT;
		kfree(obj);
	}

616
	return status;
L
Linus Torvalds 已提交
617 618 619 620 621 622
}

/* bus */

/*
 *  Arg:
623 624
 *	video		: video bus device pointer
 *	bios_flag	:
L
Linus Torvalds 已提交
625 626 627
 *		0.	The system BIOS should NOT automatically switch(toggle)
 *			the active display output.
 *		1.	The system BIOS should automatically switch (toggle) the
628
 *			active display output. No switch event.
L
Linus Torvalds 已提交
629 630 631 632 633 634
 *		2.	The _DGS value should be locked.
 *		3.	The system BIOS should not automatically switch (toggle) the
 *			active display output, but instead generate the display switch
 *			event notify code.
 *	lcd_flag	:
 *		0.	The system BIOS should automatically control the brightness level
635
 *			of the LCD when the power changes from AC to DC
636
 *		1.	The system BIOS should NOT automatically control the brightness
637
 *			level of the LCD when the power changes from AC to DC.
638
 *  Return Value:
639
 *		-EINVAL	wrong arg.
L
Linus Torvalds 已提交
640 641 642
 */

static int
L
Len Brown 已提交
643
acpi_video_bus_DOS(struct acpi_video_bus *video, int bios_flag, int lcd_flag)
L
Linus Torvalds 已提交
644
{
645
	acpi_status status;
L
Len Brown 已提交
646 647
	union acpi_object arg0 = { ACPI_TYPE_INTEGER };
	struct acpi_object_list args = { 1, &arg0 };
L
Linus Torvalds 已提交
648

649 650
	if (!video->cap._DOS)
		return 0;
L
Linus Torvalds 已提交
651

652 653
	if (bios_flag < 0 || bios_flag > 3 || lcd_flag < 0 || lcd_flag > 1)
		return -EINVAL;
L
Linus Torvalds 已提交
654 655
	arg0.integer.value = (lcd_flag << 2) | bios_flag;
	video->dos_setting = arg0.integer.value;
656 657 658 659
	status = acpi_evaluate_object(video->device->handle, "_DOS",
		&args, NULL);
	if (ACPI_FAILURE(status))
		return -EIO;
L
Linus Torvalds 已提交
660

661
	return 0;
L
Linus Torvalds 已提交
662 663
}

664 665 666 667 668 669 670 671 672 673
/*
 * Simple comparison function used to sort backlight levels.
 */

static int
acpi_video_cmp_level(const void *a, const void *b)
{
	return *(int *)a - *(int *)b;
}

674 675 676 677 678 679 680 681 682 683 684 685 686 687 688 689 690 691 692 693 694 695 696 697 698 699 700 701 702 703 704 705 706 707 708 709 710 711 712 713 714 715 716 717 718 719 720 721 722 723
/*
 * Decides if _BQC/_BCQ for this system is usable
 *
 * We do this by changing the level first and then read out the current
 * brightness level, if the value does not match, find out if it is using
 * index. If not, clear the _BQC/_BCQ capability.
 */
static int acpi_video_bqc_quirk(struct acpi_video_device *device,
				int max_level, int current_level)
{
	struct acpi_video_device_brightness *br = device->brightness;
	int result;
	unsigned long long level;
	int test_level;

	/* don't mess with existing known broken systems */
	if (bqc_offset_aml_bug_workaround)
		return 0;

	/*
	 * Some systems always report current brightness level as maximum
	 * through _BQC, we need to test another value for them.
	 */
	test_level = current_level == max_level ? br->levels[2] : max_level;

	result = acpi_video_device_lcd_set_level(device, test_level);
	if (result)
		return result;

	result = acpi_video_device_lcd_get_level_current(device, &level, true);
	if (result)
		return result;

	if (level != test_level) {
		/* buggy _BQC found, need to find out if it uses index */
		if (level < br->count) {
			if (br->flags._BCL_reversed)
				level = br->count - 3 - level;
			if (br->levels[level + 2] == test_level)
				br->flags._BQC_use_index = 1;
		}

		if (!br->flags._BQC_use_index)
			device->cap._BQC = device->cap._BCQ = 0;
	}

	return 0;
}


L
Linus Torvalds 已提交
724
/*
725 726
 *  Arg:
 *	device	: video output device (LCD, CRT, ..)
L
Linus Torvalds 已提交
727 728
 *
 *  Return Value:
729 730 731 732 733 734 735 736 737
 *	Maximum brightness level
 *
 *  Allocate and initialize device->brightness.
 */

static int
acpi_video_init_brightness(struct acpi_video_device *device)
{
	union acpi_object *obj = NULL;
738
	int i, max_level = 0, count = 0, level_ac_battery = 0;
739
	unsigned long long level, level_old;
740 741
	union acpi_object *o;
	struct acpi_video_device_brightness *br = NULL;
742
	int result = -EINVAL;
743 744 745 746 747 748 749 750 751 752 753 754 755

	if (!ACPI_SUCCESS(acpi_video_device_lcd_query_levels(device, &obj))) {
		ACPI_DEBUG_PRINT((ACPI_DB_INFO, "Could not query available "
						"LCD brightness level\n"));
		goto out;
	}

	if (obj->package.count < 2)
		goto out;

	br = kzalloc(sizeof(*br), GFP_KERNEL);
	if (!br) {
		printk(KERN_ERR "can't allocate memory\n");
756
		result = -ENOMEM;
757 758 759
		goto out;
	}

760
	br->levels = kmalloc((obj->package.count + 2) * sizeof *(br->levels),
761
				GFP_KERNEL);
762 763
	if (!br->levels) {
		result = -ENOMEM;
764
		goto out_free;
765
	}
766 767 768 769 770 771 772 773 774 775 776 777 778 779

	for (i = 0; i < obj->package.count; i++) {
		o = (union acpi_object *)&obj->package.elements[i];
		if (o->type != ACPI_TYPE_INTEGER) {
			printk(KERN_ERR PREFIX "Invalid data\n");
			continue;
		}
		br->levels[count] = (u32) o->integer.value;

		if (br->levels[count] > max_level)
			max_level = br->levels[count];
		count++;
	}

780 781 782 783 784 785
	/*
	 * some buggy BIOS don't export the levels
	 * when machine is on AC/Battery in _BCL package.
	 * In this case, the first two elements in _BCL packages
	 * are also supported brightness levels that OS should take care of.
	 */
786 787
	for (i = 2; i < count; i++) {
		if (br->levels[i] == br->levels[0])
788
			level_ac_battery++;
789 790 791
		if (br->levels[i] == br->levels[1])
			level_ac_battery++;
	}
792 793 794 795 796 797 798 799

	if (level_ac_battery < 2) {
		level_ac_battery = 2 - level_ac_battery;
		br->flags._BCL_no_ac_battery_levels = 1;
		for (i = (count - 1 + level_ac_battery); i >= 2; i--)
			br->levels[i] = br->levels[i - level_ac_battery];
		count += level_ac_battery;
	} else if (level_ac_battery > 2)
800
		ACPI_ERROR((AE_INFO, "Too many duplicates in _BCL package"));
801

802 803 804 805 806 807 808
	/* Check if the _BCL package is in a reversed order */
	if (max_level == br->levels[2]) {
		br->flags._BCL_reversed = 1;
		sort(&br->levels[2], count - 2, sizeof(br->levels[2]),
			acpi_video_cmp_level, NULL);
	} else if (max_level != br->levels[count - 1])
		ACPI_ERROR((AE_INFO,
809
			    "Found unordered _BCL package"));
810 811 812

	br->count = count;
	device->brightness = br;
813 814 815 816 817 818 819 820 821 822 823 824

	/* Check the input/output of _BQC/_BCL/_BCM */
	if ((max_level < 100) && (max_level <= (count - 2)))
		br->flags._BCL_use_index = 1;

	/*
	 * _BCM is always consistent with _BCL,
	 * at least for all the laptops we have ever seen.
	 */
	br->flags._BCM_use_index = br->flags._BCL_use_index;

	/* _BQC uses INDEX while _BCL uses VALUE in some laptops */
825
	br->curr = level = max_level;
826 827 828 829

	if (!device->cap._BQC)
		goto set_level;

830 831
	result = acpi_video_device_lcd_get_level_current(device,
							 &level_old, true);
832 833 834
	if (result)
		goto out_free_levels;

835
	result = acpi_video_bqc_quirk(device, max_level, level_old);
836 837
	if (result)
		goto out_free_levels;
838 839 840 841 842 843
	/*
	 * cap._BQC may get cleared due to _BQC is found to be broken
	 * in acpi_video_bqc_quirk, so check again here.
	 */
	if (!device->cap._BQC)
		goto set_level;
844

845 846
	if (use_bios_initial_backlight) {
		level = acpi_video_bqc_value_to_level(device, level_old);
847
		/*
848 849 850 851
		 * On some buggy laptops, _BQC returns an uninitialized
		 * value when invoked for the first time, i.e.
		 * level_old is invalid (no matter whether it's a level
		 * or an index). Set the backlight to max_level in this case.
852
		 */
853 854 855 856 857
		for (i = 2; i < br->count; i++)
			if (level_old == br->levels[i])
				break;
		if (i == br->count)
			level = max_level;
858
	}
859 860

set_level:
861
	result = acpi_video_device_lcd_set_level(device, level);
862 863
	if (result)
		goto out_free_levels;
864

865 866
	ACPI_DEBUG_PRINT((ACPI_DB_INFO,
			  "found %d brightness levels\n", count - 2));
867
	kfree(obj);
868
	return result;
869 870 871 872 873 874 875 876

out_free_levels:
	kfree(br->levels);
out_free:
	kfree(br);
out:
	device->brightness = NULL;
	kfree(obj);
877
	return result;
878 879 880 881 882 883 884
}

/*
 *  Arg:
 *	device	: video output device (LCD, CRT, ..)
 *
 *  Return Value:
885
 *	None
L
Linus Torvalds 已提交
886
 *
887
 *  Find out all required AML methods defined under the output
L
Linus Torvalds 已提交
888 889 890
 *  device.
 */

L
Len Brown 已提交
891
static void acpi_video_device_find_cap(struct acpi_video_device *device)
L
Linus Torvalds 已提交
892 893 894
{
	acpi_handle h_dummy1;

895
	if (ACPI_SUCCESS(acpi_get_handle(device->dev->handle, "_ADR", &h_dummy1))) {
L
Linus Torvalds 已提交
896 897
		device->cap._ADR = 1;
	}
898
	if (ACPI_SUCCESS(acpi_get_handle(device->dev->handle, "_BCL", &h_dummy1))) {
L
Len Brown 已提交
899
		device->cap._BCL = 1;
L
Linus Torvalds 已提交
900
	}
901
	if (ACPI_SUCCESS(acpi_get_handle(device->dev->handle, "_BCM", &h_dummy1))) {
L
Len Brown 已提交
902
		device->cap._BCM = 1;
L
Linus Torvalds 已提交
903
	}
904 905
	if (ACPI_SUCCESS(acpi_get_handle(device->dev->handle,"_BQC",&h_dummy1)))
		device->cap._BQC = 1;
906 907 908 909 910 911
	else if (ACPI_SUCCESS(acpi_get_handle(device->dev->handle, "_BCQ",
				&h_dummy1))) {
		printk(KERN_WARNING FW_BUG "_BCQ is used instead of _BQC\n");
		device->cap._BCQ = 1;
	}

912
	if (ACPI_SUCCESS(acpi_get_handle(device->dev->handle, "_DDC", &h_dummy1))) {
L
Len Brown 已提交
913
		device->cap._DDC = 1;
L
Linus Torvalds 已提交
914 915
	}

916 917 918
	if (acpi_video_init_brightness(device))
		return;

919
	if (acpi_video_backlight_support()) {
920
		struct backlight_properties props;
921 922 923
		struct pci_dev *pdev;
		acpi_handle acpi_parent;
		struct device *parent = NULL;
924
		int result;
925 926
		static int count = 0;
		char *name;
927

J
Julia Lawall 已提交
928
		name = kasprintf(GFP_KERNEL, "acpi_video%d", count);
929 930
		if (!name)
			return;
J
Julia Lawall 已提交
931
		count++;
932

933 934 935 936 937 938 939 940
		acpi_get_parent(device->dev->handle, &acpi_parent);

		pdev = acpi_get_pci_dev(acpi_parent);
		if (pdev) {
			parent = &pdev->dev;
			pci_dev_put(pdev);
		}

941
		memset(&props, 0, sizeof(struct backlight_properties));
M
Matthew Garrett 已提交
942
		props.type = BACKLIGHT_FIRMWARE;
943
		props.max_brightness = device->brightness->count - 3;
944 945 946
		device->backlight = backlight_device_register(name,
							      parent,
							      device,
947 948
							      &acpi_backlight_ops,
							      &props);
949
		kfree(name);
950 951
		if (IS_ERR(device->backlight))
			return;
952

953 954 955 956 957 958
		/*
		 * Save current brightness level in case we have to restore it
		 * before acpi_video_device_lcd_set_level() is called next time.
		 */
		device->backlight->props.brightness =
				acpi_video_get_brightness(device->backlight);
959

960
		device->cooling_dev = thermal_cooling_device_register("LCD",
961
					device->dev, &video_cooling_ops);
962
		if (IS_ERR(device->cooling_dev)) {
963
			/*
964
			 * Set cooling_dev to NULL so we don't crash trying to
965 966 967 968 969 970
			 * free it.
			 * Also, why the hell we are returning early and
			 * not attempt to register video output if cooling
			 * device registration failed?
			 * -- dtor
			 */
971
			device->cooling_dev = NULL;
972
			return;
973
		}
974

975
		dev_info(&device->dev->dev, "registered as cooling_device%d\n",
976
			 device->cooling_dev->id);
977
		result = sysfs_create_link(&device->dev->dev.kobj,
978
				&device->cooling_dev->device.kobj,
979 980 981
				"thermal_cooling");
		if (result)
			printk(KERN_ERR PREFIX "Create sysfs link\n");
982
		result = sysfs_create_link(&device->cooling_dev->device.kobj,
983 984 985 986
				&device->dev->dev.kobj, "device");
		if (result)
			printk(KERN_ERR PREFIX "Create sysfs link\n");

987 988 989 990 991
	} else {
		/* Remove the brightness object. */
		kfree(device->brightness->levels);
		kfree(device->brightness);
		device->brightness = NULL;
992
	}
L
Linus Torvalds 已提交
993 994 995
}

/*
996 997
 *  Arg:
 *	device	: video output device (VGA)
L
Linus Torvalds 已提交
998 999
 *
 *  Return Value:
1000
 *	None
L
Linus Torvalds 已提交
1001
 *
1002
 *  Find out all required AML methods defined under the video bus device.
L
Linus Torvalds 已提交
1003 1004
 */

L
Len Brown 已提交
1005
static void acpi_video_bus_find_cap(struct acpi_video_bus *video)
L
Linus Torvalds 已提交
1006
{
L
Len Brown 已提交
1007
	acpi_handle h_dummy1;
L
Linus Torvalds 已提交
1008

1009
	if (ACPI_SUCCESS(acpi_get_handle(video->device->handle, "_DOS", &h_dummy1))) {
L
Linus Torvalds 已提交
1010 1011
		video->cap._DOS = 1;
	}
1012
	if (ACPI_SUCCESS(acpi_get_handle(video->device->handle, "_DOD", &h_dummy1))) {
L
Linus Torvalds 已提交
1013 1014
		video->cap._DOD = 1;
	}
1015
	if (ACPI_SUCCESS(acpi_get_handle(video->device->handle, "_ROM", &h_dummy1))) {
L
Linus Torvalds 已提交
1016 1017
		video->cap._ROM = 1;
	}
1018
	if (ACPI_SUCCESS(acpi_get_handle(video->device->handle, "_GPD", &h_dummy1))) {
L
Linus Torvalds 已提交
1019 1020
		video->cap._GPD = 1;
	}
1021
	if (ACPI_SUCCESS(acpi_get_handle(video->device->handle, "_SPD", &h_dummy1))) {
L
Linus Torvalds 已提交
1022 1023
		video->cap._SPD = 1;
	}
1024
	if (ACPI_SUCCESS(acpi_get_handle(video->device->handle, "_VPO", &h_dummy1))) {
L
Linus Torvalds 已提交
1025 1026 1027 1028 1029 1030 1031 1032 1033
		video->cap._VPO = 1;
	}
}

/*
 * Check whether the video bus device has required AML method to
 * support the desired features
 */

L
Len Brown 已提交
1034
static int acpi_video_bus_check(struct acpi_video_bus *video)
L
Linus Torvalds 已提交
1035
{
L
Len Brown 已提交
1036
	acpi_status status = -ENOENT;
1037
	struct pci_dev *dev;
L
Linus Torvalds 已提交
1038 1039

	if (!video)
1040
		return -EINVAL;
L
Linus Torvalds 已提交
1041

1042
	dev = acpi_get_pci_dev(video->device->handle);
1043 1044
	if (!dev)
		return -ENODEV;
1045
	pci_dev_put(dev);
1046

1047 1048
	/*
	 * Since there is no HID, CID and so on for VGA driver, we have
L
Linus Torvalds 已提交
1049 1050 1051
	 * to check well known required nodes.
	 */

1052
	/* Does this device support video switching? */
1053 1054 1055 1056 1057 1058
	if (video->cap._DOS || video->cap._DOD) {
		if (!video->cap._DOS) {
			printk(KERN_WARNING FW_BUG
				"ACPI(%s) defines _DOD but not _DOS\n",
				acpi_device_bid(video->device));
		}
L
Linus Torvalds 已提交
1059 1060 1061 1062
		video->flags.multihead = 1;
		status = 0;
	}

1063
	/* Does this device support retrieving a video ROM? */
L
Len Brown 已提交
1064
	if (video->cap._ROM) {
L
Linus Torvalds 已提交
1065 1066 1067 1068
		video->flags.rom = 1;
		status = 0;
	}

1069
	/* Does this device support configuring which video device to POST? */
L
Len Brown 已提交
1070
	if (video->cap._GPD && video->cap._SPD && video->cap._VPO) {
L
Linus Torvalds 已提交
1071 1072 1073 1074
		video->flags.post = 1;
		status = 0;
	}

1075
	return status;
L
Linus Torvalds 已提交
1076 1077
}

1078 1079 1080 1081 1082
/*
 * --------------------------------------------------------------------------
 *                               Driver Interface
 * --------------------------------------------------------------------------
 */
L
Linus Torvalds 已提交
1083 1084

/* device interface */
1085 1086 1087
static struct acpi_video_device_attrib*
acpi_video_get_device_attr(struct acpi_video_bus *video, unsigned long device_id)
{
1088 1089 1090 1091 1092 1093 1094 1095
	struct acpi_video_enumerated_device *ids;
	int i;

	for (i = 0; i < video->attached_count; i++) {
		ids = &video->attached_array[i];
		if ((ids->value.int_val & 0xffff) == device_id)
			return &ids->value.attrib;
	}
1096 1097 1098

	return NULL;
}
L
Linus Torvalds 已提交
1099

1100 1101 1102 1103 1104 1105 1106 1107 1108 1109 1110 1111 1112 1113 1114 1115
static int
acpi_video_get_device_type(struct acpi_video_bus *video,
			   unsigned long device_id)
{
	struct acpi_video_enumerated_device *ids;
	int i;

	for (i = 0; i < video->attached_count; i++) {
		ids = &video->attached_array[i];
		if ((ids->value.int_val & 0xffff) == device_id)
			return ids->value.int_val;
	}

	return 0;
}

L
Linus Torvalds 已提交
1116
static int
L
Len Brown 已提交
1117 1118
acpi_video_bus_get_one_device(struct acpi_device *device,
			      struct acpi_video_bus *video)
L
Linus Torvalds 已提交
1119
{
1120
	unsigned long long device_id;
1121
	int status, device_type;
L
Len Brown 已提交
1122
	struct acpi_video_device *data;
1123
	struct acpi_video_device_attrib* attribute;
L
Linus Torvalds 已提交
1124

L
Len Brown 已提交
1125 1126
	status =
	    acpi_evaluate_integer(device->handle, "_ADR", NULL, &device_id);
1127 1128 1129
	/* Some device omits _ADR, we skip them instead of fail */
	if (ACPI_FAILURE(status))
		return 0;
L
Linus Torvalds 已提交
1130

1131 1132 1133
	data = kzalloc(sizeof(struct acpi_video_device), GFP_KERNEL);
	if (!data)
		return -ENOMEM;
1134

1135 1136 1137 1138 1139 1140 1141 1142 1143 1144 1145 1146 1147 1148 1149 1150 1151 1152 1153 1154 1155 1156 1157 1158 1159 1160 1161 1162 1163 1164 1165 1166 1167 1168 1169
	strcpy(acpi_device_name(device), ACPI_VIDEO_DEVICE_NAME);
	strcpy(acpi_device_class(device), ACPI_VIDEO_CLASS);
	device->driver_data = data;

	data->device_id = device_id;
	data->video = video;
	data->dev = device;

	attribute = acpi_video_get_device_attr(video, device_id);

	if((attribute != NULL) && attribute->device_id_scheme) {
		switch (attribute->display_type) {
		case ACPI_VIDEO_DISPLAY_CRT:
			data->flags.crt = 1;
			break;
		case ACPI_VIDEO_DISPLAY_TV:
			data->flags.tvout = 1;
			break;
		case ACPI_VIDEO_DISPLAY_DVI:
			data->flags.dvi = 1;
			break;
		case ACPI_VIDEO_DISPLAY_LCD:
			data->flags.lcd = 1;
			break;
		default:
			data->flags.unknown = 1;
			break;
		}
		if(attribute->bios_can_detect)
			data->flags.bios = 1;
	} else {
		/* Check for legacy IDs */
		device_type = acpi_video_get_device_type(video, device_id);
		/* Ignore bits 16 and 18-20 */
		switch (device_type & 0xffe2ffff) {
1170 1171 1172 1173 1174 1175 1176 1177 1178 1179 1180 1181
			case ACPI_VIDEO_DISPLAY_LEGACY_MONITOR:
				data->flags.crt = 1;
				break;
			case ACPI_VIDEO_DISPLAY_LEGACY_PANEL:
				data->flags.lcd = 1;
				break;
			case ACPI_VIDEO_DISPLAY_LEGACY_TV:
				data->flags.tvout = 1;
				break;
			default:
				data->flags.unknown = 1;
		}
1182
	}
L
Len Brown 已提交
1183

1184 1185
	acpi_video_device_bind(video, data);
	acpi_video_device_find_cap(data);
L
Linus Torvalds 已提交
1186

1187 1188 1189 1190 1191 1192
	status = acpi_install_notify_handler(device->handle, ACPI_DEVICE_NOTIFY,
					     acpi_video_device_notify, data);
	if (ACPI_FAILURE(status))
		dev_err(&device->dev, "Error installing notify handler\n");
	else
		data->flags.notify = 1;
L
Linus Torvalds 已提交
1193

1194 1195 1196
	mutex_lock(&video->device_list_lock);
	list_add_tail(&data->entry, &video->video_device_list);
	mutex_unlock(&video->device_list_lock);
L
Linus Torvalds 已提交
1197

1198
	return status;
L
Linus Torvalds 已提交
1199 1200 1201 1202
}

/*
 *  Arg:
1203
 *	video	: video bus device
L
Linus Torvalds 已提交
1204 1205
 *
 *  Return:
1206 1207 1208
 *	none
 *
 *  Enumerate the video device list of the video bus,
L
Linus Torvalds 已提交
1209 1210
 *  bind the ids with the corresponding video devices
 *  under the video bus.
L
Len Brown 已提交
1211
 */
L
Linus Torvalds 已提交
1212

L
Len Brown 已提交
1213
static void acpi_video_device_rebind(struct acpi_video_bus *video)
L
Linus Torvalds 已提交
1214
{
1215 1216
	struct acpi_video_device *dev;

1217
	mutex_lock(&video->device_list_lock);
1218 1219

	list_for_each_entry(dev, &video->video_device_list, entry)
L
Len Brown 已提交
1220
		acpi_video_device_bind(video, dev);
1221

1222
	mutex_unlock(&video->device_list_lock);
L
Linus Torvalds 已提交
1223 1224 1225 1226
}

/*
 *  Arg:
1227 1228 1229
 *	video	: video bus device
 *	device	: video output device under the video
 *		bus
L
Linus Torvalds 已提交
1230 1231
 *
 *  Return:
1232 1233
 *	none
 *
L
Linus Torvalds 已提交
1234 1235
 *  Bind the ids with the corresponding video devices
 *  under the video bus.
L
Len Brown 已提交
1236
 */
L
Linus Torvalds 已提交
1237 1238

static void
L
Len Brown 已提交
1239 1240
acpi_video_device_bind(struct acpi_video_bus *video,
		       struct acpi_video_device *device)
L
Linus Torvalds 已提交
1241
{
1242
	struct acpi_video_enumerated_device *ids;
L
Len Brown 已提交
1243
	int i;
L
Linus Torvalds 已提交
1244

1245 1246 1247 1248
	for (i = 0; i < video->attached_count; i++) {
		ids = &video->attached_array[i];
		if (device->device_id == (ids->value.int_val & 0xffff)) {
			ids->bind_info = device;
L
Linus Torvalds 已提交
1249 1250 1251 1252 1253 1254 1255
			ACPI_DEBUG_PRINT((ACPI_DB_INFO, "device_bind %d\n", i));
		}
	}
}

/*
 *  Arg:
1256
 *	video	: video bus device
L
Linus Torvalds 已提交
1257 1258
 *
 *  Return:
1259 1260
 *	< 0	: error
 *
L
Linus Torvalds 已提交
1261 1262
 *  Call _DOD to enumerate all devices attached to display adapter
 *
L
Len Brown 已提交
1263
 */
L
Linus Torvalds 已提交
1264 1265 1266

static int acpi_video_device_enumerate(struct acpi_video_bus *video)
{
L
Len Brown 已提交
1267 1268 1269
	int status;
	int count;
	int i;
1270
	struct acpi_video_enumerated_device *active_list;
L
Len Brown 已提交
1271 1272 1273
	struct acpi_buffer buffer = { ACPI_ALLOCATE_BUFFER, NULL };
	union acpi_object *dod = NULL;
	union acpi_object *obj;
L
Linus Torvalds 已提交
1274

1275
	status = acpi_evaluate_object(video->device->handle, "_DOD", NULL, &buffer);
L
Linus Torvalds 已提交
1276
	if (!ACPI_SUCCESS(status)) {
1277
		ACPI_EXCEPTION((AE_INFO, status, "Evaluating _DOD"));
1278
		return status;
L
Linus Torvalds 已提交
1279 1280
	}

1281
	dod = buffer.pointer;
L
Linus Torvalds 已提交
1282
	if (!dod || (dod->type != ACPI_TYPE_PACKAGE)) {
1283
		ACPI_EXCEPTION((AE_INFO, status, "Invalid _DOD data"));
L
Linus Torvalds 已提交
1284 1285 1286 1287 1288
		status = -EFAULT;
		goto out;
	}

	ACPI_DEBUG_PRINT((ACPI_DB_INFO, "Found %d video heads in _DOD\n",
L
Len Brown 已提交
1289
			  dod->package.count));
L
Linus Torvalds 已提交
1290

1291 1292 1293 1294
	active_list = kcalloc(1 + dod->package.count,
			      sizeof(struct acpi_video_enumerated_device),
			      GFP_KERNEL);
	if (!active_list) {
L
Linus Torvalds 已提交
1295 1296 1297 1298 1299 1300
		status = -ENOMEM;
		goto out;
	}

	count = 0;
	for (i = 0; i < dod->package.count; i++) {
1301
		obj = &dod->package.elements[i];
L
Linus Torvalds 已提交
1302 1303

		if (obj->type != ACPI_TYPE_INTEGER) {
1304 1305 1306
			printk(KERN_ERR PREFIX
				"Invalid _DOD data in element %d\n", i);
			continue;
L
Linus Torvalds 已提交
1307
		}
1308 1309 1310

		active_list[count].value.int_val = obj->integer.value;
		active_list[count].bind_info = NULL;
L
Len Brown 已提交
1311 1312
		ACPI_DEBUG_PRINT((ACPI_DB_INFO, "dod element[%d] = %d\n", i,
				  (int)obj->integer.value));
L
Linus Torvalds 已提交
1313 1314 1315
		count++;
	}

1316
	kfree(video->attached_array);
L
Len Brown 已提交
1317

1318
	video->attached_array = active_list;
L
Linus Torvalds 已提交
1319
	video->attached_count = count;
1320 1321

 out:
1322
	kfree(buffer.pointer);
1323
	return status;
L
Linus Torvalds 已提交
1324 1325
}

L
Len Brown 已提交
1326 1327 1328
static int
acpi_video_get_next_level(struct acpi_video_device *device,
			  u32 level_current, u32 event)
L
Linus Torvalds 已提交
1329
{
1330
	int min, max, min_above, max_below, i, l, delta = 255;
1331 1332
	max = max_below = 0;
	min = min_above = 255;
1333
	/* Find closest level to level_current */
1334
	for (i = 2; i < device->brightness->count; i++) {
1335 1336 1337 1338 1339 1340 1341 1342 1343
		l = device->brightness->levels[i];
		if (abs(l - level_current) < abs(delta)) {
			delta = l - level_current;
			if (!delta)
				break;
		}
	}
	/* Ajust level_current to closest available level */
	level_current += delta;
1344
	for (i = 2; i < device->brightness->count; i++) {
1345 1346 1347 1348 1349 1350 1351 1352 1353 1354 1355 1356 1357 1358 1359 1360 1361 1362 1363 1364 1365 1366 1367 1368
		l = device->brightness->levels[i];
		if (l < min)
			min = l;
		if (l > max)
			max = l;
		if (l < min_above && l > level_current)
			min_above = l;
		if (l > max_below && l < level_current)
			max_below = l;
	}

	switch (event) {
	case ACPI_VIDEO_NOTIFY_CYCLE_BRIGHTNESS:
		return (level_current < max) ? min_above : min;
	case ACPI_VIDEO_NOTIFY_INC_BRIGHTNESS:
		return (level_current < max) ? min_above : max;
	case ACPI_VIDEO_NOTIFY_DEC_BRIGHTNESS:
		return (level_current > min) ? max_below : min;
	case ACPI_VIDEO_NOTIFY_ZERO_BRIGHTNESS:
	case ACPI_VIDEO_NOTIFY_DISPLAY_OFF:
		return 0;
	default:
		return level_current;
	}
L
Linus Torvalds 已提交
1369 1370
}

1371
static int
L
Len Brown 已提交
1372
acpi_video_switch_brightness(struct acpi_video_device *device, int event)
L
Linus Torvalds 已提交
1373
{
1374
	unsigned long long level_current, level_next;
1375 1376
	int result = -EINVAL;

1377 1378
	/* no warning message if acpi_backlight=vendor is used */
	if (!acpi_video_backlight_support())
1379 1380
		return 0;

1381
	if (!device->brightness)
1382 1383 1384
		goto out;

	result = acpi_video_device_lcd_get_level_current(device,
1385 1386
							 &level_current,
							 false);
1387 1388 1389
	if (result)
		goto out;

L
Linus Torvalds 已提交
1390
	level_next = acpi_video_get_next_level(device, level_current, event);
1391

1392
	result = acpi_video_device_lcd_set_level(device, level_next);
1393

1394 1395 1396 1397
	if (!result)
		backlight_force_update(device->backlight,
				       BACKLIGHT_UPDATE_HOTKEY);

1398 1399 1400 1401 1402
out:
	if (result)
		printk(KERN_ERR PREFIX "Failed to switch the brightness\n");

	return result;
L
Linus Torvalds 已提交
1403 1404
}

1405 1406 1407 1408 1409 1410 1411 1412 1413 1414 1415 1416 1417 1418 1419 1420 1421 1422 1423 1424 1425
int acpi_video_get_edid(struct acpi_device *device, int type, int device_id,
			void **edid)
{
	struct acpi_video_bus *video;
	struct acpi_video_device *video_device;
	union acpi_object *buffer = NULL;
	acpi_status status;
	int i, length;

	if (!device || !acpi_driver_data(device))
		return -EINVAL;

	video = acpi_driver_data(device);

	for (i = 0; i < video->attached_count; i++) {
		video_device = video->attached_array[i].bind_info;
		length = 256;

		if (!video_device)
			continue;

1426 1427 1428
		if (!video_device->cap._DDC)
			continue;

1429 1430 1431 1432 1433 1434 1435 1436 1437 1438 1439 1440 1441 1442 1443 1444 1445 1446 1447 1448 1449 1450 1451 1452 1453 1454 1455 1456 1457 1458 1459 1460 1461 1462 1463 1464 1465 1466 1467 1468 1469 1470 1471 1472
		if (type) {
			switch (type) {
			case ACPI_VIDEO_DISPLAY_CRT:
				if (!video_device->flags.crt)
					continue;
				break;
			case ACPI_VIDEO_DISPLAY_TV:
				if (!video_device->flags.tvout)
					continue;
				break;
			case ACPI_VIDEO_DISPLAY_DVI:
				if (!video_device->flags.dvi)
					continue;
				break;
			case ACPI_VIDEO_DISPLAY_LCD:
				if (!video_device->flags.lcd)
					continue;
				break;
			}
		} else if (video_device->device_id != device_id) {
			continue;
		}

		status = acpi_video_device_EDID(video_device, &buffer, length);

		if (ACPI_FAILURE(status) || !buffer ||
		    buffer->type != ACPI_TYPE_BUFFER) {
			length = 128;
			status = acpi_video_device_EDID(video_device, &buffer,
							length);
			if (ACPI_FAILURE(status) || !buffer ||
			    buffer->type != ACPI_TYPE_BUFFER) {
				continue;
			}
		}

		*edid = buffer->buffer.pointer;
		return length;
	}

	return -ENODEV;
}
EXPORT_SYMBOL(acpi_video_get_edid);

L
Linus Torvalds 已提交
1473
static int
L
Len Brown 已提交
1474 1475
acpi_video_bus_get_devices(struct acpi_video_bus *video,
			   struct acpi_device *device)
L
Linus Torvalds 已提交
1476
{
1477
	int status = 0;
1478
	struct acpi_device *dev;
L
Linus Torvalds 已提交
1479

1480 1481 1482 1483 1484 1485
	/*
	 * There are systems where video module known to work fine regardless
	 * of broken _DOD and ignoring returned value here doesn't cause
	 * any issues later.
	 */
	acpi_video_device_enumerate(video);
L
Linus Torvalds 已提交
1486

1487
	list_for_each_entry(dev, &device->children, node) {
L
Linus Torvalds 已提交
1488 1489

		status = acpi_video_bus_get_one_device(dev, video);
1490
		if (status) {
1491 1492
			dev_err(&dev->dev, "Can't attach device\n");
			break;
L
Linus Torvalds 已提交
1493 1494
		}
	}
1495
	return status;
L
Linus Torvalds 已提交
1496 1497
}

L
Len Brown 已提交
1498
static int acpi_video_bus_put_one_device(struct acpi_video_device *device)
L
Linus Torvalds 已提交
1499
{
1500
	acpi_status status;
L
Linus Torvalds 已提交
1501 1502

	if (!device || !device->video)
1503
		return -ENOENT;
L
Linus Torvalds 已提交
1504

1505 1506 1507 1508 1509 1510
	if (device->flags.notify) {
		status = acpi_remove_notify_handler(device->dev->handle,
				ACPI_DEVICE_NOTIFY, acpi_video_device_notify);
		if (ACPI_FAILURE(status))
			dev_err(&device->dev->dev,
					"Can't remove video notify handler\n");
1511
	}
1512

1513 1514 1515 1516
	if (device->backlight) {
		backlight_device_unregister(device->backlight);
		device->backlight = NULL;
	}
1517
	if (device->cooling_dev) {
1518 1519
		sysfs_remove_link(&device->dev->dev.kobj,
				  "thermal_cooling");
1520
		sysfs_remove_link(&device->cooling_dev->device.kobj,
1521
				  "device");
1522 1523
		thermal_cooling_device_unregister(device->cooling_dev);
		device->cooling_dev = NULL;
1524
	}
1525

1526
	return 0;
L
Linus Torvalds 已提交
1527 1528
}

L
Len Brown 已提交
1529
static int acpi_video_bus_put_devices(struct acpi_video_bus *video)
L
Linus Torvalds 已提交
1530
{
L
Len Brown 已提交
1531
	int status;
1532
	struct acpi_video_device *dev, *next;
L
Linus Torvalds 已提交
1533

1534
	mutex_lock(&video->device_list_lock);
L
Linus Torvalds 已提交
1535

1536
	list_for_each_entry_safe(dev, next, &video->video_device_list, entry) {
L
Linus Torvalds 已提交
1537

1538
		status = acpi_video_bus_put_one_device(dev);
L
Len Brown 已提交
1539 1540 1541
		if (ACPI_FAILURE(status))
			printk(KERN_WARNING PREFIX
			       "hhuuhhuu bug in acpi video driver.\n");
L
Linus Torvalds 已提交
1542

1543 1544 1545 1546 1547 1548
		if (dev->brightness) {
			kfree(dev->brightness->levels);
			kfree(dev->brightness);
		}
		list_del(&dev->entry);
		kfree(dev);
L
Linus Torvalds 已提交
1549 1550
	}

1551
	mutex_unlock(&video->device_list_lock);
1552

1553
	return 0;
L
Linus Torvalds 已提交
1554 1555 1556 1557
}

/* acpi_video interface */

1558 1559 1560 1561
/*
 * Win8 requires setting bit2 of _DOS to let firmware know it shouldn't
 * preform any automatic brightness change on receiving a notification.
 */
L
Len Brown 已提交
1562
static int acpi_video_bus_start_devices(struct acpi_video_bus *video)
L
Linus Torvalds 已提交
1563
{
1564 1565
	return acpi_video_bus_DOS(video, 0,
				  acpi_video_backlight_quirks() ? 1 : 0);
L
Linus Torvalds 已提交
1566 1567
}

L
Len Brown 已提交
1568
static int acpi_video_bus_stop_devices(struct acpi_video_bus *video)
L
Linus Torvalds 已提交
1569
{
1570 1571
	return acpi_video_bus_DOS(video, 0,
				  acpi_video_backlight_quirks() ? 0 : 1);
L
Linus Torvalds 已提交
1572 1573
}

1574
static void acpi_video_bus_notify(struct acpi_device *device, u32 event)
L
Linus Torvalds 已提交
1575
{
1576
	struct acpi_video_bus *video = acpi_driver_data(device);
1577
	struct input_dev *input;
1578
	int keycode = 0;
1579

L
Linus Torvalds 已提交
1580
	if (!video)
1581
		return;
L
Linus Torvalds 已提交
1582

1583
	input = video->input;
L
Linus Torvalds 已提交
1584 1585

	switch (event) {
1586
	case ACPI_VIDEO_NOTIFY_SWITCH:	/* User requested a switch,
L
Linus Torvalds 已提交
1587
					 * most likely via hotkey. */
1588
		acpi_bus_generate_proc_event(device, event, 0);
1589
		keycode = KEY_SWITCHVIDEOMODE;
L
Linus Torvalds 已提交
1590 1591
		break;

1592
	case ACPI_VIDEO_NOTIFY_PROBE:	/* User plugged in or removed a video
L
Linus Torvalds 已提交
1593 1594 1595
					 * connector. */
		acpi_video_device_enumerate(video);
		acpi_video_device_rebind(video);
1596
		acpi_bus_generate_proc_event(device, event, 0);
1597
		keycode = KEY_SWITCHVIDEOMODE;
L
Linus Torvalds 已提交
1598 1599
		break;

L
Len Brown 已提交
1600
	case ACPI_VIDEO_NOTIFY_CYCLE:	/* Cycle Display output hotkey pressed. */
L
Len Brown 已提交
1601
		acpi_bus_generate_proc_event(device, event, 0);
1602 1603
		keycode = KEY_SWITCHVIDEOMODE;
		break;
L
Len Brown 已提交
1604
	case ACPI_VIDEO_NOTIFY_NEXT_OUTPUT:	/* Next Display output hotkey pressed. */
L
Len Brown 已提交
1605
		acpi_bus_generate_proc_event(device, event, 0);
1606 1607
		keycode = KEY_VIDEO_NEXT;
		break;
L
Len Brown 已提交
1608
	case ACPI_VIDEO_NOTIFY_PREV_OUTPUT:	/* previous Display output hotkey pressed. */
1609
		acpi_bus_generate_proc_event(device, event, 0);
1610
		keycode = KEY_VIDEO_PREV;
L
Linus Torvalds 已提交
1611 1612 1613 1614
		break;

	default:
		ACPI_DEBUG_PRINT((ACPI_DB_INFO,
L
Len Brown 已提交
1615
				  "Unsupported event [0x%x]\n", event));
L
Linus Torvalds 已提交
1616 1617 1618
		break;
	}

1619 1620 1621
	if (acpi_notifier_call_chain(device, event, 0))
		/* Something vetoed the keypress. */
		keycode = 0;
1622 1623 1624 1625 1626 1627 1628

	if (keycode) {
		input_report_key(input, keycode, 1);
		input_sync(input);
		input_report_key(input, keycode, 0);
		input_sync(input);
	}
1629

1630
	return;
L
Linus Torvalds 已提交
1631 1632
}

L
Len Brown 已提交
1633
static void acpi_video_device_notify(acpi_handle handle, u32 event, void *data)
L
Linus Torvalds 已提交
1634
{
1635
	struct acpi_video_device *video_device = data;
L
Len Brown 已提交
1636
	struct acpi_device *device = NULL;
1637 1638
	struct acpi_video_bus *bus;
	struct input_dev *input;
1639
	int keycode = 0;
L
Linus Torvalds 已提交
1640 1641

	if (!video_device)
1642
		return;
L
Linus Torvalds 已提交
1643

1644
	device = video_device->dev;
1645 1646
	bus = video_device->video;
	input = bus->input;
L
Linus Torvalds 已提交
1647 1648

	switch (event) {
L
Len Brown 已提交
1649
	case ACPI_VIDEO_NOTIFY_CYCLE_BRIGHTNESS:	/* Cycle brightness */
1650 1651
		if (brightness_switch_enabled)
			acpi_video_switch_brightness(video_device, event);
1652
		acpi_bus_generate_proc_event(device, event, 0);
1653 1654
		keycode = KEY_BRIGHTNESS_CYCLE;
		break;
L
Len Brown 已提交
1655
	case ACPI_VIDEO_NOTIFY_INC_BRIGHTNESS:	/* Increase brightness */
1656 1657
		if (brightness_switch_enabled)
			acpi_video_switch_brightness(video_device, event);
L
Len Brown 已提交
1658
		acpi_bus_generate_proc_event(device, event, 0);
1659 1660
		keycode = KEY_BRIGHTNESSUP;
		break;
L
Len Brown 已提交
1661
	case ACPI_VIDEO_NOTIFY_DEC_BRIGHTNESS:	/* Decrease brightness */
1662 1663
		if (brightness_switch_enabled)
			acpi_video_switch_brightness(video_device, event);
L
Len Brown 已提交
1664
		acpi_bus_generate_proc_event(device, event, 0);
1665 1666
		keycode = KEY_BRIGHTNESSDOWN;
		break;
1667
	case ACPI_VIDEO_NOTIFY_ZERO_BRIGHTNESS:	/* zero brightness */
1668 1669
		if (brightness_switch_enabled)
			acpi_video_switch_brightness(video_device, event);
L
Len Brown 已提交
1670
		acpi_bus_generate_proc_event(device, event, 0);
1671 1672
		keycode = KEY_BRIGHTNESS_ZERO;
		break;
L
Len Brown 已提交
1673
	case ACPI_VIDEO_NOTIFY_DISPLAY_OFF:	/* display device off */
1674 1675
		if (brightness_switch_enabled)
			acpi_video_switch_brightness(video_device, event);
1676
		acpi_bus_generate_proc_event(device, event, 0);
1677
		keycode = KEY_DISPLAY_OFF;
L
Linus Torvalds 已提交
1678 1679 1680
		break;
	default:
		ACPI_DEBUG_PRINT((ACPI_DB_INFO,
L
Len Brown 已提交
1681
				  "Unsupported event [0x%x]\n", event));
L
Linus Torvalds 已提交
1682 1683
		break;
	}
1684

1685
	acpi_notifier_call_chain(device, event, 0);
1686 1687 1688 1689 1690 1691 1692

	if (keycode) {
		input_report_key(input, keycode, 1);
		input_sync(input);
		input_report_key(input, keycode, 0);
		input_sync(input);
	}
1693

1694
	return;
L
Linus Torvalds 已提交
1695 1696
}

1697 1698
static int acpi_video_resume(struct notifier_block *nb,
				unsigned long val, void *ign)
1699 1700 1701 1702 1703
{
	struct acpi_video_bus *video;
	struct acpi_video_device *video_device;
	int i;

1704 1705 1706 1707 1708 1709
	switch (val) {
	case PM_HIBERNATION_PREPARE:
	case PM_SUSPEND_PREPARE:
	case PM_RESTORE_PREPARE:
		return NOTIFY_DONE;
	}
1710

1711 1712 1713
	video = container_of(nb, struct acpi_video_bus, pm_nb);

	dev_info(&video->device->dev, "Restoring backlight state\n");
1714 1715 1716 1717 1718 1719

	for (i = 0; i < video->attached_count; i++) {
		video_device = video->attached_array[i].bind_info;
		if (video_device && video_device->backlight)
			acpi_video_set_brightness(video_device->backlight);
	}
1720 1721

	return NOTIFY_OK;
1722 1723
}

1724 1725 1726 1727 1728 1729 1730 1731 1732 1733 1734 1735 1736 1737 1738 1739 1740 1741 1742 1743 1744
static acpi_status
acpi_video_bus_match(acpi_handle handle, u32 level, void *context,
			void **return_value)
{
	struct acpi_device *device = context;
	struct acpi_device *sibling;
	int result;

	if (handle == device->handle)
		return AE_CTRL_TERMINATE;

	result = acpi_bus_get_device(handle, &sibling);
	if (result)
		return AE_OK;

	if (!strcmp(acpi_device_name(sibling), ACPI_VIDEO_BUS_NAME))
			return AE_ALREADY_EXISTS;

	return AE_OK;
}

1745 1746
static int instance;

L
Len Brown 已提交
1747
static int acpi_video_bus_add(struct acpi_device *device)
L
Linus Torvalds 已提交
1748
{
1749
	struct acpi_video_bus *video;
1750
	struct input_dev *input;
1751
	int error;
1752 1753 1754 1755 1756 1757 1758 1759 1760 1761 1762 1763 1764 1765 1766
	acpi_status status;

	status = acpi_walk_namespace(ACPI_TYPE_DEVICE,
				device->parent->handle, 1,
				acpi_video_bus_match, NULL,
				device, NULL);
	if (status == AE_ALREADY_EXISTS) {
		printk(KERN_WARNING FW_BUG
			"Duplicate ACPI video bus devices for the"
			" same VGA controller, please try module "
			"parameter \"video.allow_duplicates=1\""
			"if the current driver doesn't work.\n");
		if (!allow_duplicates)
			return -ENODEV;
	}
L
Linus Torvalds 已提交
1767

1768
	video = kzalloc(sizeof(struct acpi_video_bus), GFP_KERNEL);
L
Linus Torvalds 已提交
1769
	if (!video)
1770
		return -ENOMEM;
L
Linus Torvalds 已提交
1771

1772 1773 1774 1775 1776 1777
	/* a hack to fix the duplicate name "VID" problem on T61 */
	if (!strcmp(device->pnp.bus_id, "VID")) {
		if (instance)
			device->pnp.bus_id[3] = '0' + instance;
		instance ++;
	}
1778 1779 1780 1781 1782 1783
	/* a hack to fix the duplicate name "VGA" problem on Pa 3553 */
	if (!strcmp(device->pnp.bus_id, "VGA")) {
		if (instance)
			device->pnp.bus_id[3] = '0' + instance;
		instance++;
	}
1784

1785
	video->device = device;
L
Linus Torvalds 已提交
1786 1787
	strcpy(acpi_device_name(device), ACPI_VIDEO_BUS_NAME);
	strcpy(acpi_device_class(device), ACPI_VIDEO_CLASS);
1788
	device->driver_data = video;
L
Linus Torvalds 已提交
1789 1790

	acpi_video_bus_find_cap(video);
1791 1792 1793
	error = acpi_video_bus_check(video);
	if (error)
		goto err_free_video;
L
Linus Torvalds 已提交
1794

1795
	mutex_init(&video->device_list_lock);
L
Linus Torvalds 已提交
1796 1797
	INIT_LIST_HEAD(&video->video_device_list);

1798 1799
	error = acpi_video_bus_get_devices(video, device);
	if (error)
1800
		goto err_put_video;
L
Linus Torvalds 已提交
1801

1802
	video->input = input = input_allocate_device();
1803 1804
	if (!input) {
		error = -ENOMEM;
1805
		goto err_put_video;
1806
	}
1807

1808 1809 1810 1811
	error = acpi_video_bus_start_devices(video);
	if (error)
		goto err_free_input_dev;

1812 1813 1814 1815 1816 1817 1818
	snprintf(video->phys, sizeof(video->phys),
		"%s/video/input0", acpi_device_hid(video->device));

	input->name = acpi_device_name(video->device);
	input->phys = video->phys;
	input->id.bustype = BUS_HOST;
	input->id.product = 0x06;
1819
	input->dev.parent = &device->dev;
1820 1821 1822 1823 1824 1825 1826 1827 1828 1829
	input->evbit[0] = BIT(EV_KEY);
	set_bit(KEY_SWITCHVIDEOMODE, input->keybit);
	set_bit(KEY_VIDEO_NEXT, input->keybit);
	set_bit(KEY_VIDEO_PREV, input->keybit);
	set_bit(KEY_BRIGHTNESS_CYCLE, input->keybit);
	set_bit(KEY_BRIGHTNESSUP, input->keybit);
	set_bit(KEY_BRIGHTNESSDOWN, input->keybit);
	set_bit(KEY_BRIGHTNESS_ZERO, input->keybit);
	set_bit(KEY_DISPLAY_OFF, input->keybit);

L
Linus Torvalds 已提交
1830
	printk(KERN_INFO PREFIX "%s [%s] (multi-head: %s  rom: %s  post: %s)\n",
L
Len Brown 已提交
1831 1832 1833 1834
	       ACPI_VIDEO_DEVICE_NAME, acpi_device_bid(device),
	       video->flags.multihead ? "yes" : "no",
	       video->flags.rom ? "yes" : "no",
	       video->flags.post ? "yes" : "no");
L
Linus Torvalds 已提交
1835

1836 1837
	video->pm_nb.notifier_call = acpi_video_resume;
	video->pm_nb.priority = 0;
1838 1839
	error = register_pm_notifier(&video->pm_nb);
	if (error)
1840 1841 1842 1843 1844
		goto err_stop_video;

	error = input_register_device(input);
	if (error)
		goto err_unregister_pm_notifier;
1845

1846 1847
	return 0;

1848 1849
 err_unregister_pm_notifier:
	unregister_pm_notifier(&video->pm_nb);
1850 1851
 err_stop_video:
	acpi_video_bus_stop_devices(video);
1852 1853
 err_free_input_dev:
	input_free_device(input);
1854
 err_put_video:
1855 1856 1857 1858
	acpi_video_bus_put_devices(video);
	kfree(video->attached_array);
 err_free_video:
	kfree(video);
1859
	device->driver_data = NULL;
L
Linus Torvalds 已提交
1860

1861
	return error;
L
Linus Torvalds 已提交
1862 1863
}

1864
static int acpi_video_bus_remove(struct acpi_device *device)
L
Linus Torvalds 已提交
1865
{
L
Len Brown 已提交
1866
	struct acpi_video_bus *video = NULL;
L
Linus Torvalds 已提交
1867 1868 1869


	if (!device || !acpi_driver_data(device))
1870
		return -EINVAL;
L
Linus Torvalds 已提交
1871

1872
	video = acpi_driver_data(device);
L
Linus Torvalds 已提交
1873

1874 1875
	unregister_pm_notifier(&video->pm_nb);

L
Linus Torvalds 已提交
1876 1877 1878
	acpi_video_bus_stop_devices(video);
	acpi_video_bus_put_devices(video);

1879
	input_unregister_device(video->input);
1880
	kfree(video->attached_array);
L
Linus Torvalds 已提交
1881 1882
	kfree(video);

1883
	return 0;
L
Linus Torvalds 已提交
1884 1885
}

1886 1887 1888 1889 1890 1891 1892 1893 1894
static int __init is_i740(struct pci_dev *dev)
{
	if (dev->device == 0x00D1)
		return 1;
	if (dev->device == 0x7000)
		return 1;
	return 0;
}

1895 1896
static int __init intel_opregion_present(void)
{
1897
	int opregion = 0;
1898 1899 1900 1901 1902 1903 1904 1905
	struct pci_dev *dev = NULL;
	u32 address;

	for_each_pci_dev(dev) {
		if ((dev->class >> 8) != PCI_CLASS_DISPLAY_VGA)
			continue;
		if (dev->vendor != PCI_VENDOR_ID_INTEL)
			continue;
1906 1907 1908
		/* We don't want to poke around undefined i740 registers */
		if (is_i740(dev))
			continue;
1909 1910 1911
		pci_read_config_dword(dev, 0xfc, &address);
		if (!address)
			continue;
1912
		opregion = 1;
1913
	}
1914
	return opregion;
1915 1916
}

1917
int acpi_video_register(void)
L
Linus Torvalds 已提交
1918
{
1919
	int result = 0;
1920 1921
	if (register_count) {
		/*
1922 1923
		 * if the function of acpi_video_register is already called,
		 * don't register the acpi_vide_bus again and return no error.
1924 1925 1926
		 */
		return 0;
	}
L
Linus Torvalds 已提交
1927 1928

	result = acpi_bus_register_driver(&acpi_video_bus);
1929
	if (result < 0)
1930
		return -ENODEV;
L
Linus Torvalds 已提交
1931

1932 1933 1934 1935 1936 1937
	/*
	 * When the acpi_video_bus is loaded successfully, increase
	 * the counter reference.
	 */
	register_count = 1;

1938
	return 0;
L
Linus Torvalds 已提交
1939
}
1940
EXPORT_SYMBOL(acpi_video_register);
1941

1942 1943 1944 1945 1946 1947 1948 1949 1950 1951 1952 1953 1954 1955 1956 1957 1958
void acpi_video_unregister(void)
{
	if (!register_count) {
		/*
		 * If the acpi video bus is already unloaded, don't
		 * unload it again and return directly.
		 */
		return;
	}
	acpi_bus_unregister_driver(&acpi_video_bus);

	register_count = 0;

	return;
}
EXPORT_SYMBOL(acpi_video_unregister);

1959 1960 1961 1962 1963 1964 1965 1966 1967
/*
 * This is kind of nasty. Hardware using Intel chipsets may require
 * the video opregion code to be run first in order to initialise
 * state before any ACPI video calls are made. To handle this we defer
 * registration of the video class until the opregion code has run.
 */

static int __init acpi_video_init(void)
{
1968 1969
	dmi_check_system(video_dmi_table);

1970 1971 1972 1973 1974
	if (intel_opregion_present())
		return 0;

	return acpi_video_register();
}
L
Linus Torvalds 已提交
1975

1976
static void __exit acpi_video_exit(void)
L
Linus Torvalds 已提交
1977
{
1978
	acpi_video_unregister();
L
Linus Torvalds 已提交
1979

1980
	return;
L
Linus Torvalds 已提交
1981 1982 1983 1984
}

module_init(acpi_video_init);
module_exit(acpi_video_exit);