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/init.h>
14 #include <linux/kernel.h>
15 #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 if (node
->flags
& ACPI_APMT_FLAGS_DUAL_PAGE
) {
39 res
[num_res
].start
= node
->base_address1
;
40 res
[num_res
].end
= node
->base_address1
+ SZ_4K
- 1;
41 res
[num_res
].flags
= IORESOURCE_MEM
;
46 if (node
->ovflw_irq
!= 0) {
47 trigger
= (node
->ovflw_irq_flags
& ACPI_APMT_OVFLW_IRQ_FLAGS_MODE
);
48 trigger
= (trigger
== ACPI_APMT_OVFLW_IRQ_FLAGS_MODE_LEVEL
) ?
49 ACPI_LEVEL_SENSITIVE
: ACPI_EDGE_SENSITIVE
;
50 irq
= acpi_register_gsi(NULL
, node
->ovflw_irq
, trigger
,
54 pr_warn("APMT could not register gsi hwirq %d\n", irq
);
58 res
[num_res
].start
= irq
;
59 res
[num_res
].end
= irq
;
60 res
[num_res
].flags
= IORESOURCE_IRQ
;
69 * apmt_add_platform_device() - Allocate a platform device for APMT node
70 * @node: Pointer to device ACPI APMT node
71 * @fwnode: fwnode associated with the APMT node
73 * Returns: 0 on success, <0 failure
75 static int __init
apmt_add_platform_device(struct acpi_apmt_node
*node
,
76 struct fwnode_handle
*fwnode
)
78 struct platform_device
*pdev
;
80 struct resource res
[DEV_MAX_RESOURCE_COUNT
];
82 pdev
= platform_device_alloc(DEV_NAME
, PLATFORM_DEVID_AUTO
);
86 memset(res
, 0, sizeof(res
));
88 count
= apmt_init_resources(res
, node
);
90 ret
= platform_device_add_resources(pdev
, res
, count
);
95 * Add a copy of APMT node pointer to platform_data to be used to
96 * retrieve APMT data information.
98 ret
= platform_device_add_data(pdev
, &node
, sizeof(node
));
102 pdev
->dev
.fwnode
= fwnode
;
104 ret
= platform_device_add(pdev
);
112 platform_device_put(pdev
);
117 static int __init
apmt_init_platform_devices(void)
119 struct acpi_apmt_node
*apmt_node
;
120 struct acpi_table_apmt
*apmt
;
121 struct fwnode_handle
*fwnode
;
126 * apmt_table and apmt both point to the start of APMT table, but
127 * have different struct types
129 apmt
= (struct acpi_table_apmt
*)apmt_table
;
130 offset
= sizeof(*apmt
);
131 end
= apmt
->header
.length
;
133 while (offset
< end
) {
134 apmt_node
= ACPI_ADD_PTR(struct acpi_apmt_node
, apmt
,
137 fwnode
= acpi_alloc_fwnode_static();
141 ret
= apmt_add_platform_device(apmt_node
, fwnode
);
143 acpi_free_fwnode_static(fwnode
);
147 offset
+= apmt_node
->length
;
153 void __init
acpi_apmt_init(void)
159 * APMT table nodes will be used at runtime after the apmt init,
160 * so we don't need to call acpi_put_table() to release
161 * the APMT table mapping.
163 status
= acpi_get_table(ACPI_SIG_APMT
, 0, &apmt_table
);
165 if (ACPI_FAILURE(status
)) {
166 if (status
!= AE_NOT_FOUND
) {
167 const char *msg
= acpi_format_exception(status
);
169 pr_err("Failed to get APMT table, %s\n", msg
);
175 ret
= apmt_init_platform_devices();
177 pr_err("Failed to initialize APMT platform devices, ret: %d\n", ret
);
178 acpi_put_table(apmt_table
);