Skip to content

Commit a3dc36c

Browse files
author
CKI KWF Bot
committed
Merge: Update platform/x86/amd/pmc
MR: https://gitlab.com/redhat/centos-stream/src/kernel/centos-stream-9/-/merge_requests/7148 The AMD PMC driver interfaces with the Power Management Controller (PMC) to assist with power usage and suspend/resume. JIRA: https://issues.redhat.com/browse/RHEL-102860 Signed-off-by: David Arcari <darcari@redhat.com> Approved-by: Steve Best <sbest@redhat.com> Approved-by: Tony Camuso <tcamuso@redhat.com> Approved-by: Lenny Szubowicz <lszubowi@redhat.com> Approved-by: CKI KWF Bot <cki-ci-bot+kwf-gitlab-com@redhat.com> Merged-by: CKI GitLab Kmaint Pipeline Bot <26919896-cki-kmaint-pipeline-bot@users.noreply.gitlab.com>
2 parents 45686de + 2ae4460 commit a3dc36c

File tree

11 files changed

+945
-454
lines changed

11 files changed

+945
-454
lines changed

arch/x86/kernel/cpu/amd.c

Lines changed: 1 addition & 1 deletion
Original file line numberDiff line numberDiff line change
@@ -9,7 +9,7 @@
99
#include <linux/sched/clock.h>
1010
#include <linux/random.h>
1111
#include <linux/topology.h>
12-
#include <asm/amd/fch.h>
12+
#include <linux/platform_data/x86/amd-fch.h>
1313
#include <asm/processor.h>
1414
#include <asm/apic.h>
1515
#include <asm/cacheinfo.h>

drivers/i2c/busses/i2c-piix4.c

Lines changed: 1 addition & 1 deletion
Original file line numberDiff line numberDiff line change
@@ -34,7 +34,7 @@
3434
#include <linux/dmi.h>
3535
#include <linux/acpi.h>
3636
#include <linux/io.h>
37-
#include <asm/amd/fch.h>
37+
#include <linux/platform_data/x86/amd-fch.h>
3838

3939
#include "i2c-piix4.h"
4040

drivers/platform/x86/amd/pmc/Kconfig

Lines changed: 15 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -18,3 +18,18 @@ config AMD_PMC
1818

1919
If you choose to compile this driver as a module the module will be
2020
called amd-pmc.
21+
22+
config AMD_MP2_STB
23+
bool "AMD SoC MP2 STB function"
24+
depends on AMD_PMC
25+
default AMD_PMC
26+
help
27+
AMD MP2 STB function provides a data buffer used to log debug
28+
information about the system execution during S2Idle suspend/resume.
29+
A data buffer known as the STB (Smart Trace Buffer) is a circular
30+
buffer which is a low-level log for the SoC which is used to debug
31+
any hangs/stalls during S2Idle suspend/resume.
32+
33+
Creates debugfs to get STB, a userspace daemon can access STB log of
34+
last S2Idle suspend/resume which can help to debug if hangs/stalls
35+
during S2Idle suspend/resume.

drivers/platform/x86/amd/pmc/Makefile

Lines changed: 3 additions & 2 deletions
Original file line numberDiff line numberDiff line change
@@ -4,5 +4,6 @@
44
# AMD Power Management Controller Driver
55
#
66

7-
amd-pmc-objs := pmc.o pmc-quirks.o
8-
obj-$(CONFIG_AMD_PMC) += amd-pmc.o
7+
obj-$(CONFIG_AMD_PMC) += amd-pmc.o
8+
amd-pmc-y := pmc.o pmc-quirks.o mp1_stb.o
9+
amd-pmc-$(CONFIG_AMD_MP2_STB) += mp2_stb.o
Lines changed: 332 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,332 @@
1+
// SPDX-License-Identifier: GPL-2.0-or-later
2+
/*
3+
* AMD MP1 Smart Trace Buffer (STB) Layer
4+
*
5+
* Copyright (c) 2024, Advanced Micro Devices, Inc.
6+
* All Rights Reserved.
7+
*
8+
* Authors: Shyam Sundar S K <Shyam-sundar.S-k@amd.com>
9+
* Sanket Goswami <Sanket.Goswami@amd.com>
10+
*/
11+
12+
#define pr_fmt(fmt) KBUILD_MODNAME ": " fmt
13+
14+
#include <asm/amd_nb.h>
15+
#include <linux/debugfs.h>
16+
#include <linux/seq_file.h>
17+
#include <linux/uaccess.h>
18+
19+
#include "pmc.h"
20+
21+
/* STB Spill to DRAM Parameters */
22+
#define S2D_TELEMETRY_DRAMBYTES_MAX 0x1000000
23+
#define S2D_TELEMETRY_BYTES_MAX 0x100000U
24+
#define S2D_RSVD_RAM_SPACE 0x100000
25+
26+
/* STB Registers */
27+
#define AMD_STB_PMI_0 0x03E30600
28+
#define AMD_PMC_STB_DUMMY_PC 0xC6000007
29+
30+
/* STB Spill to DRAM Message Definition */
31+
#define STB_FORCE_FLUSH_DATA 0xCF
32+
#define FIFO_SIZE 4096
33+
34+
/* STB S2D(Spill to DRAM) has different message port offset */
35+
#define AMD_S2D_REGISTER_MESSAGE 0xA20
36+
#define AMD_S2D_REGISTER_RESPONSE 0xA80
37+
#define AMD_S2D_REGISTER_ARGUMENT 0xA88
38+
39+
/* STB S2D (Spill to DRAM) message port offset for 44h model */
40+
#define AMD_GNR_REGISTER_MESSAGE 0x524
41+
#define AMD_GNR_REGISTER_RESPONSE 0x570
42+
#define AMD_GNR_REGISTER_ARGUMENT 0xA40
43+
44+
static bool enable_stb;
45+
module_param(enable_stb, bool, 0644);
46+
MODULE_PARM_DESC(enable_stb, "Enable the STB debug mechanism");
47+
48+
static bool dump_custom_stb;
49+
module_param(dump_custom_stb, bool, 0644);
50+
MODULE_PARM_DESC(dump_custom_stb, "Enable to dump full STB buffer");
51+
52+
enum s2d_arg {
53+
S2D_TELEMETRY_SIZE = 0x01,
54+
S2D_PHYS_ADDR_LOW,
55+
S2D_PHYS_ADDR_HIGH,
56+
S2D_NUM_SAMPLES,
57+
S2D_DRAM_SIZE,
58+
};
59+
60+
struct amd_stb_v2_data {
61+
size_t size;
62+
u8 data[] __counted_by(size);
63+
};
64+
65+
int amd_stb_write(struct amd_pmc_dev *dev, u32 data)
66+
{
67+
int err;
68+
69+
err = amd_smn_write(0, AMD_STB_PMI_0, data);
70+
if (err) {
71+
dev_err(dev->dev, "failed to write data in stb: 0x%X\n", AMD_STB_PMI_0);
72+
return pcibios_err_to_errno(err);
73+
}
74+
75+
return 0;
76+
}
77+
78+
int amd_stb_read(struct amd_pmc_dev *dev, u32 *buf)
79+
{
80+
int i, err;
81+
82+
for (i = 0; i < FIFO_SIZE; i++) {
83+
err = amd_smn_read(0, AMD_STB_PMI_0, buf++);
84+
if (err) {
85+
dev_err(dev->dev, "error reading data from stb: 0x%X\n", AMD_STB_PMI_0);
86+
return pcibios_err_to_errno(err);
87+
}
88+
}
89+
90+
return 0;
91+
}
92+
93+
static int amd_stb_debugfs_open(struct inode *inode, struct file *filp)
94+
{
95+
struct amd_pmc_dev *dev = filp->f_inode->i_private;
96+
u32 size = FIFO_SIZE * sizeof(u32);
97+
u32 *buf;
98+
int rc;
99+
100+
buf = kzalloc(size, GFP_KERNEL);
101+
if (!buf)
102+
return -ENOMEM;
103+
104+
rc = amd_stb_read(dev, buf);
105+
if (rc) {
106+
kfree(buf);
107+
return rc;
108+
}
109+
110+
filp->private_data = buf;
111+
return rc;
112+
}
113+
114+
static ssize_t amd_stb_debugfs_read(struct file *filp, char __user *buf, size_t size, loff_t *pos)
115+
{
116+
if (!filp->private_data)
117+
return -EINVAL;
118+
119+
return simple_read_from_buffer(buf, size, pos, filp->private_data,
120+
FIFO_SIZE * sizeof(u32));
121+
}
122+
123+
static int amd_stb_debugfs_release(struct inode *inode, struct file *filp)
124+
{
125+
kfree(filp->private_data);
126+
return 0;
127+
}
128+
129+
static const struct file_operations amd_stb_debugfs_fops = {
130+
.owner = THIS_MODULE,
131+
.open = amd_stb_debugfs_open,
132+
.read = amd_stb_debugfs_read,
133+
.release = amd_stb_debugfs_release,
134+
};
135+
136+
/* Enhanced STB Firmware Reporting Mechanism */
137+
static int amd_stb_handle_efr(struct file *filp)
138+
{
139+
struct amd_pmc_dev *dev = filp->f_inode->i_private;
140+
struct amd_stb_v2_data *stb_data_arr;
141+
u32 fsize;
142+
143+
fsize = dev->dram_size - S2D_RSVD_RAM_SPACE;
144+
stb_data_arr = kmalloc(struct_size(stb_data_arr, data, fsize), GFP_KERNEL);
145+
if (!stb_data_arr)
146+
return -ENOMEM;
147+
148+
stb_data_arr->size = fsize;
149+
memcpy_fromio(stb_data_arr->data, dev->stb_virt_addr, fsize);
150+
filp->private_data = stb_data_arr;
151+
152+
return 0;
153+
}
154+
155+
static int amd_stb_debugfs_open_v2(struct inode *inode, struct file *filp)
156+
{
157+
struct amd_pmc_dev *dev = filp->f_inode->i_private;
158+
u32 fsize, num_samples, val, stb_rdptr_offset = 0;
159+
struct amd_stb_v2_data *stb_data_arr;
160+
int ret;
161+
162+
/* Write dummy postcode while reading the STB buffer */
163+
ret = amd_stb_write(dev, AMD_PMC_STB_DUMMY_PC);
164+
if (ret)
165+
dev_err(dev->dev, "error writing to STB: %d\n", ret);
166+
167+
/* Spill to DRAM num_samples uses separate SMU message port */
168+
dev->msg_port = MSG_PORT_S2D;
169+
170+
ret = amd_pmc_send_cmd(dev, 0, &val, STB_FORCE_FLUSH_DATA, 1);
171+
if (ret)
172+
dev_dbg_once(dev->dev, "S2D force flush not supported: %d\n", ret);
173+
174+
/*
175+
* We have a custom stb size and the PMFW is supposed to give
176+
* the enhanced dram size. Note that we land here only for the
177+
* platforms that support enhanced dram size reporting.
178+
*/
179+
if (dump_custom_stb)
180+
return amd_stb_handle_efr(filp);
181+
182+
/* Get the num_samples to calculate the last push location */
183+
ret = amd_pmc_send_cmd(dev, S2D_NUM_SAMPLES, &num_samples, dev->stb_arg.s2d_msg_id, true);
184+
/* Clear msg_port for other SMU operation */
185+
dev->msg_port = MSG_PORT_PMC;
186+
if (ret) {
187+
dev_err(dev->dev, "error: S2D_NUM_SAMPLES not supported : %d\n", ret);
188+
return ret;
189+
}
190+
191+
fsize = min(num_samples, S2D_TELEMETRY_BYTES_MAX);
192+
stb_data_arr = kmalloc(struct_size(stb_data_arr, data, fsize), GFP_KERNEL);
193+
if (!stb_data_arr)
194+
return -ENOMEM;
195+
196+
stb_data_arr->size = fsize;
197+
198+
/*
199+
* Start capturing data from the last push location.
200+
* This is for general cases, where the stb limits
201+
* are meant for standard usage.
202+
*/
203+
if (num_samples > S2D_TELEMETRY_BYTES_MAX) {
204+
/* First read oldest data starting 1 behind last write till end of ringbuffer */
205+
stb_rdptr_offset = num_samples % S2D_TELEMETRY_BYTES_MAX;
206+
fsize = S2D_TELEMETRY_BYTES_MAX - stb_rdptr_offset;
207+
208+
memcpy_fromio(stb_data_arr->data, dev->stb_virt_addr + stb_rdptr_offset, fsize);
209+
/* Second copy the newer samples from offset 0 - last write */
210+
memcpy_fromio(stb_data_arr->data + fsize, dev->stb_virt_addr, stb_rdptr_offset);
211+
} else {
212+
memcpy_fromio(stb_data_arr->data, dev->stb_virt_addr, fsize);
213+
}
214+
215+
filp->private_data = stb_data_arr;
216+
217+
return 0;
218+
}
219+
220+
static ssize_t amd_stb_debugfs_read_v2(struct file *filp, char __user *buf, size_t size,
221+
loff_t *pos)
222+
{
223+
struct amd_stb_v2_data *data = filp->private_data;
224+
225+
return simple_read_from_buffer(buf, size, pos, data->data, data->size);
226+
}
227+
228+
static int amd_stb_debugfs_release_v2(struct inode *inode, struct file *filp)
229+
{
230+
kfree(filp->private_data);
231+
return 0;
232+
}
233+
234+
static const struct file_operations amd_stb_debugfs_fops_v2 = {
235+
.owner = THIS_MODULE,
236+
.open = amd_stb_debugfs_open_v2,
237+
.read = amd_stb_debugfs_read_v2,
238+
.release = amd_stb_debugfs_release_v2,
239+
};
240+
241+
static void amd_stb_update_args(struct amd_pmc_dev *dev)
242+
{
243+
if (cpu_feature_enabled(X86_FEATURE_ZEN5))
244+
switch (boot_cpu_data.x86_model) {
245+
case 0x44:
246+
dev->stb_arg.msg = AMD_GNR_REGISTER_MESSAGE;
247+
dev->stb_arg.arg = AMD_GNR_REGISTER_ARGUMENT;
248+
dev->stb_arg.resp = AMD_GNR_REGISTER_RESPONSE;
249+
return;
250+
default:
251+
break;
252+
}
253+
254+
dev->stb_arg.msg = AMD_S2D_REGISTER_MESSAGE;
255+
dev->stb_arg.arg = AMD_S2D_REGISTER_ARGUMENT;
256+
dev->stb_arg.resp = AMD_S2D_REGISTER_RESPONSE;
257+
}
258+
259+
static bool amd_is_stb_supported(struct amd_pmc_dev *dev)
260+
{
261+
switch (dev->cpu_id) {
262+
case AMD_CPU_ID_YC:
263+
case AMD_CPU_ID_CB:
264+
if (boot_cpu_data.x86_model == 0x44)
265+
dev->stb_arg.s2d_msg_id = 0x9B;
266+
else
267+
dev->stb_arg.s2d_msg_id = 0xBE;
268+
break;
269+
case AMD_CPU_ID_PS:
270+
dev->stb_arg.s2d_msg_id = 0x85;
271+
break;
272+
case PCI_DEVICE_ID_AMD_1AH_M20H_ROOT:
273+
case PCI_DEVICE_ID_AMD_1AH_M60H_ROOT:
274+
if (boot_cpu_data.x86_model == 0x70)
275+
dev->stb_arg.s2d_msg_id = 0xF1;
276+
else
277+
dev->stb_arg.s2d_msg_id = 0xDE;
278+
break;
279+
default:
280+
return false;
281+
}
282+
283+
amd_stb_update_args(dev);
284+
return true;
285+
}
286+
287+
int amd_stb_s2d_init(struct amd_pmc_dev *dev)
288+
{
289+
u32 phys_addr_low, phys_addr_hi;
290+
u64 stb_phys_addr;
291+
u32 size = 0;
292+
int ret;
293+
294+
if (!enable_stb)
295+
return 0;
296+
297+
if (amd_is_stb_supported(dev)) {
298+
debugfs_create_file("stb_read", 0644, dev->dbgfs_dir, dev,
299+
&amd_stb_debugfs_fops_v2);
300+
} else {
301+
debugfs_create_file("stb_read", 0644, dev->dbgfs_dir, dev,
302+
&amd_stb_debugfs_fops);
303+
return 0;
304+
}
305+
306+
/* Spill to DRAM feature uses separate SMU message port */
307+
dev->msg_port = MSG_PORT_S2D;
308+
309+
amd_pmc_send_cmd(dev, S2D_TELEMETRY_SIZE, &size, dev->stb_arg.s2d_msg_id, true);
310+
if (size != S2D_TELEMETRY_BYTES_MAX)
311+
return -EIO;
312+
313+
/* Get DRAM size */
314+
ret = amd_pmc_send_cmd(dev, S2D_DRAM_SIZE, &dev->dram_size, dev->stb_arg.s2d_msg_id, true);
315+
if (ret || !dev->dram_size)
316+
dev->dram_size = S2D_TELEMETRY_DRAMBYTES_MAX;
317+
318+
/* Get STB DRAM address */
319+
amd_pmc_send_cmd(dev, S2D_PHYS_ADDR_LOW, &phys_addr_low, dev->stb_arg.s2d_msg_id, true);
320+
amd_pmc_send_cmd(dev, S2D_PHYS_ADDR_HIGH, &phys_addr_hi, dev->stb_arg.s2d_msg_id, true);
321+
322+
stb_phys_addr = ((u64)phys_addr_hi << 32 | phys_addr_low);
323+
324+
/* Clear msg_port for other SMU operation */
325+
dev->msg_port = MSG_PORT_PMC;
326+
327+
dev->stb_virt_addr = devm_ioremap(dev->dev, stb_phys_addr, dev->dram_size);
328+
if (!dev->stb_virt_addr)
329+
return -ENOMEM;
330+
331+
return 0;
332+
}

0 commit comments

Comments
 (0)