hp-wmi.c 23.6 KB
Newer Older
1 2 3 4
/*
 * HP WMI hotkeys
 *
 * Copyright (C) 2008 Red Hat <mjg@redhat.com>
5
 * Copyright (C) 2010, 2011 Anssi Hannula <anssi.hannula@iki.fi>
6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26
 *
 * Portions based on wistron_btns.c:
 * Copyright (C) 2005 Miloslav Trmac <mitr@volny.cz>
 * Copyright (C) 2005 Bernhard Rosenkraenzer <bero@arklinux.org>
 * Copyright (C) 2005 Dmitry Torokhov <dtor@mail.ru>
 *
 *  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
 */

27 28
#define pr_fmt(fmt) KBUILD_MODNAME ": " fmt

29 30 31
#include <linux/kernel.h>
#include <linux/module.h>
#include <linux/init.h>
32
#include <linux/slab.h>
33 34
#include <linux/types.h>
#include <linux/input.h>
35
#include <linux/input/sparse-keymap.h>
36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53
#include <linux/platform_device.h>
#include <linux/acpi.h>
#include <linux/rfkill.h>
#include <linux/string.h>

MODULE_AUTHOR("Matthew Garrett <mjg59@srcf.ucam.org>");
MODULE_DESCRIPTION("HP laptop WMI hotkeys driver");
MODULE_LICENSE("GPL");

MODULE_ALIAS("wmi:95F24279-4D7B-4334-9387-ACCDC67EF61C");
MODULE_ALIAS("wmi:5FB7F034-2C63-45e9-BE91-3D44E2C707E4");

#define HPWMI_EVENT_GUID "95F24279-4D7B-4334-9387-ACCDC67EF61C"
#define HPWMI_BIOS_GUID "5FB7F034-2C63-45e9-BE91-3D44E2C707E4"

#define HPWMI_DISPLAY_QUERY 0x1
#define HPWMI_HDDTEMP_QUERY 0x2
#define HPWMI_ALS_QUERY 0x3
54
#define HPWMI_HARDWARE_QUERY 0x4
55
#define HPWMI_WIRELESS_QUERY 0x5
56
#define HPWMI_HOTKEY_QUERY 0xc
57
#define HPWMI_WIRELESS2_QUERY 0x1b
58

A
Alan Jenkins 已提交
59 60 61 62
enum hp_wmi_radio {
	HPWMI_WIFI = 0,
	HPWMI_BLUETOOTH = 1,
	HPWMI_WWAN = 2,
63
	HPWMI_GPS = 3,
A
Alan Jenkins 已提交
64 65
};

66 67
enum hp_wmi_event_ids {
	HPWMI_DOCK_EVENT = 1,
68 69
	HPWMI_PARK_HDD = 2,
	HPWMI_SMART_ADAPTER = 3,
70 71
	HPWMI_BEZEL_BUTTON = 4,
	HPWMI_WIRELESS = 5,
72 73
	HPWMI_CPU_BATTERY_THROTTLE = 6,
	HPWMI_LOCK_SWITCH = 7,
74 75 76 77 78 79 80 81
	HPWMI_LID_SWITCH = 8,
	HPWMI_SCREEN_ROTATION = 9,
	HPWMI_COOLSENSE_SYSTEM_MOBILE = 0x0A,
	HPWMI_COOLSENSE_SYSTEM_HOT = 0x0B,
	HPWMI_PROXIMITY_SENSOR = 0x0C,
	HPWMI_BACKLIT_KB_BRIGHTNESS = 0x0D,
	HPWMI_PEAKSHIFT_PERIOD = 0x0F,
	HPWMI_BATTERY_CHARGE_PERIOD = 0x10,
82 83
};

84 85 86 87 88
struct bios_args {
	u32 signature;
	u32 command;
	u32 commandtype;
	u32 datasize;
M
Matthew Garrett 已提交
89
	u32 data;
90 91 92 93 94 95 96
};

struct bios_return {
	u32 sigpass;
	u32 return_code;
};

97 98 99 100 101 102 103
enum hp_return_value {
	HPWMI_RET_WRONG_SIGNATURE	= 0x02,
	HPWMI_RET_UNKNOWN_COMMAND	= 0x03,
	HPWMI_RET_UNKNOWN_CMDTYPE	= 0x04,
	HPWMI_RET_INVALID_PARAMETERS	= 0x05,
};

104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136
enum hp_wireless2_bits {
	HPWMI_POWER_STATE	= 0x01,
	HPWMI_POWER_SOFT	= 0x02,
	HPWMI_POWER_BIOS	= 0x04,
	HPWMI_POWER_HARD	= 0x08,
};

#define IS_HWBLOCKED(x) ((x & (HPWMI_POWER_BIOS | HPWMI_POWER_HARD)) \
			 != (HPWMI_POWER_BIOS | HPWMI_POWER_HARD))
#define IS_SWBLOCKED(x) !(x & HPWMI_POWER_SOFT)

struct bios_rfkill2_device_state {
	u8 radio_type;
	u8 bus_type;
	u16 vendor_id;
	u16 product_id;
	u16 subsys_vendor_id;
	u16 subsys_product_id;
	u8 rfkill_id;
	u8 power;
	u8 unknown[4];
};

/* 7 devices fit into the 128 byte buffer */
#define HPWMI_MAX_RFKILL2_DEVICES	7

struct bios_rfkill2_state {
	u8 unknown[7];
	u8 count;
	u8 pad[8];
	struct bios_rfkill2_device_state device[HPWMI_MAX_RFKILL2_DEVICES];
};

137 138 139 140 141 142 143 144 145 146
static const struct key_entry hp_wmi_keymap[] = {
	{ KE_KEY, 0x02,   { KEY_BRIGHTNESSUP } },
	{ KE_KEY, 0x03,   { KEY_BRIGHTNESSDOWN } },
	{ KE_KEY, 0x20e6, { KEY_PROG1 } },
	{ KE_KEY, 0x20e8, { KEY_MEDIA } },
	{ KE_KEY, 0x2142, { KEY_MEDIA } },
	{ KE_KEY, 0x213b, { KEY_INFO } },
	{ KE_KEY, 0x2169, { KEY_DIRECTION } },
	{ KE_KEY, 0x231b, { KEY_HELP } },
	{ KE_END, 0 }
147 148 149 150 151 152 153 154
};

static struct input_dev *hp_wmi_input_dev;
static struct platform_device *hp_wmi_platform_dev;

static struct rfkill *wifi_rfkill;
static struct rfkill *bluetooth_rfkill;
static struct rfkill *wwan_rfkill;
155
static struct rfkill *gps_rfkill;
156

157 158 159 160 161 162 163 164 165
struct rfkill2_device {
	u8 id;
	int num;
	struct rfkill *rfkill;
};

static int rfkill2_count;
static struct rfkill2_device rfkill2[HPWMI_MAX_RFKILL2_DEVICES];

166 167 168 169 170 171
/*
 * hp_wmi_perform_query
 *
 * query:	The commandtype -> What should be queried
 * write:	The command -> 0 read, 1 write, 3 ODM specific
 * buffer:	Buffer used as input and/or output
172 173
 * insize:	Size of input buffer
 * outsize:	Size of output buffer
174 175 176 177 178 179 180 181 182 183
 *
 * returns zero on success
 *         an HP WMI query specific error code (which is positive)
 *         -EINVAL if the query was not successful at all
 *         -EINVAL if the output buffer size exceeds buffersize
 *
 * Note: The buffersize must at least be the maximum of the input and output
 *       size. E.g. Battery info query (0x7) is defined to have 1 byte input
 *       and 128 byte output. The caller would do:
 *       buffer = kzalloc(128, GFP_KERNEL);
184
 *       ret = hp_wmi_perform_query(0x7, 0, buffer, 1, 128)
185
 */
186 187
static int hp_wmi_perform_query(int query, int write, void *buffer,
				int insize, int outsize)
188
{
189 190
	struct bios_return *bios_return;
	int actual_outsize;
191 192 193 194 195
	union acpi_object *obj;
	struct bios_args args = {
		.signature = 0x55434553,
		.command = write ? 0x2 : 0x1,
		.commandtype = query,
196 197
		.datasize = insize,
		.data = 0,
198 199 200
	};
	struct acpi_buffer input = { sizeof(struct bios_args), &args };
	struct acpi_buffer output = { ACPI_ALLOCATE_BUFFER, NULL };
E
Eric Dumazet 已提交
201
	u32 rc;
202

203 204 205 206
	if (WARN_ON(insize > sizeof(args.data)))
		return -EINVAL;
	memcpy(&args.data, buffer, insize);

207
	wmi_evaluate_method(HPWMI_BIOS_GUID, 0, 0x3, &input, &output);
208 209 210

	obj = output.pointer;

T
Thomas Renninger 已提交
211
	if (!obj)
212
		return -EINVAL;
T
Thomas Renninger 已提交
213 214 215 216
	else if (obj->type != ACPI_TYPE_BUFFER) {
		kfree(obj);
		return -EINVAL;
	}
217

218
	bios_return = (struct bios_return *)obj->buffer.pointer;
E
Eric Dumazet 已提交
219
	rc = bios_return->return_code;
220

E
Eric Dumazet 已提交
221 222 223
	if (rc) {
		if (rc != HPWMI_RET_UNKNOWN_CMDTYPE)
			pr_warn("query 0x%x returned error 0x%x\n", query, rc);
224
		kfree(obj);
E
Eric Dumazet 已提交
225
		return rc;
226 227
	}

228 229 230 231 232
	if (!outsize) {
		/* ignore output data */
		kfree(obj);
		return 0;
	}
233

234 235 236
	actual_outsize = min(outsize, (int)(obj->buffer.length - sizeof(*bios_return)));
	memcpy(buffer, obj->buffer.pointer + sizeof(*bios_return), actual_outsize);
	memset(buffer + actual_outsize, 0, outsize - actual_outsize);
237
	kfree(obj);
238
	return 0;
239 240 241 242
}

static int hp_wmi_display_state(void)
{
M
Matthew Garrett 已提交
243 244
	int state = 0;
	int ret = hp_wmi_perform_query(HPWMI_DISPLAY_QUERY, 0, &state,
245
				       sizeof(state), sizeof(state));
246 247 248
	if (ret)
		return -EINVAL;
	return state;
249 250 251 252
}

static int hp_wmi_hddtemp_state(void)
{
M
Matthew Garrett 已提交
253 254
	int state = 0;
	int ret = hp_wmi_perform_query(HPWMI_HDDTEMP_QUERY, 0, &state,
255
				       sizeof(state), sizeof(state));
256 257 258
	if (ret)
		return -EINVAL;
	return state;
259 260 261 262
}

static int hp_wmi_als_state(void)
{
M
Matthew Garrett 已提交
263 264
	int state = 0;
	int ret = hp_wmi_perform_query(HPWMI_ALS_QUERY, 0, &state,
265
				       sizeof(state), sizeof(state));
266 267 268
	if (ret)
		return -EINVAL;
	return state;
269 270 271 272
}

static int hp_wmi_dock_state(void)
{
M
Matthew Garrett 已提交
273 274
	int state = 0;
	int ret = hp_wmi_perform_query(HPWMI_HARDWARE_QUERY, 0, &state,
275
				       sizeof(state), sizeof(state));
276

277 278
	if (ret)
		return -EINVAL;
279

280
	return state & 0x1;
281 282
}

283
static int hp_wmi_tablet_state(void)
284
{
M
Matthew Garrett 已提交
285 286
	int state = 0;
	int ret = hp_wmi_perform_query(HPWMI_HARDWARE_QUERY, 0, &state,
287
				       sizeof(state), sizeof(state));
288
	if (ret)
289 290
		return ret;

291
	return (state & 0x4) ? 1 : 0;
292 293
}

J
Johannes Berg 已提交
294
static int hp_wmi_set_block(void *data, bool blocked)
295
{
A
Alan Jenkins 已提交
296 297
	enum hp_wmi_radio r = (enum hp_wmi_radio) data;
	int query = BIT(r + 8) | ((!blocked) << r);
298
	int ret;
299

300
	ret = hp_wmi_perform_query(HPWMI_WIRELESS_QUERY, 1,
301
				   &query, sizeof(query), 0);
302 303 304
	if (ret)
		return -EINVAL;
	return 0;
305 306
}

J
Johannes Berg 已提交
307 308 309
static const struct rfkill_ops hp_wmi_rfkill_ops = {
	.set_block = hp_wmi_set_block,
};
310

A
Alan Jenkins 已提交
311
static bool hp_wmi_get_sw_state(enum hp_wmi_radio r)
312
{
M
Matthew Garrett 已提交
313
	int wireless = 0;
314 315
	int mask;
	hp_wmi_perform_query(HPWMI_WIRELESS_QUERY, 0,
316 317
			     &wireless, sizeof(wireless),
			     sizeof(wireless));
318 319 320
	/* TBD: Pass error */

	mask = 0x200 << (r * 8);
321

A
Alan Jenkins 已提交
322
	if (wireless & mask)
J
Johannes Berg 已提交
323
		return false;
324
	else
J
Johannes Berg 已提交
325
		return true;
326 327
}

A
Alan Jenkins 已提交
328
static bool hp_wmi_get_hw_state(enum hp_wmi_radio r)
329
{
M
Matthew Garrett 已提交
330
	int wireless = 0;
331 332
	int mask;
	hp_wmi_perform_query(HPWMI_WIRELESS_QUERY, 0,
333 334
			     &wireless, sizeof(wireless),
			     sizeof(wireless));
335 336 337
	/* TBD: Pass error */

	mask = 0x800 << (r * 8);
338

A
Alan Jenkins 已提交
339
	if (wireless & mask)
J
Johannes Berg 已提交
340
		return false;
341
	else
J
Johannes Berg 已提交
342
		return true;
343 344
}

345 346 347 348 349 350 351 352 353 354 355 356 357 358 359 360 361 362 363 364 365 366 367 368 369 370 371 372 373 374 375 376
static int hp_wmi_rfkill2_set_block(void *data, bool blocked)
{
	int rfkill_id = (int)(long)data;
	char buffer[4] = { 0x01, 0x00, rfkill_id, !blocked };

	if (hp_wmi_perform_query(HPWMI_WIRELESS2_QUERY, 1,
				   buffer, sizeof(buffer), 0))
		return -EINVAL;
	return 0;
}

static const struct rfkill_ops hp_wmi_rfkill2_ops = {
	.set_block = hp_wmi_rfkill2_set_block,
};

static int hp_wmi_rfkill2_refresh(void)
{
	int err, i;
	struct bios_rfkill2_state state;

	err = hp_wmi_perform_query(HPWMI_WIRELESS2_QUERY, 0, &state,
				   0, sizeof(state));
	if (err)
		return err;

	for (i = 0; i < rfkill2_count; i++) {
		int num = rfkill2[i].num;
		struct bios_rfkill2_device_state *devstate;
		devstate = &state.device[num];

		if (num >= state.count ||
		    devstate->rfkill_id != rfkill2[i].id) {
377
			pr_warn("power configuration of the wireless devices unexpectedly changed\n");
378 379 380 381 382 383 384 385 386 387 388
			continue;
		}

		rfkill_set_states(rfkill2[i].rfkill,
				  IS_SWBLOCKED(devstate->power),
				  IS_HWBLOCKED(devstate->power));
	}

	return 0;
}

389 390 391 392 393 394 395 396 397 398 399 400 401 402 403 404 405 406 407 408 409 410 411 412 413 414 415 416 417 418 419 420 421 422 423 424
static ssize_t show_display(struct device *dev, struct device_attribute *attr,
			    char *buf)
{
	int value = hp_wmi_display_state();
	if (value < 0)
		return -EINVAL;
	return sprintf(buf, "%d\n", value);
}

static ssize_t show_hddtemp(struct device *dev, struct device_attribute *attr,
			    char *buf)
{
	int value = hp_wmi_hddtemp_state();
	if (value < 0)
		return -EINVAL;
	return sprintf(buf, "%d\n", value);
}

static ssize_t show_als(struct device *dev, struct device_attribute *attr,
			char *buf)
{
	int value = hp_wmi_als_state();
	if (value < 0)
		return -EINVAL;
	return sprintf(buf, "%d\n", value);
}

static ssize_t show_dock(struct device *dev, struct device_attribute *attr,
			 char *buf)
{
	int value = hp_wmi_dock_state();
	if (value < 0)
		return -EINVAL;
	return sprintf(buf, "%d\n", value);
}

425 426 427 428 429 430 431 432 433
static ssize_t show_tablet(struct device *dev, struct device_attribute *attr,
			 char *buf)
{
	int value = hp_wmi_tablet_state();
	if (value < 0)
		return -EINVAL;
	return sprintf(buf, "%d\n", value);
}

434 435 436 437
static ssize_t set_als(struct device *dev, struct device_attribute *attr,
		       const char *buf, size_t count)
{
	u32 tmp = simple_strtoul(buf, NULL, 10);
M
Matthew Garrett 已提交
438
	int ret = hp_wmi_perform_query(HPWMI_ALS_QUERY, 1, &tmp,
439
				       sizeof(tmp), sizeof(tmp));
440 441 442
	if (ret)
		return -EINVAL;

443 444 445 446 447 448 449
	return count;
}

static DEVICE_ATTR(display, S_IRUGO, show_display, NULL);
static DEVICE_ATTR(hddtemp, S_IRUGO, show_hddtemp, NULL);
static DEVICE_ATTR(als, S_IRUGO | S_IWUSR, show_als, set_als);
static DEVICE_ATTR(dock, S_IRUGO, show_dock, NULL);
450
static DEVICE_ATTR(tablet, S_IRUGO, show_tablet, NULL);
451

A
Adrian Bunk 已提交
452
static void hp_wmi_notify(u32 value, void *context)
453 454 455
{
	struct acpi_buffer response = { ACPI_ALLOCATE_BUFFER, NULL };
	union acpi_object *obj;
456
	u32 event_id, event_data;
M
Matthew Garrett 已提交
457
	int key_code = 0, ret;
458
	u32 *location;
459
	acpi_status status;
460

461 462
	status = wmi_get_event_data(value, &response);
	if (status != AE_OK) {
463
		pr_info("bad event status 0x%x\n", status);
464 465
		return;
	}
466 467 468

	obj = (union acpi_object *)response.pointer;

469 470 471
	if (!obj)
		return;
	if (obj->type != ACPI_TYPE_BUFFER) {
472
		pr_info("Unknown response received %d\n", obj->type);
T
Thomas Renninger 已提交
473
		kfree(obj);
A
Alan Jenkins 已提交
474 475 476
		return;
	}

477 478 479 480 481 482 483 484 485 486 487 488
	/*
	 * Depending on ACPI version the concatenation of id and event data
	 * inside _WED function will result in a 8 or 16 byte buffer.
	 */
	location = (u32 *)obj->buffer.pointer;
	if (obj->buffer.length == 8) {
		event_id = *location;
		event_data = *(location + 1);
	} else if (obj->buffer.length == 16) {
		event_id = *location;
		event_data = *(location + 2);
	} else {
489
		pr_info("Unknown buffer length %d\n", obj->buffer.length);
490 491 492
		kfree(obj);
		return;
	}
T
Thomas Renninger 已提交
493
	kfree(obj);
494 495

	switch (event_id) {
496
	case HPWMI_DOCK_EVENT:
A
Alan Jenkins 已提交
497 498 499 500 501
		input_report_switch(hp_wmi_input_dev, SW_DOCK,
				    hp_wmi_dock_state());
		input_report_switch(hp_wmi_input_dev, SW_TABLET_MODE,
				    hp_wmi_tablet_state());
		input_sync(hp_wmi_input_dev);
502
		break;
503 504 505 506
	case HPWMI_PARK_HDD:
		break;
	case HPWMI_SMART_ADAPTER:
		break;
507
	case HPWMI_BEZEL_BUTTON:
508
		ret = hp_wmi_perform_query(HPWMI_HOTKEY_QUERY, 0,
M
Matthew Garrett 已提交
509
					   &key_code,
510
					   sizeof(key_code),
511 512 513
					   sizeof(key_code));
		if (ret)
			break;
514 515 516

		if (!sparse_keymap_report_event(hp_wmi_input_dev,
						key_code, 1, true))
517
			pr_info("Unknown key code - 0x%x\n", key_code);
518 519
		break;
	case HPWMI_WIRELESS:
520 521 522 523 524
		if (rfkill2_count) {
			hp_wmi_rfkill2_refresh();
			break;
		}

A
Alan Jenkins 已提交
525 526 527 528 529 530 531 532 533 534 535 536
		if (wifi_rfkill)
			rfkill_set_states(wifi_rfkill,
					  hp_wmi_get_sw_state(HPWMI_WIFI),
					  hp_wmi_get_hw_state(HPWMI_WIFI));
		if (bluetooth_rfkill)
			rfkill_set_states(bluetooth_rfkill,
					  hp_wmi_get_sw_state(HPWMI_BLUETOOTH),
					  hp_wmi_get_hw_state(HPWMI_BLUETOOTH));
		if (wwan_rfkill)
			rfkill_set_states(wwan_rfkill,
					  hp_wmi_get_sw_state(HPWMI_WWAN),
					  hp_wmi_get_hw_state(HPWMI_WWAN));
537 538 539 540
		if (gps_rfkill)
			rfkill_set_states(gps_rfkill,
					  hp_wmi_get_sw_state(HPWMI_GPS),
					  hp_wmi_get_hw_state(HPWMI_GPS));
541
		break;
542
	case HPWMI_CPU_BATTERY_THROTTLE:
543
		pr_info("Unimplemented CPU throttle because of 3 Cell battery event detected\n");
544 545 546
		break;
	case HPWMI_LOCK_SWITCH:
		break;
547 548 549 550 551 552 553 554 555 556 557 558 559 560 561 562
	case HPWMI_LID_SWITCH:
		break;
	case HPWMI_SCREEN_ROTATION:
		break;
	case HPWMI_COOLSENSE_SYSTEM_MOBILE:
		break;
	case HPWMI_COOLSENSE_SYSTEM_HOT:
		break;
	case HPWMI_PROXIMITY_SENSOR:
		break;
	case HPWMI_BACKLIT_KB_BRIGHTNESS:
		break;
	case HPWMI_PEAKSHIFT_PERIOD:
		break;
	case HPWMI_BATTERY_CHARGE_PERIOD:
		break;
563
	default:
564
		pr_info("Unknown event_id - %d - 0x%x\n", event_id, event_data);
565 566
		break;
	}
567 568 569 570
}

static int __init hp_wmi_input_setup(void)
{
571
	acpi_status status;
572 573 574
	int err;

	hp_wmi_input_dev = input_allocate_device();
575 576
	if (!hp_wmi_input_dev)
		return -ENOMEM;
577 578 579 580 581

	hp_wmi_input_dev->name = "HP WMI hotkeys";
	hp_wmi_input_dev->phys = "wmi/input0";
	hp_wmi_input_dev->id.bustype = BUS_HOST;

582 583 584 585 586 587 588
	__set_bit(EV_SW, hp_wmi_input_dev->evbit);
	__set_bit(SW_DOCK, hp_wmi_input_dev->swbit);
	__set_bit(SW_TABLET_MODE, hp_wmi_input_dev->swbit);

	err = sparse_keymap_setup(hp_wmi_input_dev, hp_wmi_keymap, NULL);
	if (err)
		goto err_free_dev;
589 590 591 592 593 594 595

	/* Set initial hardware state */
	input_report_switch(hp_wmi_input_dev, SW_DOCK, hp_wmi_dock_state());
	input_report_switch(hp_wmi_input_dev, SW_TABLET_MODE,
			    hp_wmi_tablet_state());
	input_sync(hp_wmi_input_dev);

596 597 598 599
	status = wmi_install_notify_handler(HPWMI_EVENT_GUID, hp_wmi_notify, NULL);
	if (ACPI_FAILURE(status)) {
		err = -EIO;
		goto err_free_keymap;
600 601
	}

602 603 604 605
	err = input_register_device(hp_wmi_input_dev);
	if (err)
		goto err_uninstall_notifier;

606
	return 0;
607 608 609 610 611 612 613 614 615 616 617 618 619 620 621

 err_uninstall_notifier:
	wmi_remove_notify_handler(HPWMI_EVENT_GUID);
 err_free_keymap:
	sparse_keymap_free(hp_wmi_input_dev);
 err_free_dev:
	input_free_device(hp_wmi_input_dev);
	return err;
}

static void hp_wmi_input_destroy(void)
{
	wmi_remove_notify_handler(HPWMI_EVENT_GUID);
	sparse_keymap_free(hp_wmi_input_dev);
	input_unregister_device(hp_wmi_input_dev);
622 623 624 625 626 627 628 629
}

static void cleanup_sysfs(struct platform_device *device)
{
	device_remove_file(&device->dev, &dev_attr_display);
	device_remove_file(&device->dev, &dev_attr_hddtemp);
	device_remove_file(&device->dev, &dev_attr_als);
	device_remove_file(&device->dev, &dev_attr_dock);
630
	device_remove_file(&device->dev, &dev_attr_tablet);
631 632
}

633
static int hp_wmi_rfkill_setup(struct platform_device *device)
634 635
{
	int err;
M
Matthew Garrett 已提交
636
	int wireless = 0;
637

M
Matthew Garrett 已提交
638
	err = hp_wmi_perform_query(HPWMI_WIRELESS_QUERY, 0, &wireless,
639
				   sizeof(wireless), sizeof(wireless));
640 641
	if (err)
		return err;
642

643
	if (wireless & 0x1) {
J
Johannes Berg 已提交
644 645 646
		wifi_rfkill = rfkill_alloc("hp-wifi", &device->dev,
					   RFKILL_TYPE_WLAN,
					   &hp_wmi_rfkill_ops,
A
Alan Jenkins 已提交
647
					   (void *) HPWMI_WIFI);
648 649
		if (!wifi_rfkill)
			return -ENOMEM;
A
Alan Jenkins 已提交
650 651 652 653
		rfkill_init_sw_state(wifi_rfkill,
				     hp_wmi_get_sw_state(HPWMI_WIFI));
		rfkill_set_hw_state(wifi_rfkill,
				    hp_wmi_get_hw_state(HPWMI_WIFI));
654 655
		err = rfkill_register(wifi_rfkill);
		if (err)
J
Johannes Berg 已提交
656
			goto register_wifi_error;
657 658 659
	}

	if (wireless & 0x2) {
J
Johannes Berg 已提交
660 661 662
		bluetooth_rfkill = rfkill_alloc("hp-bluetooth", &device->dev,
						RFKILL_TYPE_BLUETOOTH,
						&hp_wmi_rfkill_ops,
A
Alan Jenkins 已提交
663
						(void *) HPWMI_BLUETOOTH);
664 665 666 667
		if (!bluetooth_rfkill) {
			err = -ENOMEM;
			goto register_wifi_error;
		}
A
Alan Jenkins 已提交
668 669 670 671
		rfkill_init_sw_state(bluetooth_rfkill,
				     hp_wmi_get_sw_state(HPWMI_BLUETOOTH));
		rfkill_set_hw_state(bluetooth_rfkill,
				    hp_wmi_get_hw_state(HPWMI_BLUETOOTH));
672
		err = rfkill_register(bluetooth_rfkill);
673
		if (err)
674
			goto register_bluetooth_error;
675 676 677
	}

	if (wireless & 0x4) {
J
Johannes Berg 已提交
678 679 680
		wwan_rfkill = rfkill_alloc("hp-wwan", &device->dev,
					   RFKILL_TYPE_WWAN,
					   &hp_wmi_rfkill_ops,
A
Alan Jenkins 已提交
681
					   (void *) HPWMI_WWAN);
682 683
		if (!wwan_rfkill) {
			err = -ENOMEM;
684
			goto register_gps_error;
685
		}
A
Alan Jenkins 已提交
686 687 688 689
		rfkill_init_sw_state(wwan_rfkill,
				     hp_wmi_get_sw_state(HPWMI_WWAN));
		rfkill_set_hw_state(wwan_rfkill,
				    hp_wmi_get_hw_state(HPWMI_WWAN));
690 691 692
		err = rfkill_register(wwan_rfkill);
		if (err)
			goto register_wwan_err;
693
	}
694

695 696 697 698 699 700 701 702 703 704 705
	if (wireless & 0x8) {
		gps_rfkill = rfkill_alloc("hp-gps", &device->dev,
						RFKILL_TYPE_GPS,
						&hp_wmi_rfkill_ops,
						(void *) HPWMI_GPS);
		if (!gps_rfkill) {
			err = -ENOMEM;
			goto register_bluetooth_error;
		}
		rfkill_init_sw_state(gps_rfkill,
				     hp_wmi_get_sw_state(HPWMI_GPS));
706
		rfkill_set_hw_state(gps_rfkill,
707 708 709 710 711 712
				    hp_wmi_get_hw_state(HPWMI_GPS));
		err = rfkill_register(gps_rfkill);
		if (err)
			goto register_gps_error;
	}

713
	return 0;
714
register_wwan_err:
J
Johannes Berg 已提交
715
	rfkill_destroy(wwan_rfkill);
716
	wwan_rfkill = NULL;
717 718 719 720 721
	if (gps_rfkill)
		rfkill_unregister(gps_rfkill);
register_gps_error:
	rfkill_destroy(gps_rfkill);
	gps_rfkill = NULL;
722 723
	if (bluetooth_rfkill)
		rfkill_unregister(bluetooth_rfkill);
724
register_bluetooth_error:
J
Johannes Berg 已提交
725
	rfkill_destroy(bluetooth_rfkill);
726
	bluetooth_rfkill = NULL;
727 728
	if (wifi_rfkill)
		rfkill_unregister(wifi_rfkill);
J
Johannes Berg 已提交
729 730
register_wifi_error:
	rfkill_destroy(wifi_rfkill);
731
	wifi_rfkill = NULL;
732 733 734
	return err;
}

735
static int hp_wmi_rfkill2_setup(struct platform_device *device)
736 737 738 739 740 741 742 743 744
{
	int err, i;
	struct bios_rfkill2_state state;
	err = hp_wmi_perform_query(HPWMI_WIRELESS2_QUERY, 0, &state,
				   0, sizeof(state));
	if (err)
		return err;

	if (state.count > HPWMI_MAX_RFKILL2_DEVICES) {
745
		pr_warn("unable to parse 0x1b query output\n");
746 747 748 749 750 751 752 753 754 755 756 757 758 759 760 761 762 763 764 765
		return -EINVAL;
	}

	for (i = 0; i < state.count; i++) {
		struct rfkill *rfkill;
		enum rfkill_type type;
		char *name;
		switch (state.device[i].radio_type) {
		case HPWMI_WIFI:
			type = RFKILL_TYPE_WLAN;
			name = "hp-wifi";
			break;
		case HPWMI_BLUETOOTH:
			type = RFKILL_TYPE_BLUETOOTH;
			name = "hp-bluetooth";
			break;
		case HPWMI_WWAN:
			type = RFKILL_TYPE_WWAN;
			name = "hp-wwan";
			break;
766 767 768 769
		case HPWMI_GPS:
			type = RFKILL_TYPE_GPS;
			name = "hp-gps";
			break;
770
		default:
771 772
			pr_warn("unknown device type 0x%x\n",
				state.device[i].radio_type);
773 774 775 776
			continue;
		}

		if (!state.device[i].vendor_id) {
777 778
			pr_warn("zero device %d while %d reported\n",
				i, state.count);
779 780 781 782 783 784 785 786 787 788 789 790 791 792 793 794 795 796 797 798
			continue;
		}

		rfkill = rfkill_alloc(name, &device->dev, type,
				      &hp_wmi_rfkill2_ops, (void *)(long)i);
		if (!rfkill) {
			err = -ENOMEM;
			goto fail;
		}

		rfkill2[rfkill2_count].id = state.device[i].rfkill_id;
		rfkill2[rfkill2_count].num = i;
		rfkill2[rfkill2_count].rfkill = rfkill;

		rfkill_init_sw_state(rfkill,
				     IS_SWBLOCKED(state.device[i].power));
		rfkill_set_hw_state(rfkill,
				    IS_HWBLOCKED(state.device[i].power));

		if (!(state.device[i].power & HPWMI_POWER_BIOS))
799
			pr_info("device %s blocked by BIOS\n", name);
800 801 802 803 804 805 806 807 808 809 810 811 812 813 814 815 816 817 818

		err = rfkill_register(rfkill);
		if (err) {
			rfkill_destroy(rfkill);
			goto fail;
		}

		rfkill2_count++;
	}

	return 0;
fail:
	for (; rfkill2_count > 0; rfkill2_count--) {
		rfkill_unregister(rfkill2[rfkill2_count - 1].rfkill);
		rfkill_destroy(rfkill2[rfkill2_count - 1].rfkill);
	}
	return err;
}

819
static int __init hp_wmi_bios_setup(struct platform_device *device)
820 821 822
{
	int err;

823 824 825 826
	/* clear detected rfkill devices */
	wifi_rfkill = NULL;
	bluetooth_rfkill = NULL;
	wwan_rfkill = NULL;
827
	gps_rfkill = NULL;
828
	rfkill2_count = 0;
829

830 831
	if (hp_wmi_rfkill_setup(device))
		hp_wmi_rfkill2_setup(device);
832 833 834 835 836 837 838 839 840 841 842 843 844 845 846 847 848 849

	err = device_create_file(&device->dev, &dev_attr_display);
	if (err)
		goto add_sysfs_error;
	err = device_create_file(&device->dev, &dev_attr_hddtemp);
	if (err)
		goto add_sysfs_error;
	err = device_create_file(&device->dev, &dev_attr_als);
	if (err)
		goto add_sysfs_error;
	err = device_create_file(&device->dev, &dev_attr_dock);
	if (err)
		goto add_sysfs_error;
	err = device_create_file(&device->dev, &dev_attr_tablet);
	if (err)
		goto add_sysfs_error;
	return 0;

850 851 852 853 854 855 856
add_sysfs_error:
	cleanup_sysfs(device);
	return err;
}

static int __exit hp_wmi_bios_remove(struct platform_device *device)
{
857
	int i;
858 859
	cleanup_sysfs(device);

860 861 862 863 864
	for (i = 0; i < rfkill2_count; i++) {
		rfkill_unregister(rfkill2[i].rfkill);
		rfkill_destroy(rfkill2[i].rfkill);
	}

J
Johannes Berg 已提交
865
	if (wifi_rfkill) {
866
		rfkill_unregister(wifi_rfkill);
J
Johannes Berg 已提交
867 868 869
		rfkill_destroy(wifi_rfkill);
	}
	if (bluetooth_rfkill) {
870
		rfkill_unregister(bluetooth_rfkill);
871
		rfkill_destroy(bluetooth_rfkill);
J
Johannes Berg 已提交
872 873
	}
	if (wwan_rfkill) {
874
		rfkill_unregister(wwan_rfkill);
J
Johannes Berg 已提交
875 876
		rfkill_destroy(wwan_rfkill);
	}
877 878 879 880
	if (gps_rfkill) {
		rfkill_unregister(gps_rfkill);
		rfkill_destroy(gps_rfkill);
	}
881 882 883 884

	return 0;
}

F
Frans Pop 已提交
885
static int hp_wmi_resume_handler(struct device *device)
886 887
{
	/*
888 889
	 * Hardware state may have changed while suspended, so trigger
	 * input events for the current state. As this is a switch,
890 891 892
	 * the input layer will only actually pass it on if the state
	 * changed.
	 */
893 894 895 896 897 898 899
	if (hp_wmi_input_dev) {
		input_report_switch(hp_wmi_input_dev, SW_DOCK,
				    hp_wmi_dock_state());
		input_report_switch(hp_wmi_input_dev, SW_TABLET_MODE,
				    hp_wmi_tablet_state());
		input_sync(hp_wmi_input_dev);
	}
900

901 902 903
	if (rfkill2_count)
		hp_wmi_rfkill2_refresh();

A
Alan Jenkins 已提交
904 905 906 907 908 909 910 911 912 913 914 915
	if (wifi_rfkill)
		rfkill_set_states(wifi_rfkill,
				  hp_wmi_get_sw_state(HPWMI_WIFI),
				  hp_wmi_get_hw_state(HPWMI_WIFI));
	if (bluetooth_rfkill)
		rfkill_set_states(bluetooth_rfkill,
				  hp_wmi_get_sw_state(HPWMI_BLUETOOTH),
				  hp_wmi_get_hw_state(HPWMI_BLUETOOTH));
	if (wwan_rfkill)
		rfkill_set_states(wwan_rfkill,
				  hp_wmi_get_sw_state(HPWMI_WWAN),
				  hp_wmi_get_hw_state(HPWMI_WWAN));
916 917 918 919
	if (gps_rfkill)
		rfkill_set_states(gps_rfkill,
				  hp_wmi_get_sw_state(HPWMI_GPS),
				  hp_wmi_get_hw_state(HPWMI_GPS));
A
Alan Jenkins 已提交
920

921 922 923
	return 0;
}

924 925 926 927 928 929 930 931 932 933 934 935 936 937
static const struct dev_pm_ops hp_wmi_pm_ops = {
	.resume  = hp_wmi_resume_handler,
	.restore  = hp_wmi_resume_handler,
};

static struct platform_driver hp_wmi_driver = {
	.driver = {
		.name = "hp-wmi",
		.owner = THIS_MODULE,
		.pm = &hp_wmi_pm_ops,
	},
	.remove = __exit_p(hp_wmi_bios_remove),
};

938 939 940
static int __init hp_wmi_init(void)
{
	int err;
941 942
	int event_capable = wmi_has_guid(HPWMI_EVENT_GUID);
	int bios_capable = wmi_has_guid(HPWMI_BIOS_GUID);
943

944 945 946
	if (!bios_capable && !event_capable)
		return -ENODEV;

947
	if (event_capable) {
948
		err = hp_wmi_input_setup();
949
		if (err)
950
			return err;
951 952
	}

953
	if (bios_capable) {
954 955 956 957 958
		hp_wmi_platform_dev =
			platform_device_register_simple("hp-wmi", -1, NULL, 0);
		if (IS_ERR(hp_wmi_platform_dev)) {
			err = PTR_ERR(hp_wmi_platform_dev);
			goto err_destroy_input;
959
		}
960 961

		err = platform_driver_probe(&hp_wmi_driver, hp_wmi_bios_setup);
962
		if (err)
963
			goto err_unregister_device;
964 965 966
	}

	return 0;
967

968 969 970
err_unregister_device:
	platform_device_unregister(hp_wmi_platform_dev);
err_destroy_input:
971 972
	if (event_capable)
		hp_wmi_input_destroy();
973 974

	return err;
975
}
976
module_init(hp_wmi_init);
977 978 979

static void __exit hp_wmi_exit(void)
{
980 981 982
	if (wmi_has_guid(HPWMI_EVENT_GUID))
		hp_wmi_input_destroy();

983
	if (hp_wmi_platform_dev) {
984
		platform_device_unregister(hp_wmi_platform_dev);
985 986 987 988
		platform_driver_unregister(&hp_wmi_driver);
	}
}
module_exit(hp_wmi_exit);