/src/u-boot/lib/acpi/acpi_writer.c
Line | Count | Source |
1 | | // SPDX-License-Identifier: GPL-2.0+ |
2 | | /* |
3 | | * Handles writing the declared ACPI tables |
4 | | * |
5 | | * Copyright 2021 Google LLC |
6 | | */ |
7 | | |
8 | | #define LOG_CATEGORY LOGC_ACPI |
9 | | |
10 | | #include <log.h> |
11 | | #include <malloc.h> |
12 | | #include <mapmem.h> |
13 | | #include <acpi/acpi_table.h> |
14 | | #include <asm/global_data.h> |
15 | | #include <dm/acpi.h> |
16 | | #include <linux/errno.h> |
17 | | |
18 | | DECLARE_GLOBAL_DATA_PTR; |
19 | | |
20 | | int acpi_write_one(struct acpi_ctx *ctx, const struct acpi_writer *entry) |
21 | 0 | { |
22 | 0 | int ret; |
23 | |
|
24 | 0 | log_debug("%s: writing table '%s'\n", entry->name, |
25 | 0 | entry->table); |
26 | 0 | ctx->tab_start = ctx->current; |
27 | 0 | ret = entry->h_write(ctx, entry); |
28 | 0 | if (ret == -ENOENT) { |
29 | 0 | log_debug("%s: Omitted due to being empty\n", |
30 | 0 | entry->name); |
31 | 0 | ret = 0; |
32 | 0 | ctx->current = ctx->tab_start; /* drop the table */ |
33 | 0 | return ret; |
34 | 0 | } |
35 | 0 | if (ret) |
36 | 0 | return log_msg_ret("write", ret); |
37 | | |
38 | 0 | if (entry->flags & ACPIWF_ALIGN64) |
39 | 0 | acpi_align64(ctx); |
40 | 0 | else |
41 | 0 | acpi_align(ctx); |
42 | | |
43 | | /* Add the item to the internal list */ |
44 | 0 | ret = acpi_add_other_item(ctx, entry, ctx->tab_start); |
45 | 0 | if (ret) |
46 | 0 | return log_msg_ret("add", ret); |
47 | | |
48 | 0 | return 0; |
49 | 0 | } |
50 | | |
51 | | #ifndef CONFIG_QFW_ACPI |
52 | | static int acpi_write_all(struct acpi_ctx *ctx) |
53 | 0 | { |
54 | 0 | const struct acpi_writer *writer = |
55 | 0 | ll_entry_start(struct acpi_writer, acpi_writer); |
56 | 0 | const int n_ents = ll_entry_count(struct acpi_writer, acpi_writer); |
57 | 0 | const struct acpi_writer *entry; |
58 | 0 | int ret; |
59 | |
|
60 | 0 | for (entry = writer; entry != writer + n_ents; entry++) { |
61 | 0 | ret = acpi_write_one(ctx, entry); |
62 | 0 | if (ret && ret != -ENOENT) |
63 | 0 | return log_msg_ret("one", ret); |
64 | 0 | } |
65 | | |
66 | 0 | return 0; |
67 | 0 | } |
68 | | |
69 | | /* |
70 | | * QEMU's version of write_acpi_tables is defined in drivers/misc/qfw.c |
71 | | */ |
72 | | ulong write_acpi_tables(ulong start_addr) |
73 | 0 | { |
74 | 0 | struct acpi_ctx *ctx; |
75 | 0 | ulong addr; |
76 | 0 | int ret; |
77 | |
|
78 | 0 | ctx = malloc(sizeof(*ctx)); |
79 | 0 | if (!ctx) |
80 | 0 | return log_msg_ret("mem", -ENOMEM); |
81 | | |
82 | 0 | log_debug("ACPI: Writing ACPI tables at %lx\n", start_addr); |
83 | |
|
84 | 0 | acpi_reset_items(); |
85 | 0 | acpi_setup_ctx(ctx, start_addr); |
86 | |
|
87 | 0 | ret = acpi_write_all(ctx); |
88 | 0 | if (ret) { |
89 | 0 | log_err("Failed to write ACPI tables (err=%d)\n", ret); |
90 | 0 | return log_msg_ret("write", -ENOMEM); |
91 | 0 | } |
92 | | |
93 | 0 | addr = map_to_sysmem(ctx->current); |
94 | 0 | log_debug("ACPI current = %lx\n", addr); |
95 | |
|
96 | 0 | return addr; |
97 | 0 | } |
98 | | |
99 | | int write_dev_tables(struct acpi_ctx *ctx, const struct acpi_writer *entry) |
100 | 0 | { |
101 | 0 | int ret; |
102 | |
|
103 | 0 | ret = acpi_write_dev_tables(ctx); |
104 | 0 | if (ret) |
105 | 0 | return log_msg_ret("write", ret); |
106 | | |
107 | 0 | return 0; |
108 | 0 | } |
109 | | ACPI_WRITER(8dev, NULL, write_dev_tables, 0); |
110 | | |
111 | | ulong acpi_get_rsdp_addr(void) |
112 | 0 | { |
113 | 0 | if (!gd->acpi_ctx) |
114 | 0 | return 0; |
115 | | |
116 | 0 | return map_to_sysmem(gd->acpi_ctx->rsdp); |
117 | 0 | } |
118 | | #endif /* QFW_ACPI */ |
119 | | |
120 | | void acpi_setup_ctx(struct acpi_ctx *ctx, ulong start) |
121 | 0 | { |
122 | 0 | gd->acpi_ctx = ctx; |
123 | 0 | memset(ctx, '\0', sizeof(*ctx)); |
124 | | |
125 | | /* Align ACPI tables to 16-byte boundary */ |
126 | 0 | start = ALIGN(start, 16); |
127 | 0 | ctx->base = map_sysmem(start, 0); |
128 | 0 | ctx->current = ctx->base; |
129 | |
|
130 | 0 | gd_set_acpi_start(start); |
131 | 0 | } |