hwgpe.c 13.0 KB
Newer Older
L
Linus Torvalds 已提交
1 2 3 4 5 6 7
/******************************************************************************
 *
 * Module Name: hwgpe - Low level GPE enable/disable/clear functions
 *
 *****************************************************************************/

/*
8
 * Copyright (C) 2000 - 2014, Intel Corp.
L
Linus Torvalds 已提交
9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44
 * All rights reserved.
 *
 * Redistribution and use in source and binary forms, with or without
 * modification, are permitted provided that the following conditions
 * are met:
 * 1. Redistributions of source code must retain the above copyright
 *    notice, this list of conditions, and the following disclaimer,
 *    without modification.
 * 2. Redistributions in binary form must reproduce at minimum a disclaimer
 *    substantially similar to the "NO WARRANTY" disclaimer below
 *    ("Disclaimer") and any redistribution must be conditioned upon
 *    including a substantially similar Disclaimer requirement for further
 *    binary redistribution.
 * 3. Neither the names of the above-listed copyright holders nor the names
 *    of any contributors may be used to endorse or promote products derived
 *    from this software without specific prior written permission.
 *
 * Alternatively, this software may be distributed under the terms of the
 * GNU General Public License ("GPL") version 2 as published by the Free
 * Software Foundation.
 *
 * NO WARRANTY
 * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
 * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
 * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTIBILITY AND FITNESS FOR
 * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
 * HOLDERS OR CONTRIBUTORS BE LIABLE FOR SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
 * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
 * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
 * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT,
 * STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING
 * IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
 * POSSIBILITY OF SUCH DAMAGES.
 */

#include <acpi/acpi.h>
L
Len Brown 已提交
45 46
#include "accommon.h"
#include "acevents.h"
L
Linus Torvalds 已提交
47 48

#define _COMPONENT          ACPI_HARDWARE
L
Len Brown 已提交
49
ACPI_MODULE_NAME("hwgpe")
50
#if (!ACPI_REDUCED_HARDWARE)	/* Entire module */
R
Robert Moore 已提交
51 52
/* Local prototypes */
static acpi_status
L
Len Brown 已提交
53
acpi_hw_enable_wakeup_gpe_block(struct acpi_gpe_xrupt_info *gpe_xrupt_info,
54 55
				struct acpi_gpe_block_info *gpe_block,
				void *context);
L
Linus Torvalds 已提交
56

57 58
/******************************************************************************
 *
59
 * FUNCTION:	acpi_hw_get_gpe_register_bit
60 61 62
 *
 * PARAMETERS:	gpe_event_info	    - Info block for the GPE
 *
63
 * RETURN:	Register mask with a one in the GPE bit position
64
 *
65 66
 * DESCRIPTION: Compute the register mask for this GPE. One bit is set in the
 *              correct position for the input GPE.
67 68 69
 *
 ******************************************************************************/

70
u32 acpi_hw_get_gpe_register_bit(struct acpi_gpe_event_info *gpe_event_info)
71
{
72 73 74 75

	return ((u32)1 <<
		(gpe_event_info->gpe_number -
		 gpe_event_info->register_info->base_gpe_number));
76 77
}

B
Bob Moore 已提交
78 79
/******************************************************************************
 *
80
 * FUNCTION:	acpi_hw_low_set_gpe
B
Bob Moore 已提交
81 82
 *
 * PARAMETERS:	gpe_event_info	    - Info block for the GPE to be disabled
83
 *		action		    - Enable or disable
B
Bob Moore 已提交
84 85 86
 *
 * RETURN:	Status
 *
87
 * DESCRIPTION: Enable or disable a single GPE in the parent enable register.
B
Bob Moore 已提交
88 89 90
 *
 ******************************************************************************/

91
acpi_status
92
acpi_hw_low_set_gpe(struct acpi_gpe_event_info *gpe_event_info, u32 action)
B
Bob Moore 已提交
93 94 95 96
{
	struct acpi_gpe_register_info *gpe_register_info;
	acpi_status status;
	u32 enable_mask;
97
	u32 register_bit;
B
Bob Moore 已提交
98

99 100
	ACPI_FUNCTION_ENTRY();

B
Bob Moore 已提交
101 102 103 104 105 106 107 108 109
	/* Get the info block for the entire GPE register */

	gpe_register_info = gpe_event_info->register_info;
	if (!gpe_register_info) {
		return (AE_NOT_EXIST);
	}

	/* Get current value of the enable register that contains this GPE */

110
	status = acpi_hw_read(&enable_mask, &gpe_register_info->enable_address);
B
Bob Moore 已提交
111 112 113 114
	if (ACPI_FAILURE(status)) {
		return (status);
	}

115
	/* Set or clear just the bit that corresponds to this GPE */
B
Bob Moore 已提交
116

117
	register_bit = acpi_hw_get_gpe_register_bit(gpe_event_info);
118
	switch (action) {
119
	case ACPI_GPE_CONDITIONAL_ENABLE:
120 121 122 123

		/* Only enable if the enable_for_run bit is set */

		if (!(register_bit & gpe_register_info->enable_for_run)) {
124
			return (AE_BAD_PARAMETER);
125 126 127
		}

		/*lint -fallthrough */
128

129
	case ACPI_GPE_ENABLE:
130

131 132 133 134
		ACPI_SET_BIT(enable_mask, register_bit);
		break;

	case ACPI_GPE_DISABLE:
135

136 137 138 139
		ACPI_CLEAR_BIT(enable_mask, register_bit);
		break;

	default:
140

141
		ACPI_ERROR((AE_INFO, "Invalid GPE Action, %u", action));
142 143
		return (AE_BAD_PARAMETER);
	}
B
Bob Moore 已提交
144 145 146

	/* Write the updated enable mask */

147
	status = acpi_hw_write(enable_mask, &gpe_register_info->enable_address);
B
Bob Moore 已提交
148 149 150
	return (status);
}

L
Linus Torvalds 已提交
151 152 153 154 155 156 157 158 159 160 161 162
/******************************************************************************
 *
 * FUNCTION:    acpi_hw_clear_gpe
 *
 * PARAMETERS:  gpe_event_info      - Info block for the GPE to be cleared
 *
 * RETURN:      Status
 *
 * DESCRIPTION: Clear the status bit for a single GPE.
 *
 ******************************************************************************/

L
Len Brown 已提交
163
acpi_status acpi_hw_clear_gpe(struct acpi_gpe_event_info * gpe_event_info)
L
Linus Torvalds 已提交
164
{
165
	struct acpi_gpe_register_info *gpe_register_info;
L
Len Brown 已提交
166
	acpi_status status;
167
	u32 register_bit;
L
Linus Torvalds 已提交
168

L
Len Brown 已提交
169
	ACPI_FUNCTION_ENTRY();
L
Linus Torvalds 已提交
170

171 172 173 174 175 176 177
	/* Get the info block for the entire GPE register */

	gpe_register_info = gpe_event_info->register_info;
	if (!gpe_register_info) {
		return (AE_NOT_EXIST);
	}

L
Linus Torvalds 已提交
178 179 180 181
	/*
	 * Write a one to the appropriate bit in the status register to
	 * clear this GPE.
	 */
182
	register_bit = acpi_hw_get_gpe_register_bit(gpe_event_info);
183

184
	status = acpi_hw_write(register_bit,
185
			       &gpe_register_info->status_address);
L
Linus Torvalds 已提交
186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201

	return (status);
}

/******************************************************************************
 *
 * FUNCTION:    acpi_hw_get_gpe_status
 *
 * PARAMETERS:  gpe_event_info      - Info block for the GPE to queried
 *              event_status        - Where the GPE status is returned
 *
 * RETURN:      Status
 *
 * DESCRIPTION: Return the status of a single GPE.
 *
 ******************************************************************************/
R
Robert Moore 已提交
202

L
Linus Torvalds 已提交
203
acpi_status
L
Len Brown 已提交
204
acpi_hw_get_gpe_status(struct acpi_gpe_event_info * gpe_event_info,
205
		       acpi_event_status *event_status)
L
Linus Torvalds 已提交
206
{
L
Len Brown 已提交
207
	u32 in_byte;
208
	u32 register_bit;
L
Len Brown 已提交
209 210
	struct acpi_gpe_register_info *gpe_register_info;
	acpi_event_status local_event_status = 0;
211
	acpi_status status;
L
Linus Torvalds 已提交
212

L
Len Brown 已提交
213
	ACPI_FUNCTION_ENTRY();
L
Linus Torvalds 已提交
214 215 216 217 218 219 220 221 222 223 224

	if (!event_status) {
		return (AE_BAD_PARAMETER);
	}

	/* Get the info block for the entire GPE register */

	gpe_register_info = gpe_event_info->register_info;

	/* Get the register bitmask for this GPE */

225
	register_bit = acpi_hw_get_gpe_register_bit(gpe_event_info);
L
Linus Torvalds 已提交
226 227 228 229 230 231 232 233 234 235 236 237 238 239 240

	/* GPE currently enabled? (enabled for runtime?) */

	if (register_bit & gpe_register_info->enable_for_run) {
		local_event_status |= ACPI_EVENT_FLAG_ENABLED;
	}

	/* GPE enabled for wake? */

	if (register_bit & gpe_register_info->enable_for_wake) {
		local_event_status |= ACPI_EVENT_FLAG_WAKE_ENABLED;
	}

	/* GPE currently active (status bit == 1)? */

241
	status = acpi_hw_read(&in_byte, &gpe_register_info->status_address);
L
Len Brown 已提交
242
	if (ACPI_FAILURE(status)) {
243
		return (status);
L
Linus Torvalds 已提交
244 245 246 247 248 249 250 251 252
	}

	if (register_bit & in_byte) {
		local_event_status |= ACPI_EVENT_FLAG_SET;
	}

	/* Set return value */

	(*event_status) = local_event_status;
253
	return (AE_OK);
L
Linus Torvalds 已提交
254 255 256 257 258 259 260 261 262 263 264
}

/******************************************************************************
 *
 * FUNCTION:    acpi_hw_disable_gpe_block
 *
 * PARAMETERS:  gpe_xrupt_info      - GPE Interrupt info
 *              gpe_block           - Gpe Block info
 *
 * RETURN:      Status
 *
R
Robert Moore 已提交
265
 * DESCRIPTION: Disable all GPEs within a single GPE block
L
Linus Torvalds 已提交
266 267 268 269
 *
 ******************************************************************************/

acpi_status
270 271
acpi_hw_disable_gpe_block(struct acpi_gpe_xrupt_info *gpe_xrupt_info,
			  struct acpi_gpe_block_info *gpe_block, void *context)
L
Linus Torvalds 已提交
272
{
L
Len Brown 已提交
273 274
	u32 i;
	acpi_status status;
L
Linus Torvalds 已提交
275 276 277 278

	/* Examine each GPE Register within the block */

	for (i = 0; i < gpe_block->register_count; i++) {
B
Bob Moore 已提交
279

L
Linus Torvalds 已提交
280 281
		/* Disable all GPEs in this register */

282
		status =
283 284
		    acpi_hw_write(0x00,
				  &gpe_block->register_info[i].enable_address);
L
Len Brown 已提交
285
		if (ACPI_FAILURE(status)) {
L
Linus Torvalds 已提交
286 287 288 289 290 291 292 293 294 295 296 297 298 299 300 301
			return (status);
		}
	}

	return (AE_OK);
}

/******************************************************************************
 *
 * FUNCTION:    acpi_hw_clear_gpe_block
 *
 * PARAMETERS:  gpe_xrupt_info      - GPE Interrupt info
 *              gpe_block           - Gpe Block info
 *
 * RETURN:      Status
 *
R
Robert Moore 已提交
302
 * DESCRIPTION: Clear status bits for all GPEs within a single GPE block
L
Linus Torvalds 已提交
303 304 305 306
 *
 ******************************************************************************/

acpi_status
307 308
acpi_hw_clear_gpe_block(struct acpi_gpe_xrupt_info *gpe_xrupt_info,
			struct acpi_gpe_block_info *gpe_block, void *context)
L
Linus Torvalds 已提交
309
{
L
Len Brown 已提交
310 311
	u32 i;
	acpi_status status;
L
Linus Torvalds 已提交
312 313 314 315

	/* Examine each GPE Register within the block */

	for (i = 0; i < gpe_block->register_count; i++) {
B
Bob Moore 已提交
316

L
Linus Torvalds 已提交
317 318
		/* Clear status on all GPEs in this register */

319
		status =
320 321
		    acpi_hw_write(0xFF,
				  &gpe_block->register_info[i].status_address);
L
Len Brown 已提交
322
		if (ACPI_FAILURE(status)) {
L
Linus Torvalds 已提交
323 324 325 326 327 328 329 330 331 332 333 334 335 336 337 338
			return (status);
		}
	}

	return (AE_OK);
}

/******************************************************************************
 *
 * FUNCTION:    acpi_hw_enable_runtime_gpe_block
 *
 * PARAMETERS:  gpe_xrupt_info      - GPE Interrupt info
 *              gpe_block           - Gpe Block info
 *
 * RETURN:      Status
 *
R
Robert Moore 已提交
339 340
 * DESCRIPTION: Enable all "runtime" GPEs within a single GPE block. Includes
 *              combination wake/run GPEs.
L
Linus Torvalds 已提交
341 342 343 344
 *
 ******************************************************************************/

acpi_status
345
acpi_hw_enable_runtime_gpe_block(struct acpi_gpe_xrupt_info *gpe_xrupt_info,
L
Lv Zheng 已提交
346 347
				 struct acpi_gpe_block_info * gpe_block,
				 void *context)
L
Linus Torvalds 已提交
348
{
L
Len Brown 已提交
349 350
	u32 i;
	acpi_status status;
L
Linus Torvalds 已提交
351 352 353 354 355 356 357 358 359 360 361 362

	/* NOTE: assumes that all GPEs are currently disabled */

	/* Examine each GPE Register within the block */

	for (i = 0; i < gpe_block->register_count; i++) {
		if (!gpe_block->register_info[i].enable_for_run) {
			continue;
		}

		/* Enable all "runtime" GPEs in this register */

363 364 365
		status =
		    acpi_hw_write(gpe_block->register_info[i].enable_for_run,
				  &gpe_block->register_info[i].enable_address);
L
Len Brown 已提交
366
		if (ACPI_FAILURE(status)) {
L
Linus Torvalds 已提交
367 368 369 370 371 372 373 374 375 376 377 378 379 380 381 382
			return (status);
		}
	}

	return (AE_OK);
}

/******************************************************************************
 *
 * FUNCTION:    acpi_hw_enable_wakeup_gpe_block
 *
 * PARAMETERS:  gpe_xrupt_info      - GPE Interrupt info
 *              gpe_block           - Gpe Block info
 *
 * RETURN:      Status
 *
R
Robert Moore 已提交
383 384
 * DESCRIPTION: Enable all "wake" GPEs within a single GPE block. Includes
 *              combination wake/run GPEs.
L
Linus Torvalds 已提交
385 386 387
 *
 ******************************************************************************/

R
Robert Moore 已提交
388
static acpi_status
L
Len Brown 已提交
389
acpi_hw_enable_wakeup_gpe_block(struct acpi_gpe_xrupt_info *gpe_xrupt_info,
390 391
				struct acpi_gpe_block_info *gpe_block,
				void *context)
L
Linus Torvalds 已提交
392
{
L
Len Brown 已提交
393 394
	u32 i;
	acpi_status status;
L
Linus Torvalds 已提交
395 396 397 398 399

	/* Examine each GPE Register within the block */

	for (i = 0; i < gpe_block->register_count; i++) {

400 401 402 403
		/*
		 * Enable all "wake" GPEs in this register and disable the
		 * remaining ones.
		 */
L
Linus Torvalds 已提交
404

405 406 407
		status =
		    acpi_hw_write(gpe_block->register_info[i].enable_for_wake,
				  &gpe_block->register_info[i].enable_address);
L
Len Brown 已提交
408
		if (ACPI_FAILURE(status)) {
L
Linus Torvalds 已提交
409 410 411 412 413 414 415 416 417 418 419
			return (status);
		}
	}

	return (AE_OK);
}

/******************************************************************************
 *
 * FUNCTION:    acpi_hw_disable_all_gpes
 *
420
 * PARAMETERS:  None
L
Linus Torvalds 已提交
421 422 423
 *
 * RETURN:      Status
 *
R
Robert Moore 已提交
424
 * DESCRIPTION: Disable and clear all GPEs in all GPE blocks
L
Linus Torvalds 已提交
425 426 427
 *
 ******************************************************************************/

L
Len Brown 已提交
428
acpi_status acpi_hw_disable_all_gpes(void)
L
Linus Torvalds 已提交
429
{
L
Len Brown 已提交
430
	acpi_status status;
L
Linus Torvalds 已提交
431

B
Bob Moore 已提交
432
	ACPI_FUNCTION_TRACE(hw_disable_all_gpes);
L
Linus Torvalds 已提交
433

434 435
	status = acpi_ev_walk_gpe_list(acpi_hw_disable_gpe_block, NULL);
	status = acpi_ev_walk_gpe_list(acpi_hw_clear_gpe_block, NULL);
L
Len Brown 已提交
436
	return_ACPI_STATUS(status);
L
Linus Torvalds 已提交
437 438 439 440 441 442
}

/******************************************************************************
 *
 * FUNCTION:    acpi_hw_enable_all_runtime_gpes
 *
443
 * PARAMETERS:  None
L
Linus Torvalds 已提交
444 445 446
 *
 * RETURN:      Status
 *
R
Robert Moore 已提交
447
 * DESCRIPTION: Enable all "runtime" GPEs, in all GPE blocks
L
Linus Torvalds 已提交
448 449 450
 *
 ******************************************************************************/

L
Len Brown 已提交
451
acpi_status acpi_hw_enable_all_runtime_gpes(void)
L
Linus Torvalds 已提交
452
{
L
Len Brown 已提交
453
	acpi_status status;
L
Linus Torvalds 已提交
454

B
Bob Moore 已提交
455
	ACPI_FUNCTION_TRACE(hw_enable_all_runtime_gpes);
L
Linus Torvalds 已提交
456

457
	status = acpi_ev_walk_gpe_list(acpi_hw_enable_runtime_gpe_block, NULL);
L
Len Brown 已提交
458
	return_ACPI_STATUS(status);
L
Linus Torvalds 已提交
459 460 461 462 463 464
}

/******************************************************************************
 *
 * FUNCTION:    acpi_hw_enable_all_wakeup_gpes
 *
465
 * PARAMETERS:  None
L
Linus Torvalds 已提交
466 467 468
 *
 * RETURN:      Status
 *
R
Robert Moore 已提交
469
 * DESCRIPTION: Enable all "wakeup" GPEs, in all GPE blocks
L
Linus Torvalds 已提交
470 471 472
 *
 ******************************************************************************/

L
Len Brown 已提交
473
acpi_status acpi_hw_enable_all_wakeup_gpes(void)
L
Linus Torvalds 已提交
474
{
L
Len Brown 已提交
475
	acpi_status status;
L
Linus Torvalds 已提交
476

B
Bob Moore 已提交
477
	ACPI_FUNCTION_TRACE(hw_enable_all_wakeup_gpes);
L
Linus Torvalds 已提交
478

479
	status = acpi_ev_walk_gpe_list(acpi_hw_enable_wakeup_gpe_block, NULL);
L
Len Brown 已提交
480
	return_ACPI_STATUS(status);
L
Linus Torvalds 已提交
481
}
482 483

#endif				/* !ACPI_REDUCED_HARDWARE */