1
2
3
4
5
6
7
8
9
10#include <acpi/acpi.h>
11#include "accommon.h"
12#include "acdispat.h"
13#include "acinterp.h"
14#include "amlcode.h"
15
16#define _COMPONENT ACPI_EXECUTER
17ACPI_MODULE_NAME("exfield")
18
19
20
21
22
23#define ACPI_INVALID_PROTOCOL_ID 0x80
24#define ACPI_MAX_PROTOCOL_ID 0x0F
25const u8 acpi_protocol_lengths[] = {
26 ACPI_INVALID_PROTOCOL_ID,
27 ACPI_INVALID_PROTOCOL_ID,
28 0x00,
29 ACPI_INVALID_PROTOCOL_ID,
30 0x01,
31 ACPI_INVALID_PROTOCOL_ID,
32 0x01,
33 ACPI_INVALID_PROTOCOL_ID,
34 0x02,
35 ACPI_INVALID_PROTOCOL_ID,
36 0xFF,
37 0xFF,
38 0x02,
39 0xFF,
40 0xFF,
41 0xFF
42};
43
44#define PCC_MASTER_SUBSPACE 3
45
46
47
48
49
50
51
52#define GENERIC_SUBSPACE_COMMAND(a) (4 == a || a == 5)
53#define MASTER_SUBSPACE_COMMAND(a) (12 <= a && a <= 15)
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71acpi_status
72acpi_ex_get_protocol_buffer_length(u32 protocol_id, u32 *return_length)
73{
74
75 if ((protocol_id > ACPI_MAX_PROTOCOL_ID) ||
76 (acpi_protocol_lengths[protocol_id] == ACPI_INVALID_PROTOCOL_ID)) {
77 ACPI_ERROR((AE_INFO,
78 "Invalid Field/AccessAs protocol ID: 0x%4.4X",
79 protocol_id));
80
81 return (AE_AML_PROTOCOL);
82 }
83
84 *return_length = acpi_protocol_lengths[protocol_id];
85 return (AE_OK);
86}
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104acpi_status
105acpi_ex_read_data_from_field(struct acpi_walk_state *walk_state,
106 union acpi_operand_object *obj_desc,
107 union acpi_operand_object **ret_buffer_desc)
108{
109 acpi_status status;
110 union acpi_operand_object *buffer_desc;
111 void *buffer;
112 u32 buffer_length;
113
114 ACPI_FUNCTION_TRACE_PTR(ex_read_data_from_field, obj_desc);
115
116
117
118 if (!obj_desc) {
119 return_ACPI_STATUS(AE_AML_NO_OPERAND);
120 }
121 if (!ret_buffer_desc) {
122 return_ACPI_STATUS(AE_BAD_PARAMETER);
123 }
124
125 if (obj_desc->common.type == ACPI_TYPE_BUFFER_FIELD) {
126
127
128
129
130 if (!(obj_desc->common.flags & AOPOBJ_DATA_VALID)) {
131 status = acpi_ds_get_buffer_field_arguments(obj_desc);
132 if (ACPI_FAILURE(status)) {
133 return_ACPI_STATUS(status);
134 }
135 }
136 } else if ((obj_desc->common.type == ACPI_TYPE_LOCAL_REGION_FIELD) &&
137 (obj_desc->field.region_obj->region.space_id ==
138 ACPI_ADR_SPACE_SMBUS
139 || obj_desc->field.region_obj->region.space_id ==
140 ACPI_ADR_SPACE_GSBUS
141 || obj_desc->field.region_obj->region.space_id ==
142 ACPI_ADR_SPACE_IPMI)) {
143
144
145
146 status = acpi_ex_read_serial_bus(obj_desc, ret_buffer_desc);
147 return_ACPI_STATUS(status);
148 }
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163 buffer_length =
164 (acpi_size)ACPI_ROUND_BITS_UP_TO_BYTES(obj_desc->field.bit_length);
165
166 if (buffer_length > acpi_gbl_integer_byte_width ||
167 (obj_desc->common.type == ACPI_TYPE_BUFFER_FIELD &&
168 obj_desc->buffer_field.is_create_field)) {
169
170
171
172 buffer_desc = acpi_ut_create_buffer_object(buffer_length);
173 if (!buffer_desc) {
174 return_ACPI_STATUS(AE_NO_MEMORY);
175 }
176 buffer = buffer_desc->buffer.pointer;
177 } else {
178
179
180 buffer_desc = acpi_ut_create_integer_object((u64) 0);
181 if (!buffer_desc) {
182 return_ACPI_STATUS(AE_NO_MEMORY);
183 }
184
185 buffer_length = acpi_gbl_integer_byte_width;
186 buffer = &buffer_desc->integer.value;
187 }
188
189 if ((obj_desc->common.type == ACPI_TYPE_LOCAL_REGION_FIELD) &&
190 (obj_desc->field.region_obj->region.space_id ==
191 ACPI_ADR_SPACE_GPIO)) {
192
193
194
195 status = acpi_ex_read_gpio(obj_desc, buffer);
196 goto exit;
197 } else if ((obj_desc->common.type == ACPI_TYPE_LOCAL_REGION_FIELD) &&
198 (obj_desc->field.region_obj->region.space_id ==
199 ACPI_ADR_SPACE_PLATFORM_COMM)) {
200
201
202
203
204 ACPI_DEBUG_PRINT((ACPI_DB_BFIELD,
205 "PCC FieldRead bits %u\n",
206 obj_desc->field.bit_length));
207
208 memcpy(buffer,
209 obj_desc->field.region_obj->field.internal_pcc_buffer +
210 obj_desc->field.base_byte_offset,
211 (acpi_size)ACPI_ROUND_BITS_UP_TO_BYTES(obj_desc->field.
212 bit_length));
213
214 *ret_buffer_desc = buffer_desc;
215 return AE_OK;
216 }
217
218 ACPI_DEBUG_PRINT((ACPI_DB_BFIELD,
219 "FieldRead [TO]: Obj %p, Type %X, Buf %p, ByteLen %X\n",
220 obj_desc, obj_desc->common.type, buffer,
221 buffer_length));
222 ACPI_DEBUG_PRINT((ACPI_DB_BFIELD,
223 "FieldRead [FROM]: BitLen %X, BitOff %X, ByteOff %X\n",
224 obj_desc->common_field.bit_length,
225 obj_desc->common_field.start_field_bit_offset,
226 obj_desc->common_field.base_byte_offset));
227
228
229
230 acpi_ex_acquire_global_lock(obj_desc->common_field.field_flags);
231
232
233
234 status = acpi_ex_extract_from_field(obj_desc, buffer, buffer_length);
235 acpi_ex_release_global_lock(obj_desc->common_field.field_flags);
236
237exit:
238 if (ACPI_FAILURE(status)) {
239 acpi_ut_remove_reference(buffer_desc);
240 } else {
241 *ret_buffer_desc = buffer_desc;
242 }
243
244 return_ACPI_STATUS(status);
245}
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261acpi_status
262acpi_ex_write_data_to_field(union acpi_operand_object *source_desc,
263 union acpi_operand_object *obj_desc,
264 union acpi_operand_object **result_desc)
265{
266 acpi_status status;
267 u32 buffer_length;
268 u32 data_length;
269 void *buffer;
270
271 ACPI_FUNCTION_TRACE_PTR(ex_write_data_to_field, obj_desc);
272
273
274
275 if (!source_desc || !obj_desc) {
276 return_ACPI_STATUS(AE_AML_NO_OPERAND);
277 }
278
279 if (obj_desc->common.type == ACPI_TYPE_BUFFER_FIELD) {
280
281
282
283
284 if (!(obj_desc->common.flags & AOPOBJ_DATA_VALID)) {
285 status = acpi_ds_get_buffer_field_arguments(obj_desc);
286 if (ACPI_FAILURE(status)) {
287 return_ACPI_STATUS(status);
288 }
289 }
290 } else if ((obj_desc->common.type == ACPI_TYPE_LOCAL_REGION_FIELD) &&
291 (obj_desc->field.region_obj->region.space_id ==
292 ACPI_ADR_SPACE_GPIO)) {
293
294
295
296 status = acpi_ex_write_gpio(source_desc, obj_desc, result_desc);
297 return_ACPI_STATUS(status);
298 } else if ((obj_desc->common.type == ACPI_TYPE_LOCAL_REGION_FIELD) &&
299 (obj_desc->field.region_obj->region.space_id ==
300 ACPI_ADR_SPACE_SMBUS
301 || obj_desc->field.region_obj->region.space_id ==
302 ACPI_ADR_SPACE_GSBUS
303 || obj_desc->field.region_obj->region.space_id ==
304 ACPI_ADR_SPACE_IPMI)) {
305
306
307
308 status =
309 acpi_ex_write_serial_bus(source_desc, obj_desc,
310 result_desc);
311 return_ACPI_STATUS(status);
312 } else if ((obj_desc->common.type == ACPI_TYPE_LOCAL_REGION_FIELD) &&
313 (obj_desc->field.region_obj->region.space_id ==
314 ACPI_ADR_SPACE_PLATFORM_COMM)) {
315
316
317
318
319
320
321
322 data_length =
323 (acpi_size)ACPI_ROUND_BITS_UP_TO_BYTES(obj_desc->field.
324 bit_length);
325 memcpy(obj_desc->field.region_obj->field.internal_pcc_buffer +
326 obj_desc->field.base_byte_offset,
327 source_desc->buffer.pointer, data_length);
328
329 if ((obj_desc->field.region_obj->region.address ==
330 PCC_MASTER_SUBSPACE
331 && MASTER_SUBSPACE_COMMAND(obj_desc->field.
332 base_byte_offset))
333 || GENERIC_SUBSPACE_COMMAND(obj_desc->field.
334 base_byte_offset)) {
335
336
337
338 ACPI_DEBUG_PRINT((ACPI_DB_BFIELD,
339 "PCC COMD field has been written. Invoking PCC handler now.\n"));
340
341 status =
342 acpi_ex_access_region(obj_desc, 0,
343 (u64 *)obj_desc->field.
344 region_obj->field.
345 internal_pcc_buffer,
346 ACPI_WRITE);
347 return_ACPI_STATUS(status);
348 }
349 return (AE_OK);
350 }
351
352
353
354 switch (source_desc->common.type) {
355 case ACPI_TYPE_INTEGER:
356
357 buffer = &source_desc->integer.value;
358 buffer_length = sizeof(source_desc->integer.value);
359 break;
360
361 case ACPI_TYPE_BUFFER:
362
363 buffer = source_desc->buffer.pointer;
364 buffer_length = source_desc->buffer.length;
365 break;
366
367 case ACPI_TYPE_STRING:
368
369 buffer = source_desc->string.pointer;
370 buffer_length = source_desc->string.length;
371 break;
372
373 default:
374 return_ACPI_STATUS(AE_AML_OPERAND_TYPE);
375 }
376
377 ACPI_DEBUG_PRINT((ACPI_DB_BFIELD,
378 "FieldWrite [FROM]: Obj %p (%s:%X), Buf %p, ByteLen %X\n",
379 source_desc,
380 acpi_ut_get_type_name(source_desc->common.type),
381 source_desc->common.type, buffer, buffer_length));
382
383 ACPI_DEBUG_PRINT((ACPI_DB_BFIELD,
384 "FieldWrite [TO]: Obj %p (%s:%X), BitLen %X, BitOff %X, ByteOff %X\n",
385 obj_desc,
386 acpi_ut_get_type_name(obj_desc->common.type),
387 obj_desc->common.type,
388 obj_desc->common_field.bit_length,
389 obj_desc->common_field.start_field_bit_offset,
390 obj_desc->common_field.base_byte_offset));
391
392
393
394 acpi_ex_acquire_global_lock(obj_desc->common_field.field_flags);
395
396
397
398 status = acpi_ex_insert_into_field(obj_desc, buffer, buffer_length);
399 acpi_ex_release_global_lock(obj_desc->common_field.field_flags);
400 return_ACPI_STATUS(status);
401}
402