1 // SPDX-License-Identifier: GPL-2.0
3 * ARM APMT table support.
4 * Design document number: ARM DEN0117.
6 * Copyright (c) 2022, NVIDIA CORPORATION & AFFILIATES.
10 #define pr_fmt(fmt) "ACPI: APMT: " fmt
12 #include <linux/acpi.h>
13 #include <linux/acpi_apmt.h>
14 #include <linux/init.h>
15 #include <linux/kernel.h>
16 #include <linux/platform_device.h>
18 #define DEV_NAME "arm-cs-arch-pmu"
20 /* There can be up to 3 resources: page 0 and 1 address, and interrupt. */
21 #define DEV_MAX_RESOURCE_COUNT 3
23 /* Root pointer to the mapped APMT table */
24 static struct acpi_table_header *apmt_table;
26 static int __init apmt_init_resources(struct resource *res,
27 struct acpi_apmt_node *node)
32 res[num_res].start = node->base_address0;
33 res[num_res].end = node->base_address0 + SZ_4K - 1;
34 res[num_res].flags = IORESOURCE_MEM;
38 res[num_res].start = node->base_address1;
39 res[num_res].end = node->base_address1 + SZ_4K - 1;
40 res[num_res].flags = IORESOURCE_MEM;
44 if (node->ovflw_irq != 0) {
45 trigger = (node->ovflw_irq_flags & ACPI_APMT_OVFLW_IRQ_FLAGS_MODE);
46 trigger = (trigger == ACPI_APMT_OVFLW_IRQ_FLAGS_MODE_LEVEL) ?
47 ACPI_LEVEL_SENSITIVE : ACPI_EDGE_SENSITIVE;
48 irq = acpi_register_gsi(NULL, node->ovflw_irq, trigger,
52 pr_warn("APMT could not register gsi hwirq %d\n", irq);
56 res[num_res].start = irq;
57 res[num_res].end = irq;
58 res[num_res].flags = IORESOURCE_IRQ;
67 * apmt_add_platform_device() - Allocate a platform device for APMT node
68 * @node: Pointer to device ACPI APMT node
69 * @fwnode: fwnode associated with the APMT node
71 * Returns: 0 on success, <0 failure
73 static int __init apmt_add_platform_device(struct acpi_apmt_node *node,
74 struct fwnode_handle *fwnode)
76 struct platform_device *pdev;
78 struct resource res[DEV_MAX_RESOURCE_COUNT];
80 pdev = platform_device_alloc(DEV_NAME, PLATFORM_DEVID_AUTO);
84 memset(res, 0, sizeof(res));
86 count = apmt_init_resources(res, node);
88 ret = platform_device_add_resources(pdev, res, count);
93 * Add a copy of APMT node pointer to platform_data to be used to
94 * retrieve APMT data information.
96 ret = platform_device_add_data(pdev, &node, sizeof(node));
100 pdev->dev.fwnode = fwnode;
102 ret = platform_device_add(pdev);
110 platform_device_put(pdev);
115 static int __init apmt_init_platform_devices(void)
117 struct acpi_apmt_node *apmt_node;
118 struct acpi_table_apmt *apmt;
119 struct fwnode_handle *fwnode;
124 * apmt_table and apmt both point to the start of APMT table, but
125 * have different struct types
127 apmt = (struct acpi_table_apmt *)apmt_table;
128 offset = sizeof(*apmt);
129 end = apmt->header.length;
131 while (offset < end) {
132 apmt_node = ACPI_ADD_PTR(struct acpi_apmt_node, apmt,
135 fwnode = acpi_alloc_fwnode_static();
139 ret = apmt_add_platform_device(apmt_node, fwnode);
141 acpi_free_fwnode_static(fwnode);
145 offset += apmt_node->length;
151 void __init acpi_apmt_init(void)
157 * APMT table nodes will be used at runtime after the apmt init,
158 * so we don't need to call acpi_put_table() to release
159 * the APMT table mapping.
161 status = acpi_get_table(ACPI_SIG_APMT, 0, &apmt_table);
163 if (ACPI_FAILURE(status)) {
164 if (status != AE_NOT_FOUND) {
165 const char *msg = acpi_format_exception(status);
167 pr_err("Failed to get APMT table, %s\n", msg);
173 ret = apmt_init_platform_devices();
175 pr_err("Failed to initialize APMT platform devices, ret: %d\n", ret);
176 acpi_put_table(apmt_table);