blob: 45216c1f9d6f40e6fc42ad7a5cca51c62faa17e5 [file] [log] [blame]
Simon Glass98528d42020-07-07 13:11:42 -06001// SPDX-License-Identifier: GPL-2.0
2/*
3 * Generation of ACPI (Advanced Configuration and Power Interface) tables
4 *
5 * Copyright 2019 Google LLC
6 * Mostly taken from coreboot
7 */
8
9#define LOG_CATEGORY LOGC_ACPI
10
11#include <common.h>
12#include <dm.h>
Simon Glass0f277632020-07-07 13:11:50 -060013#include <log.h>
Simon Glass48342b02020-07-07 13:11:55 -060014#include <uuid.h>
Simon Glass98528d42020-07-07 13:11:42 -060015#include <acpi/acpigen.h>
Simon Glass8a200762020-07-07 13:12:01 -060016#include <acpi/acpi_device.h>
Simon Glass98528d42020-07-07 13:11:42 -060017#include <dm/acpi.h>
18
19u8 *acpigen_get_current(struct acpi_ctx *ctx)
20{
21 return ctx->current;
22}
23
24void acpigen_emit_byte(struct acpi_ctx *ctx, uint data)
25{
26 *(u8 *)ctx->current++ = data;
27}
28
29void acpigen_emit_word(struct acpi_ctx *ctx, uint data)
30{
31 acpigen_emit_byte(ctx, data & 0xff);
32 acpigen_emit_byte(ctx, (data >> 8) & 0xff);
33}
34
35void acpigen_emit_dword(struct acpi_ctx *ctx, uint data)
36{
37 /* Output the value in little-endian format */
38 acpigen_emit_byte(ctx, data & 0xff);
39 acpigen_emit_byte(ctx, (data >> 8) & 0xff);
40 acpigen_emit_byte(ctx, (data >> 16) & 0xff);
41 acpigen_emit_byte(ctx, (data >> 24) & 0xff);
42}
Simon Glass071c4a52020-07-07 13:11:45 -060043
Simon Glass0f277632020-07-07 13:11:50 -060044/*
45 * Maximum length for an ACPI object generated by this code,
46 *
47 * If you need to change this, change acpigen_write_len_f(ctx) and
48 * acpigen_pop_len(ctx)
49 */
50#define ACPIGEN_MAXLEN 0xfffff
51
52void acpigen_write_len_f(struct acpi_ctx *ctx)
53{
54 assert(ctx->ltop < (ACPIGEN_LENSTACK_SIZE - 1));
55 ctx->len_stack[ctx->ltop++] = ctx->current;
56 acpigen_emit_byte(ctx, 0);
57 acpigen_emit_byte(ctx, 0);
58 acpigen_emit_byte(ctx, 0);
59}
60
61void acpigen_pop_len(struct acpi_ctx *ctx)
62{
63 int len;
64 char *p;
65
66 assert(ctx->ltop > 0);
67 p = ctx->len_stack[--ctx->ltop];
68 len = ctx->current - (void *)p;
69 assert(len <= ACPIGEN_MAXLEN);
70 /* generate store length for 0xfffff max */
71 p[0] = ACPI_PKG_LEN_3_BYTES | (len & 0xf);
72 p[1] = len >> 4 & 0xff;
73 p[2] = len >> 12 & 0xff;
74}
75
Simon Glass9238fac2020-07-07 13:11:59 -060076void acpigen_emit_ext_op(struct acpi_ctx *ctx, uint op)
77{
78 acpigen_emit_byte(ctx, EXT_OP_PREFIX);
79 acpigen_emit_byte(ctx, op);
80}
81
Simon Glassedc26802020-07-07 13:11:51 -060082char *acpigen_write_package(struct acpi_ctx *ctx, int nr_el)
83{
84 char *p;
85
86 acpigen_emit_byte(ctx, PACKAGE_OP);
87 acpigen_write_len_f(ctx);
88 p = ctx->current;
89 acpigen_emit_byte(ctx, nr_el);
90
91 return p;
92}
93
Simon Glass8715ce02020-07-07 13:11:52 -060094void acpigen_write_byte(struct acpi_ctx *ctx, unsigned int data)
95{
96 acpigen_emit_byte(ctx, BYTE_PREFIX);
97 acpigen_emit_byte(ctx, data & 0xff);
98}
99
100void acpigen_write_word(struct acpi_ctx *ctx, unsigned int data)
101{
102 acpigen_emit_byte(ctx, WORD_PREFIX);
103 acpigen_emit_word(ctx, data);
104}
105
106void acpigen_write_dword(struct acpi_ctx *ctx, unsigned int data)
107{
108 acpigen_emit_byte(ctx, DWORD_PREFIX);
109 acpigen_emit_dword(ctx, data);
110}
111
112void acpigen_write_qword(struct acpi_ctx *ctx, u64 data)
113{
114 acpigen_emit_byte(ctx, QWORD_PREFIX);
115 acpigen_emit_dword(ctx, data & 0xffffffff);
116 acpigen_emit_dword(ctx, (data >> 32) & 0xffffffff);
117}
118
119void acpigen_write_zero(struct acpi_ctx *ctx)
120{
121 acpigen_emit_byte(ctx, ZERO_OP);
122}
123
124void acpigen_write_one(struct acpi_ctx *ctx)
125{
126 acpigen_emit_byte(ctx, ONE_OP);
127}
128
129void acpigen_write_integer(struct acpi_ctx *ctx, u64 data)
130{
131 if (data == 0)
132 acpigen_write_zero(ctx);
133 else if (data == 1)
134 acpigen_write_one(ctx);
135 else if (data <= 0xff)
136 acpigen_write_byte(ctx, (unsigned char)data);
137 else if (data <= 0xffff)
138 acpigen_write_word(ctx, (unsigned int)data);
139 else if (data <= 0xffffffff)
140 acpigen_write_dword(ctx, (unsigned int)data);
141 else
142 acpigen_write_qword(ctx, data);
143}
144
Simon Glass071c4a52020-07-07 13:11:45 -0600145void acpigen_emit_stream(struct acpi_ctx *ctx, const char *data, int size)
146{
147 int i;
148
149 for (i = 0; i < size; i++)
150 acpigen_emit_byte(ctx, data[i]);
151}
152
153void acpigen_emit_string(struct acpi_ctx *ctx, const char *str)
154{
155 acpigen_emit_stream(ctx, str, str ? strlen(str) : 0);
156 acpigen_emit_byte(ctx, '\0');
157}
Simon Glass45d6d952020-07-07 13:11:53 -0600158
159void acpigen_write_string(struct acpi_ctx *ctx, const char *str)
160{
161 acpigen_emit_byte(ctx, STRING_PREFIX);
162 acpigen_emit_string(ctx, str);
163}
Simon Glass0dc3d512020-07-07 13:11:54 -0600164
165/*
166 * The naming conventions for ACPI namespace names are a bit tricky as
167 * each element has to be 4 chars wide ("All names are a fixed 32 bits.")
168 * and "By convention, when an ASL compiler pads a name shorter than 4
169 * characters, it is done so with trailing underscores ('_')".
170 *
171 * Check sections 5.3, 20.2.2 and 20.4 of ACPI spec 6.3 for details.
172 */
173static void acpigen_emit_simple_namestring(struct acpi_ctx *ctx,
174 const char *name)
175{
176 const char *ptr;
177 int i;
178
179 for (i = 0, ptr = name; i < 4; i++) {
180 if (!*ptr || *ptr == '.')
181 acpigen_emit_byte(ctx, '_');
182 else
183 acpigen_emit_byte(ctx, *ptr++);
184 }
185}
186
187static void acpigen_emit_double_namestring(struct acpi_ctx *ctx,
188 const char *name, int dotpos)
189{
190 acpigen_emit_byte(ctx, DUAL_NAME_PREFIX);
191 acpigen_emit_simple_namestring(ctx, name);
192 acpigen_emit_simple_namestring(ctx, &name[dotpos + 1]);
193}
194
195static void acpigen_emit_multi_namestring(struct acpi_ctx *ctx,
196 const char *name)
197{
198 unsigned char *pathlen;
199 int count = 0;
200
201 acpigen_emit_byte(ctx, MULTI_NAME_PREFIX);
202 pathlen = ctx->current;
203 acpigen_emit_byte(ctx, 0);
204
205 while (*name) {
206 acpigen_emit_simple_namestring(ctx, name);
207 /* find end or next entity */
208 while (*name != '.' && *name)
209 name++;
210 /* forward to next */
211 if (*name == '.')
212 name++;
213 count++;
214 }
215
216 *pathlen = count;
217}
218
219void acpigen_emit_namestring(struct acpi_ctx *ctx, const char *namepath)
220{
221 int dotcount;
222 int dotpos;
223 int i;
224
225 /* We can start with a '\' */
226 if (*namepath == '\\') {
227 acpigen_emit_byte(ctx, '\\');
228 namepath++;
229 }
230
231 /* And there can be any number of '^' */
232 while (*namepath == '^') {
233 acpigen_emit_byte(ctx, '^');
234 namepath++;
235 }
236
237 for (i = 0, dotcount = 0; namepath[i]; i++) {
238 if (namepath[i] == '.') {
239 dotcount++;
240 dotpos = i;
241 }
242 }
243
244 /* If we have only \\ or only ^* then we need to add a null name */
245 if (!*namepath)
246 acpigen_emit_byte(ctx, ZERO_OP);
247 else if (dotcount == 0)
248 acpigen_emit_simple_namestring(ctx, namepath);
249 else if (dotcount == 1)
250 acpigen_emit_double_namestring(ctx, namepath, dotpos);
251 else
252 acpigen_emit_multi_namestring(ctx, namepath);
253}
254
255void acpigen_write_name(struct acpi_ctx *ctx, const char *namepath)
256{
257 acpigen_emit_byte(ctx, NAME_OP);
258 acpigen_emit_namestring(ctx, namepath);
259}
Simon Glass48342b02020-07-07 13:11:55 -0600260
Simon Glass9238fac2020-07-07 13:11:59 -0600261static void acpigen_write_method_internal(struct acpi_ctx *ctx,
262 const char *name, uint flags)
263{
264 acpigen_emit_byte(ctx, METHOD_OP);
265 acpigen_write_len_f(ctx);
266 acpigen_emit_namestring(ctx, name);
267 acpigen_emit_byte(ctx, flags);
268}
269
270/* Method (name, nargs, NotSerialized) */
271void acpigen_write_method(struct acpi_ctx *ctx, const char *name, int nargs)
272{
273 acpigen_write_method_internal(ctx, name,
274 nargs & ACPI_METHOD_NARGS_MASK);
275}
276
277/* Method (name, nargs, Serialized) */
278void acpigen_write_method_serialized(struct acpi_ctx *ctx, const char *name,
279 int nargs)
280{
281 acpigen_write_method_internal(ctx, name,
282 (nargs & ACPI_METHOD_NARGS_MASK) |
283 ACPI_METHOD_SERIALIZED_MASK);
284}
285
286void acpigen_write_sta(struct acpi_ctx *ctx, uint status)
287{
288 /* Method (_STA, 0, NotSerialized) { Return (status) } */
289 acpigen_write_method(ctx, "_STA", 0);
290 acpigen_emit_byte(ctx, RETURN_OP);
291 acpigen_write_byte(ctx, status);
292 acpigen_pop_len(ctx);
293}
294
Simon Glass48342b02020-07-07 13:11:55 -0600295/*
296 * ToUUID(uuid)
297 *
298 * ACPI 6.3 Section 19.6.142 table 19-438 defines a special output order for the
299 * bytes that make up a UUID Buffer object:
300 *
301 * UUID byte order for input to this function:
302 * aabbccdd-eeff-gghh-iijj-kkllmmnnoopp
303 *
304 * UUID byte order output by this function:
305 * ddccbbaa-ffee-hhgg-iijj-kkllmmnnoopp
306 */
307int acpigen_write_uuid(struct acpi_ctx *ctx, const char *uuid)
308{
309 u8 buf[UUID_BIN_LEN];
310 int ret;
311
312 /* Parse UUID string into bytes */
313 ret = uuid_str_to_bin(uuid, buf, UUID_STR_FORMAT_GUID);
314 if (ret)
315 return log_msg_ret("bad hex", -EINVAL);
316
317 /* BufferOp */
318 acpigen_emit_byte(ctx, BUFFER_OP);
319 acpigen_write_len_f(ctx);
320
321 /* Buffer length in bytes */
322 acpigen_write_word(ctx, UUID_BIN_LEN);
323
324 /* Output UUID in expected order */
325 acpigen_emit_stream(ctx, (char *)buf, UUID_BIN_LEN);
326
327 acpigen_pop_len(ctx);
328
329 return 0;
330}
Simon Glass9238fac2020-07-07 13:11:59 -0600331
Simon Glassd0f7d9b2020-07-07 13:12:00 -0600332void acpigen_write_power_res(struct acpi_ctx *ctx, const char *name, uint level,
333 uint order, const char *const dev_states[],
334 size_t dev_states_count)
335{
336 size_t i;
337
338 for (i = 0; i < dev_states_count; i++) {
339 acpigen_write_name(ctx, dev_states[i]);
340 acpigen_write_package(ctx, 1);
341 acpigen_emit_simple_namestring(ctx, name);
342 acpigen_pop_len(ctx); /* Package */
343 }
344
345 acpigen_emit_ext_op(ctx, POWER_RES_OP);
346
347 acpigen_write_len_f(ctx);
348
349 acpigen_emit_simple_namestring(ctx, name);
350 acpigen_emit_byte(ctx, level);
351 acpigen_emit_word(ctx, order);
352}
353
Simon Glass9238fac2020-07-07 13:11:59 -0600354/* Sleep (ms) */
355void acpigen_write_sleep(struct acpi_ctx *ctx, u64 sleep_ms)
356{
357 acpigen_emit_ext_op(ctx, SLEEP_OP);
358 acpigen_write_integer(ctx, sleep_ms);
359}
360
361void acpigen_write_store(struct acpi_ctx *ctx)
362{
363 acpigen_emit_byte(ctx, STORE_OP);
364}
365
366/* Or (arg1, arg2, res) */
367void acpigen_write_or(struct acpi_ctx *ctx, u8 arg1, u8 arg2, u8 res)
368{
369 acpigen_emit_byte(ctx, OR_OP);
370 acpigen_emit_byte(ctx, arg1);
371 acpigen_emit_byte(ctx, arg2);
372 acpigen_emit_byte(ctx, res);
373}
374
375/* And (arg1, arg2, res) */
376void acpigen_write_and(struct acpi_ctx *ctx, u8 arg1, u8 arg2, u8 res)
377{
378 acpigen_emit_byte(ctx, AND_OP);
379 acpigen_emit_byte(ctx, arg1);
380 acpigen_emit_byte(ctx, arg2);
381 acpigen_emit_byte(ctx, res);
382}
383
384/* Not (arg, res) */
385void acpigen_write_not(struct acpi_ctx *ctx, u8 arg, u8 res)
386{
387 acpigen_emit_byte(ctx, NOT_OP);
388 acpigen_emit_byte(ctx, arg);
389 acpigen_emit_byte(ctx, res);
390}
391
392/* Store (str, DEBUG) */
393void acpigen_write_debug_string(struct acpi_ctx *ctx, const char *str)
394{
395 acpigen_write_store(ctx);
396 acpigen_write_string(ctx, str);
397 acpigen_emit_ext_op(ctx, DEBUG_OP);
398}
Simon Glass8a200762020-07-07 13:12:01 -0600399
400/**
401 * acpigen_get_dw0_in_local5() - Generate code to put dw0 cfg0 in local5
402 *
403 * Store (\_SB.GPC0 (addr), Local5)
404 *
405 * \_SB.GPC0 is used to read cfg0 value from dw0. It is typically defined in
406 * the board's gpiolib.asl
407 *
408 * The value needs to be stored in a local variable so that it can be used in
409 * expressions in the ACPI code.
410 *
411 * @ctx: ACPI context pointer
412 * @dw0_read: Name to use to read dw0, e.g. "\\_SB.GPC0"
413 * @addr: GPIO pin configuration register address
414 *
415 */
416static void acpigen_get_dw0_in_local5(struct acpi_ctx *ctx,
417 const char *dw0_read, ulong addr)
418{
419 acpigen_write_store(ctx);
420 acpigen_emit_namestring(ctx, dw0_read);
421 acpigen_write_integer(ctx, addr);
422 acpigen_emit_byte(ctx, LOCAL5_OP);
423}
424
425/**
426 * acpigen_set_gpio_val() - Emit code to set value of TX GPIO to on/off
427 *
428 * @ctx: ACPI context pointer
429 * @dw0_read: Method name to use to read dw0, e.g. "\\_SB.GPC0"
430 * @dw0_write: Method name to use to read dw0, e.g. "\\_SB.SPC0"
431 * @gpio_num: GPIO number to adjust
432 * @vaL: true to set on, false to set off
433 */
434static int acpigen_set_gpio_val(struct acpi_ctx *ctx, u32 tx_state_val,
435 const char *dw0_read, const char *dw0_write,
436 struct acpi_gpio *gpio, bool val)
437{
438 acpigen_get_dw0_in_local5(ctx, dw0_read, gpio->pin0_addr);
439
440 /* Store (0x40, Local0) */
441 acpigen_write_store(ctx);
442 acpigen_write_integer(ctx, tx_state_val);
443 acpigen_emit_byte(ctx, LOCAL0_OP);
444
445 if (val) {
446 /* Or (Local5, PAD_CFG0_TX_STATE, Local5) */
447 acpigen_write_or(ctx, LOCAL5_OP, LOCAL0_OP, LOCAL5_OP);
448 } else {
449 /* Not (PAD_CFG0_TX_STATE, Local6) */
450 acpigen_write_not(ctx, LOCAL0_OP, LOCAL6_OP);
451
452 /* And (Local5, Local6, Local5) */
453 acpigen_write_and(ctx, LOCAL5_OP, LOCAL6_OP, LOCAL5_OP);
454 }
455
456 /*
457 * \_SB.SPC0 (addr, Local5)
458 * \_SB.SPC0 is used to write cfg0 value in dw0. It is defined in
459 * gpiolib.asl.
460 */
461 acpigen_emit_namestring(ctx, dw0_write);
462 acpigen_write_integer(ctx, gpio->pin0_addr);
463 acpigen_emit_byte(ctx, LOCAL5_OP);
464
465 return 0;
466}
467
468int acpigen_set_enable_tx_gpio(struct acpi_ctx *ctx, u32 tx_state_val,
469 const char *dw0_read, const char *dw0_write,
470 struct acpi_gpio *gpio, bool enable)
471{
472 bool set;
473 int ret;
474
475 set = gpio->polarity == ACPI_GPIO_ACTIVE_HIGH ? enable : !enable;
476 ret = acpigen_set_gpio_val(ctx, tx_state_val, dw0_read, dw0_write, gpio,
477 set);
478 if (ret)
479 return log_msg_ret("call", ret);
480
481 return 0;
482}