blob: 1c409e82c461fc1076d020c6b5df4247f35b8f49 [file] [log] [blame]
Bob Moore70958572012-02-14 18:47:42 +08001/******************************************************************************
2 *
3 * Name: hwesleep.c - ACPI Hardware Sleep/Wake Support functions for the
4 * extended FADT-V5 sleep registers.
5 *
6 *****************************************************************************/
7
8/*
9 * Copyright (C) 2000 - 2012, Intel Corp.
10 * All rights reserved.
11 *
12 * Redistribution and use in source and binary forms, with or without
13 * modification, are permitted provided that the following conditions
14 * are met:
15 * 1. Redistributions of source code must retain the above copyright
16 * notice, this list of conditions, and the following disclaimer,
17 * without modification.
18 * 2. Redistributions in binary form must reproduce at minimum a disclaimer
19 * substantially similar to the "NO WARRANTY" disclaimer below
20 * ("Disclaimer") and any redistribution must be conditioned upon
21 * including a substantially similar Disclaimer requirement for further
22 * binary redistribution.
23 * 3. Neither the names of the above-listed copyright holders nor the names
24 * of any contributors may be used to endorse or promote products derived
25 * from this software without specific prior written permission.
26 *
27 * Alternatively, this software may be distributed under the terms of the
28 * GNU General Public License ("GPL") version 2 as published by the Free
29 * Software Foundation.
30 *
31 * NO WARRANTY
32 * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
33 * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
34 * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTIBILITY AND FITNESS FOR
35 * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
36 * HOLDERS OR CONTRIBUTORS BE LIABLE FOR SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
37 * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
38 * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
39 * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT,
40 * STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING
41 * IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
42 * POSSIBILITY OF SUCH DAMAGES.
43 */
44
45#include <acpi/acpi.h>
46#include "accommon.h"
47#include <linux/module.h>
48
49#define _COMPONENT ACPI_HARDWARE
50ACPI_MODULE_NAME("hwesleep")
51
52static unsigned int gts, bfs;
53module_param(gts, uint, 0644);
54module_param(bfs, uint, 0644);
55MODULE_PARM_DESC(gts, "Enable evaluation of _GTS on suspend.");
56MODULE_PARM_DESC(bfs, "Enable evaluation of _BFS on resume".);
57
58/*******************************************************************************
59 *
60 * FUNCTION: acpi_hw_execute_sleep_method
61 *
Bob Moore4efeeec2012-03-21 09:42:45 +080062 * PARAMETERS: method_pathname - Pathname of method to execute
Bob Moore70958572012-02-14 18:47:42 +080063 * integer_argument - Argument to pass to the method
64 *
65 * RETURN: None
66 *
67 * DESCRIPTION: Execute a sleep/wake related method with one integer argument
68 * and no return value.
69 *
70 ******************************************************************************/
Bob Moore4efeeec2012-03-21 09:42:45 +080071void acpi_hw_execute_sleep_method(char *method_pathname, u32 integer_argument)
Bob Moore70958572012-02-14 18:47:42 +080072{
73 struct acpi_object_list arg_list;
74 union acpi_object arg;
75 acpi_status status;
76
77 ACPI_FUNCTION_TRACE(hw_execute_sleep_method);
78
Bob Moore4efeeec2012-03-21 09:42:45 +080079 if (!ACPI_STRCMP(METHOD_PATHNAME__GTS, method_pathname) && !gts)
Bob Moore70958572012-02-14 18:47:42 +080080 return_VOID;
81
Bob Moore4efeeec2012-03-21 09:42:45 +080082 if (!ACPI_STRCMP(METHOD_PATHNAME__BFS, method_pathname) && !bfs)
Bob Moore70958572012-02-14 18:47:42 +080083 return_VOID;
84
85 /* One argument, integer_argument; No return value expected */
86
87 arg_list.count = 1;
88 arg_list.pointer = &arg;
89 arg.type = ACPI_TYPE_INTEGER;
90 arg.integer.value = (u64)integer_argument;
91
Bob Moore4efeeec2012-03-21 09:42:45 +080092 status = acpi_evaluate_object(NULL, method_pathname, &arg_list, NULL);
Bob Moore70958572012-02-14 18:47:42 +080093 if (ACPI_FAILURE(status) && status != AE_NOT_FOUND) {
94 ACPI_EXCEPTION((AE_INFO, status, "While executing method %s",
Bob Moore4efeeec2012-03-21 09:42:45 +080095 method_pathname));
Bob Moore70958572012-02-14 18:47:42 +080096 }
97
98 return_VOID;
99}
100
101/*******************************************************************************
102 *
103 * FUNCTION: acpi_hw_extended_sleep
104 *
105 * PARAMETERS: sleep_state - Which sleep state to enter
Lin Ming8a73b172012-03-21 10:01:49 +0800106 * Flags - ACPI_EXECUTE_GTS to run optional method
Bob Moore70958572012-02-14 18:47:42 +0800107 *
108 * RETURN: Status
109 *
110 * DESCRIPTION: Enter a system sleep state via the extended FADT sleep
111 * registers (V5 FADT).
112 * THIS FUNCTION MUST BE CALLED WITH INTERRUPTS DISABLED
113 *
114 ******************************************************************************/
115
Lin Ming8a73b172012-03-21 10:01:49 +0800116acpi_status acpi_hw_extended_sleep(u8 sleep_state, u8 flags)
Bob Moore70958572012-02-14 18:47:42 +0800117{
118 acpi_status status;
119 u8 sleep_type_value;
120 u64 sleep_status;
121
122 ACPI_FUNCTION_TRACE(hw_extended_sleep);
123
124 /* Extended sleep registers must be valid */
125
126 if (!acpi_gbl_FADT.sleep_control.address ||
127 !acpi_gbl_FADT.sleep_status.address) {
128 return_ACPI_STATUS(AE_NOT_EXIST);
129 }
130
131 /* Clear wake status (WAK_STS) */
132
133 status = acpi_write(ACPI_X_WAKE_STATUS, &acpi_gbl_FADT.sleep_status);
134 if (ACPI_FAILURE(status)) {
135 return_ACPI_STATUS(status);
136 }
137
138 acpi_gbl_system_awake_and_running = FALSE;
139
Lin Ming8a73b172012-03-21 10:01:49 +0800140 /* Optionally execute _GTS (Going To Sleep) */
Bob Moore70958572012-02-14 18:47:42 +0800141
Lin Ming8a73b172012-03-21 10:01:49 +0800142 if (flags & ACPI_EXECUTE_GTS) {
143 acpi_hw_execute_sleep_method(METHOD_PATHNAME__GTS, sleep_state);
144 }
Bob Moore70958572012-02-14 18:47:42 +0800145
146 /* Flush caches, as per ACPI specification */
147
148 ACPI_FLUSH_CPU_CACHE();
149
150 /*
151 * Set the SLP_TYP and SLP_EN bits.
152 *
153 * Note: We only use the first value returned by the \_Sx method
154 * (acpi_gbl_sleep_type_a) - As per ACPI specification.
155 */
156 ACPI_DEBUG_PRINT((ACPI_DB_INIT,
157 "Entering sleep state [S%u]\n", sleep_state));
158
159 sleep_type_value =
160 ((acpi_gbl_sleep_type_a << ACPI_X_SLEEP_TYPE_POSITION) &
161 ACPI_X_SLEEP_TYPE_MASK);
162
163 status = acpi_write((sleep_type_value | ACPI_X_SLEEP_ENABLE),
164 &acpi_gbl_FADT.sleep_control);
165 if (ACPI_FAILURE(status)) {
166 return_ACPI_STATUS(status);
167 }
168
169 /* Wait for transition back to Working State */
170
171 do {
172 status = acpi_read(&sleep_status, &acpi_gbl_FADT.sleep_status);
173 if (ACPI_FAILURE(status)) {
174 return_ACPI_STATUS(status);
175 }
176
177 } while (!(((u8)sleep_status) & ACPI_X_WAKE_STATUS));
178
179 return_ACPI_STATUS(AE_OK);
180}
181
182/*******************************************************************************
183 *
184 * FUNCTION: acpi_hw_extended_wake_prep
185 *
186 * PARAMETERS: sleep_state - Which sleep state we just exited
Lin Ming8a73b172012-03-21 10:01:49 +0800187 * Flags - ACPI_EXECUTE_BFS to run optional method
Bob Moore70958572012-02-14 18:47:42 +0800188 *
189 * RETURN: Status
190 *
191 * DESCRIPTION: Perform first part of OS-independent ACPI cleanup after
192 * a sleep. Called with interrupts ENABLED.
193 *
194 ******************************************************************************/
195
Lin Ming8a73b172012-03-21 10:01:49 +0800196acpi_status acpi_hw_extended_wake_prep(u8 sleep_state, u8 flags)
Bob Moore70958572012-02-14 18:47:42 +0800197{
198 acpi_status status;
199 u8 sleep_type_value;
200
201 ACPI_FUNCTION_TRACE(hw_extended_wake_prep);
202
203 status = acpi_get_sleep_type_data(ACPI_STATE_S0,
204 &acpi_gbl_sleep_type_a,
205 &acpi_gbl_sleep_type_b);
206 if (ACPI_SUCCESS(status)) {
207 sleep_type_value =
208 ((acpi_gbl_sleep_type_a << ACPI_X_SLEEP_TYPE_POSITION) &
209 ACPI_X_SLEEP_TYPE_MASK);
210
211 (void)acpi_write((sleep_type_value | ACPI_X_SLEEP_ENABLE),
212 &acpi_gbl_FADT.sleep_control);
213 }
214
Lin Ming8a73b172012-03-21 10:01:49 +0800215 /* Optionally execute _BFS (Back From Sleep) */
216
217 if (flags & ACPI_EXECUTE_BFS) {
218 acpi_hw_execute_sleep_method(METHOD_PATHNAME__BFS, sleep_state);
219 }
Bob Moore70958572012-02-14 18:47:42 +0800220 return_ACPI_STATUS(AE_OK);
221}
222
223/*******************************************************************************
224 *
225 * FUNCTION: acpi_hw_extended_wake
226 *
227 * PARAMETERS: sleep_state - Which sleep state we just exited
Lin Ming8a73b172012-03-21 10:01:49 +0800228 * Flags - Reserved, set to zero
Bob Moore70958572012-02-14 18:47:42 +0800229 *
230 * RETURN: Status
231 *
232 * DESCRIPTION: Perform OS-independent ACPI cleanup after a sleep
233 * Called with interrupts ENABLED.
234 *
235 ******************************************************************************/
236
Lin Ming8a73b172012-03-21 10:01:49 +0800237acpi_status acpi_hw_extended_wake(u8 sleep_state, u8 flags)
Bob Moore70958572012-02-14 18:47:42 +0800238{
239 ACPI_FUNCTION_TRACE(hw_extended_wake);
240
241 /* Ensure enter_sleep_state_prep -> enter_sleep_state ordering */
242
243 acpi_gbl_sleep_type_a = ACPI_SLEEP_TYPE_INVALID;
244
245 /* Execute the wake methods */
246
Bob Moore4efeeec2012-03-21 09:42:45 +0800247 acpi_hw_execute_sleep_method(METHOD_PATHNAME__SST, ACPI_SST_WAKING);
248 acpi_hw_execute_sleep_method(METHOD_PATHNAME__WAK, sleep_state);
Bob Moore70958572012-02-14 18:47:42 +0800249
250 /*
251 * Some BIOS code assumes that WAK_STS will be cleared on resume
252 * and use it to determine whether the system is rebooting or
253 * resuming. Clear WAK_STS for compatibility.
254 */
255 (void)acpi_write(ACPI_X_WAKE_STATUS, &acpi_gbl_FADT.sleep_status);
256 acpi_gbl_system_awake_and_running = TRUE;
257
Bob Moore4efeeec2012-03-21 09:42:45 +0800258 acpi_hw_execute_sleep_method(METHOD_PATHNAME__SST, ACPI_SST_WORKING);
Bob Moore70958572012-02-14 18:47:42 +0800259 return_ACPI_STATUS(AE_OK);
260}