i8042-x86ia64io.h 20.7 KB
Newer Older
L
Linus Torvalds 已提交
1 2 3 4 5 6 7 8 9
#ifndef _I8042_X86IA64IO_H
#define _I8042_X86IA64IO_H

/*
 * This program is free software; you can redistribute it and/or modify it
 * under the terms of the GNU General Public License version 2 as published by
 * the Free Software Foundation.
 */

10 11 12 13
#ifdef CONFIG_X86
#include <asm/x86_init.h>
#endif

L
Linus Torvalds 已提交
14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69
/*
 * Names.
 */

#define I8042_KBD_PHYS_DESC "isa0060/serio0"
#define I8042_AUX_PHYS_DESC "isa0060/serio1"
#define I8042_MUX_PHYS_DESC "isa0060/serio%d"

/*
 * IRQs.
 */

#if defined(__ia64__)
# define I8042_MAP_IRQ(x)	isa_irq_to_vector((x))
#else
# define I8042_MAP_IRQ(x)	(x)
#endif

#define I8042_KBD_IRQ	i8042_kbd_irq
#define I8042_AUX_IRQ	i8042_aux_irq

static int i8042_kbd_irq;
static int i8042_aux_irq;

/*
 * Register numbers.
 */

#define I8042_COMMAND_REG	i8042_command_reg
#define I8042_STATUS_REG	i8042_command_reg
#define I8042_DATA_REG		i8042_data_reg

static int i8042_command_reg = 0x64;
static int i8042_data_reg = 0x60;


static inline int i8042_read_data(void)
{
	return inb(I8042_DATA_REG);
}

static inline int i8042_read_status(void)
{
	return inb(I8042_STATUS_REG);
}

static inline void i8042_write_data(int val)
{
	outb(val, I8042_DATA_REG);
}

static inline void i8042_write_command(int val)
{
	outb(val, I8042_COMMAND_REG);
}

70
#ifdef CONFIG_X86
L
Linus Torvalds 已提交
71 72 73

#include <linux/dmi.h>

74
static const struct dmi_system_id __initconst i8042_dmi_noloop_table[] = {
75
	{
76 77 78 79
		/*
		 * Arima-Rioworks HDAMB -
		 * AUX LOOP command does not raise AUX IRQ
		 */
80 81 82 83 84 85
		.matches = {
			DMI_MATCH(DMI_BOARD_VENDOR, "RIOWORKS"),
			DMI_MATCH(DMI_BOARD_NAME, "HDAMB"),
			DMI_MATCH(DMI_BOARD_VERSION, "Rev E"),
		},
	},
86
	{
87
		/* ASUS G1S */
88 89 90 91 92 93
		.matches = {
			DMI_MATCH(DMI_BOARD_VENDOR, "ASUSTeK Computer Inc."),
			DMI_MATCH(DMI_BOARD_NAME, "G1S"),
			DMI_MATCH(DMI_BOARD_VERSION, "1.0"),
		},
	},
94
	{
95
		/* ASUS P65UP5 - AUX LOOP command does not raise AUX IRQ */
96 97 98 99 100 101
		.matches = {
			DMI_MATCH(DMI_BOARD_VENDOR, "ASUSTeK Computer INC."),
			DMI_MATCH(DMI_BOARD_NAME, "P/I-P65UP5"),
			DMI_MATCH(DMI_BOARD_VERSION, "REV 2.X"),
		},
	},
L
Linus Torvalds 已提交
102 103 104 105 106 107 108 109 110 111 112 113 114 115
	{
		.matches = {
			DMI_MATCH(DMI_SYS_VENDOR, "Compaq"),
			DMI_MATCH(DMI_PRODUCT_NAME , "ProLiant"),
			DMI_MATCH(DMI_PRODUCT_VERSION, "8500"),
		},
	},
	{
		.matches = {
			DMI_MATCH(DMI_SYS_VENDOR, "Compaq"),
			DMI_MATCH(DMI_PRODUCT_NAME , "ProLiant"),
			DMI_MATCH(DMI_PRODUCT_VERSION, "DL760"),
		},
	},
116
	{
117
		/* OQO Model 01 */
118 119 120 121 122 123
		.matches = {
			DMI_MATCH(DMI_SYS_VENDOR, "OQO"),
			DMI_MATCH(DMI_PRODUCT_NAME, "ZEPTO"),
			DMI_MATCH(DMI_PRODUCT_VERSION, "00"),
		},
	},
124
	{
125
		/* ULI EV4873 - AUX LOOP does not work properly */
126 127 128 129 130 131
		.matches = {
			DMI_MATCH(DMI_SYS_VENDOR, "ULI"),
			DMI_MATCH(DMI_PRODUCT_NAME, "EV4873"),
			DMI_MATCH(DMI_PRODUCT_VERSION, "5a"),
		},
	},
132
	{
133
		/* Microsoft Virtual Machine */
134 135 136 137 138 139
		.matches = {
			DMI_MATCH(DMI_SYS_VENDOR, "Microsoft Corporation"),
			DMI_MATCH(DMI_PRODUCT_NAME, "Virtual Machine"),
			DMI_MATCH(DMI_PRODUCT_VERSION, "VS2005R2"),
		},
	},
140
	{
141
		/* Medion MAM 2070 */
142 143 144 145 146 147
		.matches = {
			DMI_MATCH(DMI_SYS_VENDOR, "Notebook"),
			DMI_MATCH(DMI_PRODUCT_NAME, "MAM 2070"),
			DMI_MATCH(DMI_PRODUCT_VERSION, "5a"),
		},
	},
148
	{
149
		/* Blue FB5601 */
150 151 152 153 154 155
		.matches = {
			DMI_MATCH(DMI_SYS_VENDOR, "blue"),
			DMI_MATCH(DMI_PRODUCT_NAME, "FB5601"),
			DMI_MATCH(DMI_PRODUCT_VERSION, "M606"),
		},
	},
156
	{
157
		/* Gigabyte M912 */
158 159 160 161 162 163
		.matches = {
			DMI_MATCH(DMI_SYS_VENDOR, "GIGABYTE"),
			DMI_MATCH(DMI_PRODUCT_NAME, "M912"),
			DMI_MATCH(DMI_PRODUCT_VERSION, "01"),
		},
	},
164 165 166 167 168 169 170 171
	{
		/* Gigabyte M1022M netbook */
		.matches = {
			DMI_MATCH(DMI_BOARD_VENDOR, "Gigabyte Technology Co.,Ltd."),
			DMI_MATCH(DMI_BOARD_NAME, "M1022E"),
			DMI_MATCH(DMI_BOARD_VERSION, "1.02"),
		},
	},
172 173 174 175 176 177 178
	{
		/* Gigabyte Spring Peak - defines wrong chassis type */
		.matches = {
			DMI_MATCH(DMI_SYS_VENDOR, "GIGABYTE"),
			DMI_MATCH(DMI_PRODUCT_NAME, "Spring Peak"),
		},
	},
179 180 181 182 183 184 185
	{
		.matches = {
			DMI_MATCH(DMI_SYS_VENDOR, "Hewlett-Packard"),
			DMI_MATCH(DMI_PRODUCT_NAME, "HP Pavilion dv9700"),
			DMI_MATCH(DMI_PRODUCT_VERSION, "Rev 1"),
		},
	},
L
Linus Torvalds 已提交
186 187 188 189
	{ }
};

/*
190
 * Some Fujitsu notebooks are having trouble with touchpads if
L
Linus Torvalds 已提交
191 192
 * active multiplexing mode is activated. Luckily they don't have
 * external PS/2 ports so we can safely disable it.
193 194
 * ... apparently some Toshibas don't like MUX mode either and
 * die horrible death on reboot.
L
Linus Torvalds 已提交
195
 */
196
static const struct dmi_system_id __initconst i8042_dmi_nomux_table[] = {
L
Linus Torvalds 已提交
197
	{
198
		/* Fujitsu Lifebook P7010/P7010D */
L
Linus Torvalds 已提交
199 200 201 202 203
		.matches = {
			DMI_MATCH(DMI_SYS_VENDOR, "FUJITSU"),
			DMI_MATCH(DMI_PRODUCT_NAME, "P7010"),
		},
	},
204
	{
205
		/* Fujitsu Lifebook P7010 */
206 207 208 209 210
		.matches = {
			DMI_MATCH(DMI_SYS_VENDOR, "FUJITSU SIEMENS"),
			DMI_MATCH(DMI_PRODUCT_NAME, "0000000000"),
		},
	},
L
Linus Torvalds 已提交
211
	{
212
		/* Fujitsu Lifebook P5020D */
L
Linus Torvalds 已提交
213 214 215 216 217 218
		.matches = {
			DMI_MATCH(DMI_SYS_VENDOR, "FUJITSU"),
			DMI_MATCH(DMI_PRODUCT_NAME, "LifeBook P Series"),
		},
	},
	{
219
		/* Fujitsu Lifebook S2000 */
L
Linus Torvalds 已提交
220 221 222 223 224
		.matches = {
			DMI_MATCH(DMI_SYS_VENDOR, "FUJITSU"),
			DMI_MATCH(DMI_PRODUCT_NAME, "LifeBook S Series"),
		},
	},
225
	{
226
		/* Fujitsu Lifebook S6230 */
227 228 229 230 231
		.matches = {
			DMI_MATCH(DMI_SYS_VENDOR, "FUJITSU"),
			DMI_MATCH(DMI_PRODUCT_NAME, "LifeBook S6230"),
		},
	},
L
Linus Torvalds 已提交
232
	{
233
		/* Fujitsu T70H */
L
Linus Torvalds 已提交
234 235 236 237 238
		.matches = {
			DMI_MATCH(DMI_SYS_VENDOR, "FUJITSU"),
			DMI_MATCH(DMI_PRODUCT_NAME, "FMVLT70H"),
		},
	},
239
	{
240
		/* Fujitsu-Siemens Lifebook T3010 */
241 242 243 244 245
		.matches = {
			DMI_MATCH(DMI_SYS_VENDOR, "FUJITSU SIEMENS"),
			DMI_MATCH(DMI_PRODUCT_NAME, "LIFEBOOK T3010"),
		},
	},
246
	{
247
		/* Fujitsu-Siemens Lifebook E4010 */
248 249 250 251 252
		.matches = {
			DMI_MATCH(DMI_SYS_VENDOR, "FUJITSU SIEMENS"),
			DMI_MATCH(DMI_PRODUCT_NAME, "LIFEBOOK E4010"),
		},
	},
253
	{
254
		/* Fujitsu-Siemens Amilo Pro 2010 */
255 256 257 258 259
		.matches = {
			DMI_MATCH(DMI_SYS_VENDOR, "FUJITSU SIEMENS"),
			DMI_MATCH(DMI_PRODUCT_NAME, "AMILO Pro V2010"),
		},
	},
260
	{
261
		/* Fujitsu-Siemens Amilo Pro 2030 */
262 263 264 265 266
		.matches = {
			DMI_MATCH(DMI_SYS_VENDOR, "FUJITSU SIEMENS"),
			DMI_MATCH(DMI_PRODUCT_NAME, "AMILO PRO V2030"),
		},
	},
267 268 269 270 271
	{
		/*
		 * No data is coming from the touchscreen unless KBC
		 * is in legacy mode.
		 */
272
		/* Panasonic CF-29 */
273 274 275 276 277
		.matches = {
			DMI_MATCH(DMI_SYS_VENDOR, "Matsushita"),
			DMI_MATCH(DMI_PRODUCT_NAME, "CF-29"),
		},
	},
278 279
	{
		/*
280 281
		 * HP Pavilion DV4017EA -
		 * errors on MUX ports are reported without raising AUXDATA
282 283 284 285 286 287 288
		 * causing "spurious NAK" messages.
		 */
		.matches = {
			DMI_MATCH(DMI_SYS_VENDOR, "Hewlett-Packard"),
			DMI_MATCH(DMI_PRODUCT_NAME, "Pavilion dv4000 (EA032EA#ABF)"),
		},
	},
289 290
	{
		/*
291 292
		 * HP Pavilion ZT1000 -
		 * like DV4017EA does not raise AUXERR for errors on MUX ports.
293 294 295 296 297 298 299
		 */
		.matches = {
			DMI_MATCH(DMI_SYS_VENDOR, "Hewlett-Packard"),
			DMI_MATCH(DMI_PRODUCT_NAME, "HP Pavilion Notebook PC"),
			DMI_MATCH(DMI_PRODUCT_VERSION, "HP Pavilion Notebook ZT1000"),
		},
	},
300 301
	{
		/*
302 303
		 * HP Pavilion DV4270ca -
		 * like DV4017EA does not raise AUXERR for errors on MUX ports.
304 305 306 307 308 309
		 */
		.matches = {
			DMI_MATCH(DMI_SYS_VENDOR, "Hewlett-Packard"),
			DMI_MATCH(DMI_PRODUCT_NAME, "Pavilion dv4000 (EH476UA#ABL)"),
		},
	},
310 311 312 313 314 315
	{
		.matches = {
			DMI_MATCH(DMI_SYS_VENDOR, "TOSHIBA"),
			DMI_MATCH(DMI_PRODUCT_NAME, "Satellite P10"),
		},
	},
316 317 318 319 320 321
	{
		.matches = {
			DMI_MATCH(DMI_SYS_VENDOR, "TOSHIBA"),
			DMI_MATCH(DMI_PRODUCT_NAME, "EQUIUM A110"),
		},
	},
322 323 324 325 326 327
	{
		.matches = {
			DMI_MATCH(DMI_SYS_VENDOR, "ALIENWARE"),
			DMI_MATCH(DMI_PRODUCT_NAME, "Sentia"),
		},
	},
328
	{
329
		/* Sharp Actius MM20 */
330 331 332 333 334
		.matches = {
			DMI_MATCH(DMI_SYS_VENDOR, "SHARP"),
			DMI_MATCH(DMI_PRODUCT_NAME, "PC-MM20 Series"),
		},
	},
335
	{
336
		/* Sony Vaio FS-115b */
337 338 339 340 341
		.matches = {
			DMI_MATCH(DMI_SYS_VENDOR, "Sony Corporation"),
			DMI_MATCH(DMI_PRODUCT_NAME, "VGN-FS115B"),
		},
	},
342 343
	{
		/*
344 345
		 * Sony Vaio FZ-240E -
		 * reset and GET ID commands issued via KBD port are
346 347 348 349 350 351 352
		 * sometimes being delivered to AUX3.
		 */
		.matches = {
			DMI_MATCH(DMI_SYS_VENDOR, "Sony Corporation"),
			DMI_MATCH(DMI_PRODUCT_NAME, "VGN-FZ240E"),
		},
	},
353 354 355 356 357 358 359 360 361 362 363
	{
		/*
		 * Most (all?) VAIOs do not have external PS/2 ports nor
		 * they implement active multiplexing properly, and
		 * MUX discovery usually messes up keyboard/touchpad.
		 */
		.matches = {
			DMI_MATCH(DMI_SYS_VENDOR, "Sony Corporation"),
			DMI_MATCH(DMI_BOARD_NAME, "VAIO"),
		},
	},
364
	{
365
		/* Amoi M636/A737 */
366 367 368 369 370
		.matches = {
			DMI_MATCH(DMI_SYS_VENDOR, "Amoi Electronics CO.,LTD."),
			DMI_MATCH(DMI_PRODUCT_NAME, "M636/A737 platform"),
		},
	},
371
	{
372
		/* Lenovo 3000 n100 */
373 374
		.matches = {
			DMI_MATCH(DMI_SYS_VENDOR, "LENOVO"),
375
			DMI_MATCH(DMI_PRODUCT_NAME, "076804U"),
376 377
		},
	},
378 379 380 381 382 383
	{
		.matches = {
			DMI_MATCH(DMI_SYS_VENDOR, "Acer"),
			DMI_MATCH(DMI_PRODUCT_NAME, "Aspire 1360"),
		},
	},
384
	{
385
		/* Gericom Bellagio */
386 387 388 389 390
		.matches = {
			DMI_MATCH(DMI_SYS_VENDOR, "Gericom"),
			DMI_MATCH(DMI_PRODUCT_NAME, "N34AS6"),
		},
	},
391
	{
392
		/* IBM 2656 */
393 394 395 396 397
		.matches = {
			DMI_MATCH(DMI_SYS_VENDOR, "IBM"),
			DMI_MATCH(DMI_PRODUCT_NAME, "2656"),
		},
	},
398
	{
399
		/* Dell XPS M1530 */
400 401 402 403 404
		.matches = {
			DMI_MATCH(DMI_SYS_VENDOR, "Dell Inc."),
			DMI_MATCH(DMI_PRODUCT_NAME, "XPS M1530"),
		},
	},
405
	{
406
		/* Compal HEL80I */
407 408 409 410 411
		.matches = {
			DMI_MATCH(DMI_SYS_VENDOR, "COMPAL"),
			DMI_MATCH(DMI_PRODUCT_NAME, "HEL80I"),
		},
	},
412
	{
413
		/* Dell Vostro 1510 */
414 415 416 417 418
		.matches = {
			DMI_MATCH(DMI_SYS_VENDOR, "Dell Inc."),
			DMI_MATCH(DMI_PRODUCT_NAME, "Vostro1510"),
		},
	},
419
	{
420
		/* Acer Aspire 5536 */
421 422 423 424 425 426
		.matches = {
			DMI_MATCH(DMI_SYS_VENDOR, "Acer"),
			DMI_MATCH(DMI_PRODUCT_NAME, "Aspire 5536"),
			DMI_MATCH(DMI_PRODUCT_VERSION, "0100"),
		},
	},
L
Linus Torvalds 已提交
427 428 429
	{ }
};

430
static const struct dmi_system_id __initconst i8042_dmi_reset_table[] = {
431
	{
432
		/* MSI Wind U-100 */
433 434 435 436 437 438
		.matches = {
			DMI_MATCH(DMI_BOARD_NAME, "U-100"),
			DMI_MATCH(DMI_BOARD_VENDOR, "MICRO-STAR INTERNATIONAL CO., LTD"),
		},
	},
	{
439
		/* LG Electronics X110 */
440 441 442 443 444
		.matches = {
			DMI_MATCH(DMI_BOARD_NAME, "X110"),
			DMI_MATCH(DMI_BOARD_VENDOR, "LG Electronics Inc."),
		},
	},
445
	{
446
		/* Acer Aspire One 150 */
447 448 449 450 451 452
		.matches = {
			DMI_MATCH(DMI_SYS_VENDOR, "Acer"),
			DMI_MATCH(DMI_PRODUCT_NAME, "AOA150"),
		},
	},
	{
453
		/* Advent 4211 */
454 455 456 457 458 459
		.matches = {
			DMI_MATCH(DMI_SYS_VENDOR, "DIXONSXP"),
			DMI_MATCH(DMI_PRODUCT_NAME, "Advent 4211"),
		},
	},
	{
460
		/* Medion Akoya Mini E1210 */
461 462 463 464 465
		.matches = {
			DMI_MATCH(DMI_SYS_VENDOR, "MEDION"),
			DMI_MATCH(DMI_PRODUCT_NAME, "E1210"),
		},
	},
466 467 468 469 470 471 472
	{
		/* Medion Akoya E1222 */
		.matches = {
			DMI_MATCH(DMI_SYS_VENDOR, "MEDION"),
			DMI_MATCH(DMI_PRODUCT_NAME, "E122X"),
		},
	},
473
	{
474
		/* Mivvy M310 */
475 476 477 478 479
		.matches = {
			DMI_MATCH(DMI_SYS_VENDOR, "VIOOO"),
			DMI_MATCH(DMI_PRODUCT_NAME, "N10"),
		},
	},
480
	{
481
		/* Dell Vostro 1320 */
482 483 484 485 486 487
		.matches = {
			DMI_MATCH(DMI_SYS_VENDOR, "Dell Inc."),
			DMI_MATCH(DMI_PRODUCT_NAME, "Vostro 1320"),
		},
	},
	{
488
		/* Dell Vostro 1520 */
489 490 491 492 493 494
		.matches = {
			DMI_MATCH(DMI_SYS_VENDOR, "Dell Inc."),
			DMI_MATCH(DMI_PRODUCT_NAME, "Vostro 1520"),
		},
	},
	{
495
		/* Dell Vostro 1720 */
496 497 498 499 500
		.matches = {
			DMI_MATCH(DMI_SYS_VENDOR, "Dell Inc."),
			DMI_MATCH(DMI_PRODUCT_NAME, "Vostro 1720"),
		},
	},
501 502 503
	{ }
};

504
#ifdef CONFIG_PNP
505
static const struct dmi_system_id __initconst i8042_dmi_nopnp_table[] = {
506
	{
507
		/* Intel MBO Desktop D845PESV */
508 509 510 511 512
		.matches = {
			DMI_MATCH(DMI_BOARD_NAME, "D845PESV"),
			DMI_MATCH(DMI_BOARD_VENDOR, "Intel Corporation"),
		},
	},
513
	{
514
		/* MSI Wind U-100 */
515 516 517 518 519
		.matches = {
			DMI_MATCH(DMI_BOARD_NAME, "U-100"),
			DMI_MATCH(DMI_BOARD_VENDOR, "MICRO-STAR INTERNATIONAL CO., LTD"),
		},
	},
520 521
	{ }
};
522

523
static const struct dmi_system_id __initconst i8042_dmi_laptop_table[] = {
524 525 526 527 528 529 530 531 532 533 534 535 536 537 538 539 540 541 542 543 544 545
	{
		.matches = {
			DMI_MATCH(DMI_CHASSIS_TYPE, "8"), /* Portable */
		},
	},
	{
		.matches = {
			DMI_MATCH(DMI_CHASSIS_TYPE, "9"), /* Laptop */
		},
	},
	{
		.matches = {
			DMI_MATCH(DMI_CHASSIS_TYPE, "10"), /* Notebook */
		},
	},
	{
		.matches = {
			DMI_MATCH(DMI_CHASSIS_TYPE, "14"), /* Sub-Notebook */
		},
	},
	{ }
};
L
Linus Torvalds 已提交
546 547
#endif

548 549 550 551 552 553
/*
 * Some Wistron based laptops need us to explicitly enable the 'Dritek
 * keyboard extension' to make their extra keys start generating scancodes.
 * Originally, this was just confined to older laptops, but a few Acer laptops
 * have turned up in 2007 that also need this again.
 */
554
static const struct dmi_system_id __initconst i8042_dmi_dritek_table[] = {
555 556 557 558 559 560 561
	{
		/* Acer Aspire 5610 */
		.matches = {
			DMI_MATCH(DMI_SYS_VENDOR, "Acer"),
			DMI_MATCH(DMI_PRODUCT_NAME, "Aspire 5610"),
		},
	},
562
	{
563
		/* Acer Aspire 5630 */
564 565 566 567 568 569
		.matches = {
			DMI_MATCH(DMI_SYS_VENDOR, "Acer"),
			DMI_MATCH(DMI_PRODUCT_NAME, "Aspire 5630"),
		},
	},
	{
570
		/* Acer Aspire 5650 */
571 572 573 574 575 576
		.matches = {
			DMI_MATCH(DMI_SYS_VENDOR, "Acer"),
			DMI_MATCH(DMI_PRODUCT_NAME, "Aspire 5650"),
		},
	},
	{
577
		/* Acer Aspire 5680 */
578 579 580 581 582
		.matches = {
			DMI_MATCH(DMI_SYS_VENDOR, "Acer"),
			DMI_MATCH(DMI_PRODUCT_NAME, "Aspire 5680"),
		},
	},
583
	{
584
		/* Acer Aspire 5720 */
585 586 587 588 589
		.matches = {
			DMI_MATCH(DMI_SYS_VENDOR, "Acer"),
			DMI_MATCH(DMI_PRODUCT_NAME, "Aspire 5720"),
		},
	},
590
	{
591
		/* Acer Aspire 9110 */
592 593 594 595 596
		.matches = {
			DMI_MATCH(DMI_SYS_VENDOR, "Acer"),
			DMI_MATCH(DMI_PRODUCT_NAME, "Aspire 9110"),
		},
	},
597
	{
598
		/* Acer TravelMate 660 */
599 600 601 602 603
		.matches = {
			DMI_MATCH(DMI_SYS_VENDOR, "Acer"),
			DMI_MATCH(DMI_PRODUCT_NAME, "TravelMate 660"),
		},
	},
604
	{
605
		/* Acer TravelMate 2490 */
606 607 608 609 610
		.matches = {
			DMI_MATCH(DMI_SYS_VENDOR, "Acer"),
			DMI_MATCH(DMI_PRODUCT_NAME, "TravelMate 2490"),
		},
	},
611
	{
612
		/* Acer TravelMate 4280 */
613 614 615 616 617
		.matches = {
			DMI_MATCH(DMI_SYS_VENDOR, "Acer"),
			DMI_MATCH(DMI_PRODUCT_NAME, "TravelMate 4280"),
		},
	},
618 619 620 621 622
	{ }
};

#endif /* CONFIG_X86 */

L
Linus Torvalds 已提交
623 624 625
#ifdef CONFIG_PNP
#include <linux/pnp.h>

626
static bool i8042_pnp_kbd_registered;
627
static unsigned int i8042_pnp_kbd_devices;
628
static bool i8042_pnp_aux_registered;
629
static unsigned int i8042_pnp_aux_devices;
L
Linus Torvalds 已提交
630 631 632 633 634 635 636 637 638 639 640 641 642 643 644 645 646 647 648 649

static int i8042_pnp_command_reg;
static int i8042_pnp_data_reg;
static int i8042_pnp_kbd_irq;
static int i8042_pnp_aux_irq;

static char i8042_pnp_kbd_name[32];
static char i8042_pnp_aux_name[32];

static int i8042_pnp_kbd_probe(struct pnp_dev *dev, const struct pnp_device_id *did)
{
	if (pnp_port_valid(dev, 0) && pnp_port_len(dev, 0) == 1)
		i8042_pnp_data_reg = pnp_port_start(dev,0);

	if (pnp_port_valid(dev, 1) && pnp_port_len(dev, 1) == 1)
		i8042_pnp_command_reg = pnp_port_start(dev, 1);

	if (pnp_irq_valid(dev,0))
		i8042_pnp_kbd_irq = pnp_irq(dev, 0);

650
	strlcpy(i8042_pnp_kbd_name, did->id, sizeof(i8042_pnp_kbd_name));
L
Linus Torvalds 已提交
651
	if (strlen(pnp_dev_name(dev))) {
652 653
		strlcat(i8042_pnp_kbd_name, ":", sizeof(i8042_pnp_kbd_name));
		strlcat(i8042_pnp_kbd_name, pnp_dev_name(dev), sizeof(i8042_pnp_kbd_name));
L
Linus Torvalds 已提交
654 655
	}

656 657 658
	/* Keyboard ports are always supposed to be wakeup-enabled */
	device_set_wakeup_enable(&dev->dev, true);

659
	i8042_pnp_kbd_devices++;
L
Linus Torvalds 已提交
660 661 662 663 664 665 666 667 668 669 670 671 672 673
	return 0;
}

static int i8042_pnp_aux_probe(struct pnp_dev *dev, const struct pnp_device_id *did)
{
	if (pnp_port_valid(dev, 0) && pnp_port_len(dev, 0) == 1)
		i8042_pnp_data_reg = pnp_port_start(dev,0);

	if (pnp_port_valid(dev, 1) && pnp_port_len(dev, 1) == 1)
		i8042_pnp_command_reg = pnp_port_start(dev, 1);

	if (pnp_irq_valid(dev, 0))
		i8042_pnp_aux_irq = pnp_irq(dev, 0);

674
	strlcpy(i8042_pnp_aux_name, did->id, sizeof(i8042_pnp_aux_name));
L
Linus Torvalds 已提交
675
	if (strlen(pnp_dev_name(dev))) {
676 677
		strlcat(i8042_pnp_aux_name, ":", sizeof(i8042_pnp_aux_name));
		strlcat(i8042_pnp_aux_name, pnp_dev_name(dev), sizeof(i8042_pnp_aux_name));
L
Linus Torvalds 已提交
678 679
	}

680
	i8042_pnp_aux_devices++;
L
Linus Torvalds 已提交
681 682 683 684
	return 0;
}

static struct pnp_device_id pnp_kbd_devids[] = {
685 686 687
	{ .id = "PNP0300", .driver_data = 0 },
	{ .id = "PNP0301", .driver_data = 0 },
	{ .id = "PNP0302", .driver_data = 0 },
L
Linus Torvalds 已提交
688
	{ .id = "PNP0303", .driver_data = 0 },
689 690 691 692 693
	{ .id = "PNP0304", .driver_data = 0 },
	{ .id = "PNP0305", .driver_data = 0 },
	{ .id = "PNP0306", .driver_data = 0 },
	{ .id = "PNP0309", .driver_data = 0 },
	{ .id = "PNP030a", .driver_data = 0 },
L
Linus Torvalds 已提交
694
	{ .id = "PNP030b", .driver_data = 0 },
695 696 697 698 699
	{ .id = "PNP0320", .driver_data = 0 },
	{ .id = "PNP0343", .driver_data = 0 },
	{ .id = "PNP0344", .driver_data = 0 },
	{ .id = "PNP0345", .driver_data = 0 },
	{ .id = "CPQA0D7", .driver_data = 0 },
L
Linus Torvalds 已提交
700 701 702 703 704 705 706 707 708 709
	{ .id = "", },
};

static struct pnp_driver i8042_pnp_kbd_driver = {
	.name           = "i8042 kbd",
	.id_table       = pnp_kbd_devids,
	.probe          = i8042_pnp_kbd_probe,
};

static struct pnp_device_id pnp_aux_devids[] = {
710
	{ .id = "AUI0200", .driver_data = 0 },
711 712
	{ .id = "FJC6000", .driver_data = 0 },
	{ .id = "FJC6001", .driver_data = 0 },
L
Linus Torvalds 已提交
713 714 715 716 717 718 719 720 721 722 723 724 725 726 727 728 729 730 731
	{ .id = "PNP0f03", .driver_data = 0 },
	{ .id = "PNP0f0b", .driver_data = 0 },
	{ .id = "PNP0f0e", .driver_data = 0 },
	{ .id = "PNP0f12", .driver_data = 0 },
	{ .id = "PNP0f13", .driver_data = 0 },
	{ .id = "PNP0f19", .driver_data = 0 },
	{ .id = "PNP0f1c", .driver_data = 0 },
	{ .id = "SYN0801", .driver_data = 0 },
	{ .id = "", },
};

static struct pnp_driver i8042_pnp_aux_driver = {
	.name           = "i8042 aux",
	.id_table       = pnp_aux_devids,
	.probe          = i8042_pnp_aux_probe,
};

static void i8042_pnp_exit(void)
{
732
	if (i8042_pnp_kbd_registered) {
733
		i8042_pnp_kbd_registered = false;
L
Linus Torvalds 已提交
734
		pnp_unregister_driver(&i8042_pnp_kbd_driver);
735
	}
L
Linus Torvalds 已提交
736

737
	if (i8042_pnp_aux_registered) {
738
		i8042_pnp_aux_registered = false;
L
Linus Torvalds 已提交
739
		pnp_unregister_driver(&i8042_pnp_aux_driver);
740
	}
L
Linus Torvalds 已提交
741 742
}

743
static int __init i8042_pnp_init(void)
L
Linus Torvalds 已提交
744
{
745
	char kbd_irq_str[4] = { 0 }, aux_irq_str[4] = { 0 };
746
	bool pnp_data_busted = false;
747
	int err;
L
Linus Torvalds 已提交
748

749 750
#ifdef CONFIG_X86
	if (dmi_check_system(i8042_dmi_nopnp_table))
751
		i8042_nopnp = true;
752 753
#endif

L
Linus Torvalds 已提交
754
	if (i8042_nopnp) {
755
		printk(KERN_INFO "i8042: PNP detection disabled\n");
L
Linus Torvalds 已提交
756 757 758
		return 0;
	}

759 760
	err = pnp_register_driver(&i8042_pnp_kbd_driver);
	if (!err)
761
		i8042_pnp_kbd_registered = true;
762

763 764
	err = pnp_register_driver(&i8042_pnp_aux_driver);
	if (!err)
765
		i8042_pnp_aux_registered = true;
L
Linus Torvalds 已提交
766

767
	if (!i8042_pnp_kbd_devices && !i8042_pnp_aux_devices) {
L
Linus Torvalds 已提交
768 769 770 771
		i8042_pnp_exit();
#if defined(__ia64__)
		return -ENODEV;
#else
772
		printk(KERN_INFO "PNP: No PS/2 controller found. Probing ports directly.\n");
L
Linus Torvalds 已提交
773 774 775 776
		return 0;
#endif
	}

777
	if (i8042_pnp_kbd_devices)
778 779
		snprintf(kbd_irq_str, sizeof(kbd_irq_str),
			"%d", i8042_pnp_kbd_irq);
780
	if (i8042_pnp_aux_devices)
781 782 783 784
		snprintf(aux_irq_str, sizeof(aux_irq_str),
			"%d", i8042_pnp_aux_irq);

	printk(KERN_INFO "PNP: PS/2 Controller [%s%s%s] at %#x,%#x irq %s%s%s\n",
785
		i8042_pnp_kbd_name, (i8042_pnp_kbd_devices && i8042_pnp_aux_devices) ? "," : "",
786 787
		i8042_pnp_aux_name,
		i8042_pnp_data_reg, i8042_pnp_command_reg,
788
		kbd_irq_str, (i8042_pnp_kbd_devices && i8042_pnp_aux_devices) ? "," : "",
789 790 791
		aux_irq_str);

#if defined(__ia64__)
792
	if (!i8042_pnp_kbd_devices)
793
		i8042_nokbd = true;
794
	if (!i8042_pnp_aux_devices)
795
		i8042_noaux = true;
796 797
#endif

L
Linus Torvalds 已提交
798
	if (((i8042_pnp_data_reg & ~0xf) == (i8042_data_reg & ~0xf) &&
799 800 801 802 803
	      i8042_pnp_data_reg != i8042_data_reg) ||
	    !i8042_pnp_data_reg) {
		printk(KERN_WARNING
			"PNP: PS/2 controller has invalid data port %#x; "
			"using default %#x\n",
L
Linus Torvalds 已提交
804 805
			i8042_pnp_data_reg, i8042_data_reg);
		i8042_pnp_data_reg = i8042_data_reg;
806
		pnp_data_busted = true;
L
Linus Torvalds 已提交
807 808 809
	}

	if (((i8042_pnp_command_reg & ~0xf) == (i8042_command_reg & ~0xf) &&
810 811 812 813 814
	      i8042_pnp_command_reg != i8042_command_reg) ||
	    !i8042_pnp_command_reg) {
		printk(KERN_WARNING
			"PNP: PS/2 controller has invalid command port %#x; "
			"using default %#x\n",
L
Linus Torvalds 已提交
815 816
			i8042_pnp_command_reg, i8042_command_reg);
		i8042_pnp_command_reg = i8042_command_reg;
817
		pnp_data_busted = true;
L
Linus Torvalds 已提交
818 819
	}

820
	if (!i8042_nokbd && !i8042_pnp_kbd_irq) {
821 822 823
		printk(KERN_WARNING
			"PNP: PS/2 controller doesn't have KBD irq; "
			"using default %d\n", i8042_kbd_irq);
L
Linus Torvalds 已提交
824
		i8042_pnp_kbd_irq = i8042_kbd_irq;
825
		pnp_data_busted = true;
L
Linus Torvalds 已提交
826 827
	}

828
	if (!i8042_noaux && !i8042_pnp_aux_irq) {
829 830 831 832 833
		if (!pnp_data_busted && i8042_pnp_kbd_irq) {
			printk(KERN_WARNING
				"PNP: PS/2 appears to have AUX port disabled, "
				"if this is incorrect please boot with "
				"i8042.nopnp\n");
834
			i8042_noaux = true;
835 836 837 838 839 840
		} else {
			printk(KERN_WARNING
				"PNP: PS/2 controller doesn't have AUX irq; "
				"using default %d\n", i8042_aux_irq);
			i8042_pnp_aux_irq = i8042_aux_irq;
		}
L
Linus Torvalds 已提交
841 842 843 844 845 846 847
	}

	i8042_data_reg = i8042_pnp_data_reg;
	i8042_command_reg = i8042_pnp_command_reg;
	i8042_kbd_irq = i8042_pnp_kbd_irq;
	i8042_aux_irq = i8042_pnp_aux_irq;

848 849 850 851 852
#ifdef CONFIG_X86
	i8042_bypass_aux_irq_test = !pnp_data_busted &&
				    dmi_check_system(i8042_dmi_laptop_table);
#endif

L
Linus Torvalds 已提交
853 854 855
	return 0;
}

856 857 858
#else
static inline int i8042_pnp_init(void) { return 0; }
static inline void i8042_pnp_exit(void) { }
L
Linus Torvalds 已提交
859 860
#endif

861
static int __init i8042_platform_init(void)
L
Linus Torvalds 已提交
862
{
863 864
	int retval;

865 866 867 868 869 870
#ifdef CONFIG_X86
	/* Just return if pre-detection shows no i8042 controller exist */
	if (!x86_platform.i8042_detect())
		return -ENODEV;
#endif

L
Linus Torvalds 已提交
871 872 873 874 875
/*
 * On ix86 platforms touching the i8042 data register region can do really
 * bad things. Because of this the region is always reserved on ix86 boxes.
 *
 *	if (!request_region(I8042_DATA_REG, 16, "i8042"))
876
 *		return -EBUSY;
L
Linus Torvalds 已提交
877 878 879 880 881
 */

	i8042_kbd_irq = I8042_MAP_IRQ(1);
	i8042_aux_irq = I8042_MAP_IRQ(12);

882 883 884
	retval = i8042_pnp_init();
	if (retval)
		return retval;
L
Linus Torvalds 已提交
885 886

#if defined(__ia64__)
887
        i8042_reset = true;
L
Linus Torvalds 已提交
888 889
#endif

890
#ifdef CONFIG_X86
891
	if (dmi_check_system(i8042_dmi_reset_table))
892
		i8042_reset = true;
893

L
Linus Torvalds 已提交
894
	if (dmi_check_system(i8042_dmi_noloop_table))
895
		i8042_noloop = true;
L
Linus Torvalds 已提交
896 897

	if (dmi_check_system(i8042_dmi_nomux_table))
898
		i8042_nomux = true;
L
Linus Torvalds 已提交
899

900
	if (dmi_check_system(i8042_dmi_dritek_table))
901
		i8042_dritek = true;
902 903
#endif /* CONFIG_X86 */

904
	return retval;
L
Linus Torvalds 已提交
905 906 907 908 909 910 911 912
}

static inline void i8042_platform_exit(void)
{
	i8042_pnp_exit();
}

#endif /* _I8042_X86IA64IO_H */