xref: /linux/drivers/platform/x86/amd/hsmp/hsmp.c (revision 9df193087b9ed58a244faa516334539661fe7f11)
1*9df19308SSuma Hegde // SPDX-License-Identifier: GPL-2.0
2*9df19308SSuma Hegde /*
3*9df19308SSuma Hegde  * AMD HSMP Platform Driver
4*9df19308SSuma Hegde  * Copyright (c) 2022, AMD.
5*9df19308SSuma Hegde  * All Rights Reserved.
6*9df19308SSuma Hegde  *
7*9df19308SSuma Hegde  * This file provides a device implementation for HSMP interface
8*9df19308SSuma Hegde  */
9*9df19308SSuma Hegde 
10*9df19308SSuma Hegde #define pr_fmt(fmt) KBUILD_MODNAME ": " fmt
11*9df19308SSuma Hegde 
12*9df19308SSuma Hegde #include <asm/amd_hsmp.h>
13*9df19308SSuma Hegde #include <asm/amd_nb.h>
14*9df19308SSuma Hegde #include <linux/delay.h>
15*9df19308SSuma Hegde #include <linux/io.h>
16*9df19308SSuma Hegde #include <linux/miscdevice.h>
17*9df19308SSuma Hegde #include <linux/module.h>
18*9df19308SSuma Hegde #include <linux/pci.h>
19*9df19308SSuma Hegde #include <linux/platform_device.h>
20*9df19308SSuma Hegde #include <linux/semaphore.h>
21*9df19308SSuma Hegde #include <linux/acpi.h>
22*9df19308SSuma Hegde 
23*9df19308SSuma Hegde #define DRIVER_NAME		"amd_hsmp"
24*9df19308SSuma Hegde #define DRIVER_VERSION		"2.2"
25*9df19308SSuma Hegde #define ACPI_HSMP_DEVICE_HID	"AMDI0097"
26*9df19308SSuma Hegde 
27*9df19308SSuma Hegde /* HSMP Status / Error codes */
28*9df19308SSuma Hegde #define HSMP_STATUS_NOT_READY	0x00
29*9df19308SSuma Hegde #define HSMP_STATUS_OK		0x01
30*9df19308SSuma Hegde #define HSMP_ERR_INVALID_MSG	0xFE
31*9df19308SSuma Hegde #define HSMP_ERR_INVALID_INPUT	0xFF
32*9df19308SSuma Hegde 
33*9df19308SSuma Hegde /* Timeout in millsec */
34*9df19308SSuma Hegde #define HSMP_MSG_TIMEOUT	100
35*9df19308SSuma Hegde #define HSMP_SHORT_SLEEP	1
36*9df19308SSuma Hegde 
37*9df19308SSuma Hegde #define HSMP_WR			true
38*9df19308SSuma Hegde #define HSMP_RD			false
39*9df19308SSuma Hegde 
40*9df19308SSuma Hegde /*
41*9df19308SSuma Hegde  * To access specific HSMP mailbox register, s/w writes the SMN address of HSMP mailbox
42*9df19308SSuma Hegde  * register into the SMN_INDEX register, and reads/writes the SMN_DATA reg.
43*9df19308SSuma Hegde  * Below are required SMN address for HSMP Mailbox register offsets in SMU address space
44*9df19308SSuma Hegde  */
45*9df19308SSuma Hegde #define SMN_HSMP_BASE		0x3B00000
46*9df19308SSuma Hegde #define SMN_HSMP_MSG_ID		0x0010534
47*9df19308SSuma Hegde #define SMN_HSMP_MSG_ID_F1A_M0H	0x0010934
48*9df19308SSuma Hegde #define SMN_HSMP_MSG_RESP	0x0010980
49*9df19308SSuma Hegde #define SMN_HSMP_MSG_DATA	0x00109E0
50*9df19308SSuma Hegde 
51*9df19308SSuma Hegde #define HSMP_INDEX_REG		0xc4
52*9df19308SSuma Hegde #define HSMP_DATA_REG		0xc8
53*9df19308SSuma Hegde 
54*9df19308SSuma Hegde #define HSMP_CDEV_NAME		"hsmp_cdev"
55*9df19308SSuma Hegde #define HSMP_DEVNODE_NAME	"hsmp"
56*9df19308SSuma Hegde #define HSMP_METRICS_TABLE_NAME	"metrics_bin"
57*9df19308SSuma Hegde 
58*9df19308SSuma Hegde #define HSMP_ATTR_GRP_NAME_SIZE	10
59*9df19308SSuma Hegde 
60*9df19308SSuma Hegde /* These are the strings specified in ACPI table */
61*9df19308SSuma Hegde #define MSG_IDOFF_STR		"MsgIdOffset"
62*9df19308SSuma Hegde #define MSG_ARGOFF_STR		"MsgArgOffset"
63*9df19308SSuma Hegde #define MSG_RESPOFF_STR		"MsgRspOffset"
64*9df19308SSuma Hegde 
65*9df19308SSuma Hegde #define MAX_AMD_SOCKETS 8
66*9df19308SSuma Hegde 
67*9df19308SSuma Hegde struct hsmp_mbaddr_info {
68*9df19308SSuma Hegde 	u32 base_addr;
69*9df19308SSuma Hegde 	u32 msg_id_off;
70*9df19308SSuma Hegde 	u32 msg_resp_off;
71*9df19308SSuma Hegde 	u32 msg_arg_off;
72*9df19308SSuma Hegde 	u32 size;
73*9df19308SSuma Hegde };
74*9df19308SSuma Hegde 
75*9df19308SSuma Hegde struct hsmp_socket {
76*9df19308SSuma Hegde 	struct bin_attribute hsmp_attr;
77*9df19308SSuma Hegde 	struct hsmp_mbaddr_info mbinfo;
78*9df19308SSuma Hegde 	void __iomem *metric_tbl_addr;
79*9df19308SSuma Hegde 	void __iomem *virt_base_addr;
80*9df19308SSuma Hegde 	struct semaphore hsmp_sem;
81*9df19308SSuma Hegde 	char name[HSMP_ATTR_GRP_NAME_SIZE];
82*9df19308SSuma Hegde 	struct pci_dev *root;
83*9df19308SSuma Hegde 	struct device *dev;
84*9df19308SSuma Hegde 	u16 sock_ind;
85*9df19308SSuma Hegde };
86*9df19308SSuma Hegde 
87*9df19308SSuma Hegde struct hsmp_plat_device {
88*9df19308SSuma Hegde 	struct miscdevice hsmp_device;
89*9df19308SSuma Hegde 	struct hsmp_socket *sock;
90*9df19308SSuma Hegde 	u32 proto_ver;
91*9df19308SSuma Hegde 	u16 num_sockets;
92*9df19308SSuma Hegde 	bool is_acpi_device;
93*9df19308SSuma Hegde 	bool is_probed;
94*9df19308SSuma Hegde };
95*9df19308SSuma Hegde 
96*9df19308SSuma Hegde static struct hsmp_plat_device plat_dev;
97*9df19308SSuma Hegde 
98*9df19308SSuma Hegde static int amd_hsmp_pci_rdwr(struct hsmp_socket *sock, u32 offset,
99*9df19308SSuma Hegde 			     u32 *value, bool write)
100*9df19308SSuma Hegde {
101*9df19308SSuma Hegde 	int ret;
102*9df19308SSuma Hegde 
103*9df19308SSuma Hegde 	if (!sock->root)
104*9df19308SSuma Hegde 		return -ENODEV;
105*9df19308SSuma Hegde 
106*9df19308SSuma Hegde 	ret = pci_write_config_dword(sock->root, HSMP_INDEX_REG,
107*9df19308SSuma Hegde 				     sock->mbinfo.base_addr + offset);
108*9df19308SSuma Hegde 	if (ret)
109*9df19308SSuma Hegde 		return ret;
110*9df19308SSuma Hegde 
111*9df19308SSuma Hegde 	ret = (write ? pci_write_config_dword(sock->root, HSMP_DATA_REG, *value)
112*9df19308SSuma Hegde 		     : pci_read_config_dword(sock->root, HSMP_DATA_REG, value));
113*9df19308SSuma Hegde 
114*9df19308SSuma Hegde 	return ret;
115*9df19308SSuma Hegde }
116*9df19308SSuma Hegde 
117*9df19308SSuma Hegde static void amd_hsmp_acpi_rdwr(struct hsmp_socket *sock, u32 offset,
118*9df19308SSuma Hegde 			       u32 *value, bool write)
119*9df19308SSuma Hegde {
120*9df19308SSuma Hegde 	if (write)
121*9df19308SSuma Hegde 		iowrite32(*value, sock->virt_base_addr + offset);
122*9df19308SSuma Hegde 	else
123*9df19308SSuma Hegde 		*value = ioread32(sock->virt_base_addr + offset);
124*9df19308SSuma Hegde }
125*9df19308SSuma Hegde 
126*9df19308SSuma Hegde static int amd_hsmp_rdwr(struct hsmp_socket *sock, u32 offset,
127*9df19308SSuma Hegde 			 u32 *value, bool write)
128*9df19308SSuma Hegde {
129*9df19308SSuma Hegde 	if (plat_dev.is_acpi_device)
130*9df19308SSuma Hegde 		amd_hsmp_acpi_rdwr(sock, offset, value, write);
131*9df19308SSuma Hegde 	else
132*9df19308SSuma Hegde 		return amd_hsmp_pci_rdwr(sock, offset, value, write);
133*9df19308SSuma Hegde 
134*9df19308SSuma Hegde 	return 0;
135*9df19308SSuma Hegde }
136*9df19308SSuma Hegde 
137*9df19308SSuma Hegde /*
138*9df19308SSuma Hegde  * Send a message to the HSMP port via PCI-e config space registers
139*9df19308SSuma Hegde  * or by writing to MMIO space.
140*9df19308SSuma Hegde  *
141*9df19308SSuma Hegde  * The caller is expected to zero out any unused arguments.
142*9df19308SSuma Hegde  * If a response is expected, the number of response words should be greater than 0.
143*9df19308SSuma Hegde  *
144*9df19308SSuma Hegde  * Returns 0 for success and populates the requested number of arguments.
145*9df19308SSuma Hegde  * Returns a negative error code for failure.
146*9df19308SSuma Hegde  */
147*9df19308SSuma Hegde static int __hsmp_send_message(struct hsmp_socket *sock, struct hsmp_message *msg)
148*9df19308SSuma Hegde {
149*9df19308SSuma Hegde 	struct hsmp_mbaddr_info *mbinfo;
150*9df19308SSuma Hegde 	unsigned long timeout, short_sleep;
151*9df19308SSuma Hegde 	u32 mbox_status;
152*9df19308SSuma Hegde 	u32 index;
153*9df19308SSuma Hegde 	int ret;
154*9df19308SSuma Hegde 
155*9df19308SSuma Hegde 	mbinfo = &sock->mbinfo;
156*9df19308SSuma Hegde 
157*9df19308SSuma Hegde 	/* Clear the status register */
158*9df19308SSuma Hegde 	mbox_status = HSMP_STATUS_NOT_READY;
159*9df19308SSuma Hegde 	ret = amd_hsmp_rdwr(sock, mbinfo->msg_resp_off, &mbox_status, HSMP_WR);
160*9df19308SSuma Hegde 	if (ret) {
161*9df19308SSuma Hegde 		pr_err("Error %d clearing mailbox status register\n", ret);
162*9df19308SSuma Hegde 		return ret;
163*9df19308SSuma Hegde 	}
164*9df19308SSuma Hegde 
165*9df19308SSuma Hegde 	index = 0;
166*9df19308SSuma Hegde 	/* Write any message arguments */
167*9df19308SSuma Hegde 	while (index < msg->num_args) {
168*9df19308SSuma Hegde 		ret = amd_hsmp_rdwr(sock, mbinfo->msg_arg_off + (index << 2),
169*9df19308SSuma Hegde 				    &msg->args[index], HSMP_WR);
170*9df19308SSuma Hegde 		if (ret) {
171*9df19308SSuma Hegde 			pr_err("Error %d writing message argument %d\n", ret, index);
172*9df19308SSuma Hegde 			return ret;
173*9df19308SSuma Hegde 		}
174*9df19308SSuma Hegde 		index++;
175*9df19308SSuma Hegde 	}
176*9df19308SSuma Hegde 
177*9df19308SSuma Hegde 	/* Write the message ID which starts the operation */
178*9df19308SSuma Hegde 	ret = amd_hsmp_rdwr(sock, mbinfo->msg_id_off, &msg->msg_id, HSMP_WR);
179*9df19308SSuma Hegde 	if (ret) {
180*9df19308SSuma Hegde 		pr_err("Error %d writing message ID %u\n", ret, msg->msg_id);
181*9df19308SSuma Hegde 		return ret;
182*9df19308SSuma Hegde 	}
183*9df19308SSuma Hegde 
184*9df19308SSuma Hegde 	/*
185*9df19308SSuma Hegde 	 * Depending on when the trigger write completes relative to the SMU
186*9df19308SSuma Hegde 	 * firmware 1 ms cycle, the operation may take from tens of us to 1 ms
187*9df19308SSuma Hegde 	 * to complete. Some operations may take more. Therefore we will try
188*9df19308SSuma Hegde 	 * a few short duration sleeps and switch to long sleeps if we don't
189*9df19308SSuma Hegde 	 * succeed quickly.
190*9df19308SSuma Hegde 	 */
191*9df19308SSuma Hegde 	short_sleep = jiffies + msecs_to_jiffies(HSMP_SHORT_SLEEP);
192*9df19308SSuma Hegde 	timeout	= jiffies + msecs_to_jiffies(HSMP_MSG_TIMEOUT);
193*9df19308SSuma Hegde 
194*9df19308SSuma Hegde 	while (time_before(jiffies, timeout)) {
195*9df19308SSuma Hegde 		ret = amd_hsmp_rdwr(sock, mbinfo->msg_resp_off, &mbox_status, HSMP_RD);
196*9df19308SSuma Hegde 		if (ret) {
197*9df19308SSuma Hegde 			pr_err("Error %d reading mailbox status\n", ret);
198*9df19308SSuma Hegde 			return ret;
199*9df19308SSuma Hegde 		}
200*9df19308SSuma Hegde 
201*9df19308SSuma Hegde 		if (mbox_status != HSMP_STATUS_NOT_READY)
202*9df19308SSuma Hegde 			break;
203*9df19308SSuma Hegde 		if (time_before(jiffies, short_sleep))
204*9df19308SSuma Hegde 			usleep_range(50, 100);
205*9df19308SSuma Hegde 		else
206*9df19308SSuma Hegde 			usleep_range(1000, 2000);
207*9df19308SSuma Hegde 	}
208*9df19308SSuma Hegde 
209*9df19308SSuma Hegde 	if (unlikely(mbox_status == HSMP_STATUS_NOT_READY)) {
210*9df19308SSuma Hegde 		return -ETIMEDOUT;
211*9df19308SSuma Hegde 	} else if (unlikely(mbox_status == HSMP_ERR_INVALID_MSG)) {
212*9df19308SSuma Hegde 		return -ENOMSG;
213*9df19308SSuma Hegde 	} else if (unlikely(mbox_status == HSMP_ERR_INVALID_INPUT)) {
214*9df19308SSuma Hegde 		return -EINVAL;
215*9df19308SSuma Hegde 	} else if (unlikely(mbox_status != HSMP_STATUS_OK)) {
216*9df19308SSuma Hegde 		pr_err("Message ID %u unknown failure (status = 0x%X)\n",
217*9df19308SSuma Hegde 		       msg->msg_id, mbox_status);
218*9df19308SSuma Hegde 		return -EIO;
219*9df19308SSuma Hegde 	}
220*9df19308SSuma Hegde 
221*9df19308SSuma Hegde 	/*
222*9df19308SSuma Hegde 	 * SMU has responded OK. Read response data.
223*9df19308SSuma Hegde 	 * SMU reads the input arguments from eight 32 bit registers starting
224*9df19308SSuma Hegde 	 * from SMN_HSMP_MSG_DATA and writes the response data to the same
225*9df19308SSuma Hegde 	 * SMN_HSMP_MSG_DATA address.
226*9df19308SSuma Hegde 	 * We copy the response data if any, back to the args[].
227*9df19308SSuma Hegde 	 */
228*9df19308SSuma Hegde 	index = 0;
229*9df19308SSuma Hegde 	while (index < msg->response_sz) {
230*9df19308SSuma Hegde 		ret = amd_hsmp_rdwr(sock, mbinfo->msg_arg_off + (index << 2),
231*9df19308SSuma Hegde 				    &msg->args[index], HSMP_RD);
232*9df19308SSuma Hegde 		if (ret) {
233*9df19308SSuma Hegde 			pr_err("Error %d reading response %u for message ID:%u\n",
234*9df19308SSuma Hegde 			       ret, index, msg->msg_id);
235*9df19308SSuma Hegde 			break;
236*9df19308SSuma Hegde 		}
237*9df19308SSuma Hegde 		index++;
238*9df19308SSuma Hegde 	}
239*9df19308SSuma Hegde 
240*9df19308SSuma Hegde 	return ret;
241*9df19308SSuma Hegde }
242*9df19308SSuma Hegde 
243*9df19308SSuma Hegde static int validate_message(struct hsmp_message *msg)
244*9df19308SSuma Hegde {
245*9df19308SSuma Hegde 	/* msg_id against valid range of message IDs */
246*9df19308SSuma Hegde 	if (msg->msg_id < HSMP_TEST || msg->msg_id >= HSMP_MSG_ID_MAX)
247*9df19308SSuma Hegde 		return -ENOMSG;
248*9df19308SSuma Hegde 
249*9df19308SSuma Hegde 	/* msg_id is a reserved message ID */
250*9df19308SSuma Hegde 	if (hsmp_msg_desc_table[msg->msg_id].type == HSMP_RSVD)
251*9df19308SSuma Hegde 		return -ENOMSG;
252*9df19308SSuma Hegde 
253*9df19308SSuma Hegde 	/* num_args and response_sz against the HSMP spec */
254*9df19308SSuma Hegde 	if (msg->num_args != hsmp_msg_desc_table[msg->msg_id].num_args ||
255*9df19308SSuma Hegde 	    msg->response_sz != hsmp_msg_desc_table[msg->msg_id].response_sz)
256*9df19308SSuma Hegde 		return -EINVAL;
257*9df19308SSuma Hegde 
258*9df19308SSuma Hegde 	return 0;
259*9df19308SSuma Hegde }
260*9df19308SSuma Hegde 
261*9df19308SSuma Hegde int hsmp_send_message(struct hsmp_message *msg)
262*9df19308SSuma Hegde {
263*9df19308SSuma Hegde 	struct hsmp_socket *sock;
264*9df19308SSuma Hegde 	int ret;
265*9df19308SSuma Hegde 
266*9df19308SSuma Hegde 	if (!msg)
267*9df19308SSuma Hegde 		return -EINVAL;
268*9df19308SSuma Hegde 	ret = validate_message(msg);
269*9df19308SSuma Hegde 	if (ret)
270*9df19308SSuma Hegde 		return ret;
271*9df19308SSuma Hegde 
272*9df19308SSuma Hegde 	if (!plat_dev.sock || msg->sock_ind >= plat_dev.num_sockets)
273*9df19308SSuma Hegde 		return -ENODEV;
274*9df19308SSuma Hegde 	sock = &plat_dev.sock[msg->sock_ind];
275*9df19308SSuma Hegde 
276*9df19308SSuma Hegde 	/*
277*9df19308SSuma Hegde 	 * The time taken by smu operation to complete is between
278*9df19308SSuma Hegde 	 * 10us to 1ms. Sometime it may take more time.
279*9df19308SSuma Hegde 	 * In SMP system timeout of 100 millisecs should
280*9df19308SSuma Hegde 	 * be enough for the previous thread to finish the operation
281*9df19308SSuma Hegde 	 */
282*9df19308SSuma Hegde 	ret = down_timeout(&sock->hsmp_sem, msecs_to_jiffies(HSMP_MSG_TIMEOUT));
283*9df19308SSuma Hegde 	if (ret < 0)
284*9df19308SSuma Hegde 		return ret;
285*9df19308SSuma Hegde 
286*9df19308SSuma Hegde 	ret = __hsmp_send_message(sock, msg);
287*9df19308SSuma Hegde 
288*9df19308SSuma Hegde 	up(&sock->hsmp_sem);
289*9df19308SSuma Hegde 
290*9df19308SSuma Hegde 	return ret;
291*9df19308SSuma Hegde }
292*9df19308SSuma Hegde EXPORT_SYMBOL_GPL(hsmp_send_message);
293*9df19308SSuma Hegde 
294*9df19308SSuma Hegde static int hsmp_test(u16 sock_ind, u32 value)
295*9df19308SSuma Hegde {
296*9df19308SSuma Hegde 	struct hsmp_message msg = { 0 };
297*9df19308SSuma Hegde 	int ret;
298*9df19308SSuma Hegde 
299*9df19308SSuma Hegde 	/*
300*9df19308SSuma Hegde 	 * Test the hsmp port by performing TEST command. The test message
301*9df19308SSuma Hegde 	 * takes one argument and returns the value of that argument + 1.
302*9df19308SSuma Hegde 	 */
303*9df19308SSuma Hegde 	msg.msg_id	= HSMP_TEST;
304*9df19308SSuma Hegde 	msg.num_args	= 1;
305*9df19308SSuma Hegde 	msg.response_sz	= 1;
306*9df19308SSuma Hegde 	msg.args[0]	= value;
307*9df19308SSuma Hegde 	msg.sock_ind	= sock_ind;
308*9df19308SSuma Hegde 
309*9df19308SSuma Hegde 	ret = hsmp_send_message(&msg);
310*9df19308SSuma Hegde 	if (ret)
311*9df19308SSuma Hegde 		return ret;
312*9df19308SSuma Hegde 
313*9df19308SSuma Hegde 	/* Check the response value */
314*9df19308SSuma Hegde 	if (msg.args[0] != (value + 1)) {
315*9df19308SSuma Hegde 		dev_err(plat_dev.sock[sock_ind].dev,
316*9df19308SSuma Hegde 			"Socket %d test message failed, Expected 0x%08X, received 0x%08X\n",
317*9df19308SSuma Hegde 			sock_ind, (value + 1), msg.args[0]);
318*9df19308SSuma Hegde 		return -EBADE;
319*9df19308SSuma Hegde 	}
320*9df19308SSuma Hegde 
321*9df19308SSuma Hegde 	return ret;
322*9df19308SSuma Hegde }
323*9df19308SSuma Hegde 
324*9df19308SSuma Hegde static long hsmp_ioctl(struct file *fp, unsigned int cmd, unsigned long arg)
325*9df19308SSuma Hegde {
326*9df19308SSuma Hegde 	int __user *arguser = (int  __user *)arg;
327*9df19308SSuma Hegde 	struct hsmp_message msg = { 0 };
328*9df19308SSuma Hegde 	int ret;
329*9df19308SSuma Hegde 
330*9df19308SSuma Hegde 	if (copy_struct_from_user(&msg, sizeof(msg), arguser, sizeof(struct hsmp_message)))
331*9df19308SSuma Hegde 		return -EFAULT;
332*9df19308SSuma Hegde 
333*9df19308SSuma Hegde 	/*
334*9df19308SSuma Hegde 	 * Check msg_id is within the range of supported msg ids
335*9df19308SSuma Hegde 	 * i.e within the array bounds of hsmp_msg_desc_table
336*9df19308SSuma Hegde 	 */
337*9df19308SSuma Hegde 	if (msg.msg_id < HSMP_TEST || msg.msg_id >= HSMP_MSG_ID_MAX)
338*9df19308SSuma Hegde 		return -ENOMSG;
339*9df19308SSuma Hegde 
340*9df19308SSuma Hegde 	switch (fp->f_mode & (FMODE_WRITE | FMODE_READ)) {
341*9df19308SSuma Hegde 	case FMODE_WRITE:
342*9df19308SSuma Hegde 		/*
343*9df19308SSuma Hegde 		 * Device is opened in O_WRONLY mode
344*9df19308SSuma Hegde 		 * Execute only set/configure commands
345*9df19308SSuma Hegde 		 */
346*9df19308SSuma Hegde 		if (hsmp_msg_desc_table[msg.msg_id].type != HSMP_SET)
347*9df19308SSuma Hegde 			return -EINVAL;
348*9df19308SSuma Hegde 		break;
349*9df19308SSuma Hegde 	case FMODE_READ:
350*9df19308SSuma Hegde 		/*
351*9df19308SSuma Hegde 		 * Device is opened in O_RDONLY mode
352*9df19308SSuma Hegde 		 * Execute only get/monitor commands
353*9df19308SSuma Hegde 		 */
354*9df19308SSuma Hegde 		if (hsmp_msg_desc_table[msg.msg_id].type != HSMP_GET)
355*9df19308SSuma Hegde 			return -EINVAL;
356*9df19308SSuma Hegde 		break;
357*9df19308SSuma Hegde 	case FMODE_READ | FMODE_WRITE:
358*9df19308SSuma Hegde 		/*
359*9df19308SSuma Hegde 		 * Device is opened in O_RDWR mode
360*9df19308SSuma Hegde 		 * Execute both get/monitor and set/configure commands
361*9df19308SSuma Hegde 		 */
362*9df19308SSuma Hegde 		break;
363*9df19308SSuma Hegde 	default:
364*9df19308SSuma Hegde 		return -EINVAL;
365*9df19308SSuma Hegde 	}
366*9df19308SSuma Hegde 
367*9df19308SSuma Hegde 	ret = hsmp_send_message(&msg);
368*9df19308SSuma Hegde 	if (ret)
369*9df19308SSuma Hegde 		return ret;
370*9df19308SSuma Hegde 
371*9df19308SSuma Hegde 	if (hsmp_msg_desc_table[msg.msg_id].response_sz > 0) {
372*9df19308SSuma Hegde 		/* Copy results back to user for get/monitor commands */
373*9df19308SSuma Hegde 		if (copy_to_user(arguser, &msg, sizeof(struct hsmp_message)))
374*9df19308SSuma Hegde 			return -EFAULT;
375*9df19308SSuma Hegde 	}
376*9df19308SSuma Hegde 
377*9df19308SSuma Hegde 	return 0;
378*9df19308SSuma Hegde }
379*9df19308SSuma Hegde 
380*9df19308SSuma Hegde static const struct file_operations hsmp_fops = {
381*9df19308SSuma Hegde 	.owner		= THIS_MODULE,
382*9df19308SSuma Hegde 	.unlocked_ioctl	= hsmp_ioctl,
383*9df19308SSuma Hegde 	.compat_ioctl	= hsmp_ioctl,
384*9df19308SSuma Hegde };
385*9df19308SSuma Hegde 
386*9df19308SSuma Hegde /* This is the UUID used for HSMP */
387*9df19308SSuma Hegde static const guid_t acpi_hsmp_uuid = GUID_INIT(0xb74d619d, 0x5707, 0x48bd,
388*9df19308SSuma Hegde 						0xa6, 0x9f, 0x4e, 0xa2,
389*9df19308SSuma Hegde 						0x87, 0x1f, 0xc2, 0xf6);
390*9df19308SSuma Hegde 
391*9df19308SSuma Hegde static inline bool is_acpi_hsmp_uuid(union acpi_object *obj)
392*9df19308SSuma Hegde {
393*9df19308SSuma Hegde 	if (obj->type == ACPI_TYPE_BUFFER && obj->buffer.length == UUID_SIZE)
394*9df19308SSuma Hegde 		return guid_equal((guid_t *)obj->buffer.pointer, &acpi_hsmp_uuid);
395*9df19308SSuma Hegde 
396*9df19308SSuma Hegde 	return false;
397*9df19308SSuma Hegde }
398*9df19308SSuma Hegde 
399*9df19308SSuma Hegde static inline int hsmp_get_uid(struct device *dev, u16 *sock_ind)
400*9df19308SSuma Hegde {
401*9df19308SSuma Hegde 	char *uid;
402*9df19308SSuma Hegde 
403*9df19308SSuma Hegde 	/*
404*9df19308SSuma Hegde 	 * UID (ID00, ID01..IDXX) is used for differentiating sockets,
405*9df19308SSuma Hegde 	 * read it and strip the "ID" part of it and convert the remaining
406*9df19308SSuma Hegde 	 * bytes to integer.
407*9df19308SSuma Hegde 	 */
408*9df19308SSuma Hegde 	uid = acpi_device_uid(ACPI_COMPANION(dev));
409*9df19308SSuma Hegde 
410*9df19308SSuma Hegde 	return kstrtou16(uid + 2, 10, sock_ind);
411*9df19308SSuma Hegde }
412*9df19308SSuma Hegde 
413*9df19308SSuma Hegde static acpi_status hsmp_resource(struct acpi_resource *res, void *data)
414*9df19308SSuma Hegde {
415*9df19308SSuma Hegde 	struct hsmp_socket *sock = data;
416*9df19308SSuma Hegde 	struct resource r;
417*9df19308SSuma Hegde 
418*9df19308SSuma Hegde 	switch (res->type) {
419*9df19308SSuma Hegde 	case ACPI_RESOURCE_TYPE_FIXED_MEMORY32:
420*9df19308SSuma Hegde 		if (!acpi_dev_resource_memory(res, &r))
421*9df19308SSuma Hegde 			return AE_ERROR;
422*9df19308SSuma Hegde 		if (!r.start || r.end < r.start || !(r.flags & IORESOURCE_MEM_WRITEABLE))
423*9df19308SSuma Hegde 			return AE_ERROR;
424*9df19308SSuma Hegde 		sock->mbinfo.base_addr = r.start;
425*9df19308SSuma Hegde 		sock->mbinfo.size = resource_size(&r);
426*9df19308SSuma Hegde 		break;
427*9df19308SSuma Hegde 	case ACPI_RESOURCE_TYPE_END_TAG:
428*9df19308SSuma Hegde 		break;
429*9df19308SSuma Hegde 	default:
430*9df19308SSuma Hegde 		return AE_ERROR;
431*9df19308SSuma Hegde 	}
432*9df19308SSuma Hegde 
433*9df19308SSuma Hegde 	return AE_OK;
434*9df19308SSuma Hegde }
435*9df19308SSuma Hegde 
436*9df19308SSuma Hegde static int hsmp_read_acpi_dsd(struct hsmp_socket *sock)
437*9df19308SSuma Hegde {
438*9df19308SSuma Hegde 	struct acpi_buffer buf = { ACPI_ALLOCATE_BUFFER, NULL };
439*9df19308SSuma Hegde 	union acpi_object *guid, *mailbox_package;
440*9df19308SSuma Hegde 	union acpi_object *dsd;
441*9df19308SSuma Hegde 	acpi_status status;
442*9df19308SSuma Hegde 	int ret = 0;
443*9df19308SSuma Hegde 	int j;
444*9df19308SSuma Hegde 
445*9df19308SSuma Hegde 	status = acpi_evaluate_object_typed(ACPI_HANDLE(sock->dev), "_DSD", NULL,
446*9df19308SSuma Hegde 					    &buf, ACPI_TYPE_PACKAGE);
447*9df19308SSuma Hegde 	if (ACPI_FAILURE(status)) {
448*9df19308SSuma Hegde 		dev_err(sock->dev, "Failed to read mailbox reg offsets from DSD table, err: %s\n",
449*9df19308SSuma Hegde 			acpi_format_exception(status));
450*9df19308SSuma Hegde 		return -ENODEV;
451*9df19308SSuma Hegde 	}
452*9df19308SSuma Hegde 
453*9df19308SSuma Hegde 	dsd = buf.pointer;
454*9df19308SSuma Hegde 
455*9df19308SSuma Hegde 	/* HSMP _DSD property should contain 2 objects.
456*9df19308SSuma Hegde 	 * 1. guid which is an acpi object of type ACPI_TYPE_BUFFER
457*9df19308SSuma Hegde 	 * 2. mailbox which is an acpi object of type ACPI_TYPE_PACKAGE
458*9df19308SSuma Hegde 	 *    This mailbox object contains 3 more acpi objects of type
459*9df19308SSuma Hegde 	 *    ACPI_TYPE_PACKAGE for holding msgid, msgresp, msgarg offsets
460*9df19308SSuma Hegde 	 *    these packages inturn contain 2 acpi objects of type
461*9df19308SSuma Hegde 	 *    ACPI_TYPE_STRING and ACPI_TYPE_INTEGER
462*9df19308SSuma Hegde 	 */
463*9df19308SSuma Hegde 	if (!dsd || dsd->type != ACPI_TYPE_PACKAGE || dsd->package.count != 2) {
464*9df19308SSuma Hegde 		ret = -EINVAL;
465*9df19308SSuma Hegde 		goto free_buf;
466*9df19308SSuma Hegde 	}
467*9df19308SSuma Hegde 
468*9df19308SSuma Hegde 	guid = &dsd->package.elements[0];
469*9df19308SSuma Hegde 	mailbox_package = &dsd->package.elements[1];
470*9df19308SSuma Hegde 	if (!is_acpi_hsmp_uuid(guid) || mailbox_package->type != ACPI_TYPE_PACKAGE) {
471*9df19308SSuma Hegde 		dev_err(sock->dev, "Invalid hsmp _DSD table data\n");
472*9df19308SSuma Hegde 		ret = -EINVAL;
473*9df19308SSuma Hegde 		goto free_buf;
474*9df19308SSuma Hegde 	}
475*9df19308SSuma Hegde 
476*9df19308SSuma Hegde 	for (j = 0; j < mailbox_package->package.count; j++) {
477*9df19308SSuma Hegde 		union acpi_object *msgobj, *msgstr, *msgint;
478*9df19308SSuma Hegde 
479*9df19308SSuma Hegde 		msgobj	= &mailbox_package->package.elements[j];
480*9df19308SSuma Hegde 		msgstr	= &msgobj->package.elements[0];
481*9df19308SSuma Hegde 		msgint	= &msgobj->package.elements[1];
482*9df19308SSuma Hegde 
483*9df19308SSuma Hegde 		/* package should have 1 string and 1 integer object */
484*9df19308SSuma Hegde 		if (msgobj->type != ACPI_TYPE_PACKAGE ||
485*9df19308SSuma Hegde 		    msgstr->type != ACPI_TYPE_STRING ||
486*9df19308SSuma Hegde 		    msgint->type != ACPI_TYPE_INTEGER) {
487*9df19308SSuma Hegde 			ret = -EINVAL;
488*9df19308SSuma Hegde 			goto free_buf;
489*9df19308SSuma Hegde 		}
490*9df19308SSuma Hegde 
491*9df19308SSuma Hegde 		if (!strncmp(msgstr->string.pointer, MSG_IDOFF_STR,
492*9df19308SSuma Hegde 			     msgstr->string.length)) {
493*9df19308SSuma Hegde 			sock->mbinfo.msg_id_off = msgint->integer.value;
494*9df19308SSuma Hegde 		} else if (!strncmp(msgstr->string.pointer, MSG_RESPOFF_STR,
495*9df19308SSuma Hegde 				    msgstr->string.length)) {
496*9df19308SSuma Hegde 			sock->mbinfo.msg_resp_off =  msgint->integer.value;
497*9df19308SSuma Hegde 		} else if (!strncmp(msgstr->string.pointer, MSG_ARGOFF_STR,
498*9df19308SSuma Hegde 				    msgstr->string.length)) {
499*9df19308SSuma Hegde 			sock->mbinfo.msg_arg_off = msgint->integer.value;
500*9df19308SSuma Hegde 		} else {
501*9df19308SSuma Hegde 			ret = -ENOENT;
502*9df19308SSuma Hegde 			goto free_buf;
503*9df19308SSuma Hegde 		}
504*9df19308SSuma Hegde 	}
505*9df19308SSuma Hegde 
506*9df19308SSuma Hegde 	if (!sock->mbinfo.msg_id_off || !sock->mbinfo.msg_resp_off ||
507*9df19308SSuma Hegde 	    !sock->mbinfo.msg_arg_off)
508*9df19308SSuma Hegde 		ret = -EINVAL;
509*9df19308SSuma Hegde 
510*9df19308SSuma Hegde free_buf:
511*9df19308SSuma Hegde 	ACPI_FREE(buf.pointer);
512*9df19308SSuma Hegde 	return ret;
513*9df19308SSuma Hegde }
514*9df19308SSuma Hegde 
515*9df19308SSuma Hegde static int hsmp_read_acpi_crs(struct hsmp_socket *sock)
516*9df19308SSuma Hegde {
517*9df19308SSuma Hegde 	acpi_status status;
518*9df19308SSuma Hegde 
519*9df19308SSuma Hegde 	status = acpi_walk_resources(ACPI_HANDLE(sock->dev), METHOD_NAME__CRS,
520*9df19308SSuma Hegde 				     hsmp_resource, sock);
521*9df19308SSuma Hegde 	if (ACPI_FAILURE(status)) {
522*9df19308SSuma Hegde 		dev_err(sock->dev, "Failed to look up MP1 base address from CRS method, err: %s\n",
523*9df19308SSuma Hegde 			acpi_format_exception(status));
524*9df19308SSuma Hegde 		return -EINVAL;
525*9df19308SSuma Hegde 	}
526*9df19308SSuma Hegde 	if (!sock->mbinfo.base_addr || !sock->mbinfo.size)
527*9df19308SSuma Hegde 		return -EINVAL;
528*9df19308SSuma Hegde 
529*9df19308SSuma Hegde 	/* The mapped region should be un cached */
530*9df19308SSuma Hegde 	sock->virt_base_addr = devm_ioremap_uc(sock->dev, sock->mbinfo.base_addr,
531*9df19308SSuma Hegde 					       sock->mbinfo.size);
532*9df19308SSuma Hegde 	if (!sock->virt_base_addr) {
533*9df19308SSuma Hegde 		dev_err(sock->dev, "Failed to ioremap MP1 base address\n");
534*9df19308SSuma Hegde 		return -ENOMEM;
535*9df19308SSuma Hegde 	}
536*9df19308SSuma Hegde 
537*9df19308SSuma Hegde 	return 0;
538*9df19308SSuma Hegde }
539*9df19308SSuma Hegde 
540*9df19308SSuma Hegde /* Parse the ACPI table to read the data */
541*9df19308SSuma Hegde static int hsmp_parse_acpi_table(struct device *dev, u16 sock_ind)
542*9df19308SSuma Hegde {
543*9df19308SSuma Hegde 	struct hsmp_socket *sock = &plat_dev.sock[sock_ind];
544*9df19308SSuma Hegde 	int ret;
545*9df19308SSuma Hegde 
546*9df19308SSuma Hegde 	sock->sock_ind		= sock_ind;
547*9df19308SSuma Hegde 	sock->dev		= dev;
548*9df19308SSuma Hegde 	plat_dev.is_acpi_device	= true;
549*9df19308SSuma Hegde 
550*9df19308SSuma Hegde 	sema_init(&sock->hsmp_sem, 1);
551*9df19308SSuma Hegde 
552*9df19308SSuma Hegde 	/* Read MP1 base address from CRS method */
553*9df19308SSuma Hegde 	ret = hsmp_read_acpi_crs(sock);
554*9df19308SSuma Hegde 	if (ret)
555*9df19308SSuma Hegde 		return ret;
556*9df19308SSuma Hegde 
557*9df19308SSuma Hegde 	/* Read mailbox offsets from DSD table */
558*9df19308SSuma Hegde 	return hsmp_read_acpi_dsd(sock);
559*9df19308SSuma Hegde }
560*9df19308SSuma Hegde 
561*9df19308SSuma Hegde static ssize_t hsmp_metric_tbl_read(struct file *filp, struct kobject *kobj,
562*9df19308SSuma Hegde 				    struct bin_attribute *bin_attr, char *buf,
563*9df19308SSuma Hegde 				    loff_t off, size_t count)
564*9df19308SSuma Hegde {
565*9df19308SSuma Hegde 	struct hsmp_socket *sock = bin_attr->private;
566*9df19308SSuma Hegde 	struct hsmp_message msg = { 0 };
567*9df19308SSuma Hegde 	int ret;
568*9df19308SSuma Hegde 
569*9df19308SSuma Hegde 	if (!sock)
570*9df19308SSuma Hegde 		return -EINVAL;
571*9df19308SSuma Hegde 
572*9df19308SSuma Hegde 	/* Do not support lseek(), reads entire metric table */
573*9df19308SSuma Hegde 	if (count < bin_attr->size) {
574*9df19308SSuma Hegde 		dev_err(sock->dev, "Wrong buffer size\n");
575*9df19308SSuma Hegde 		return -EINVAL;
576*9df19308SSuma Hegde 	}
577*9df19308SSuma Hegde 
578*9df19308SSuma Hegde 	msg.msg_id	= HSMP_GET_METRIC_TABLE;
579*9df19308SSuma Hegde 	msg.sock_ind	= sock->sock_ind;
580*9df19308SSuma Hegde 
581*9df19308SSuma Hegde 	ret = hsmp_send_message(&msg);
582*9df19308SSuma Hegde 	if (ret)
583*9df19308SSuma Hegde 		return ret;
584*9df19308SSuma Hegde 	memcpy_fromio(buf, sock->metric_tbl_addr, bin_attr->size);
585*9df19308SSuma Hegde 
586*9df19308SSuma Hegde 	return bin_attr->size;
587*9df19308SSuma Hegde }
588*9df19308SSuma Hegde 
589*9df19308SSuma Hegde static int hsmp_get_tbl_dram_base(u16 sock_ind)
590*9df19308SSuma Hegde {
591*9df19308SSuma Hegde 	struct hsmp_socket *sock = &plat_dev.sock[sock_ind];
592*9df19308SSuma Hegde 	struct hsmp_message msg = { 0 };
593*9df19308SSuma Hegde 	phys_addr_t dram_addr;
594*9df19308SSuma Hegde 	int ret;
595*9df19308SSuma Hegde 
596*9df19308SSuma Hegde 	msg.sock_ind	= sock_ind;
597*9df19308SSuma Hegde 	msg.response_sz	= hsmp_msg_desc_table[HSMP_GET_METRIC_TABLE_DRAM_ADDR].response_sz;
598*9df19308SSuma Hegde 	msg.msg_id	= HSMP_GET_METRIC_TABLE_DRAM_ADDR;
599*9df19308SSuma Hegde 
600*9df19308SSuma Hegde 	ret = hsmp_send_message(&msg);
601*9df19308SSuma Hegde 	if (ret)
602*9df19308SSuma Hegde 		return ret;
603*9df19308SSuma Hegde 
604*9df19308SSuma Hegde 	/*
605*9df19308SSuma Hegde 	 * calculate the metric table DRAM address from lower and upper 32 bits
606*9df19308SSuma Hegde 	 * sent from SMU and ioremap it to virtual address.
607*9df19308SSuma Hegde 	 */
608*9df19308SSuma Hegde 	dram_addr = msg.args[0] | ((u64)(msg.args[1]) << 32);
609*9df19308SSuma Hegde 	if (!dram_addr) {
610*9df19308SSuma Hegde 		dev_err(sock->dev, "Invalid DRAM address for metric table\n");
611*9df19308SSuma Hegde 		return -ENOMEM;
612*9df19308SSuma Hegde 	}
613*9df19308SSuma Hegde 	sock->metric_tbl_addr = devm_ioremap(sock->dev, dram_addr,
614*9df19308SSuma Hegde 					     sizeof(struct hsmp_metric_table));
615*9df19308SSuma Hegde 	if (!sock->metric_tbl_addr) {
616*9df19308SSuma Hegde 		dev_err(sock->dev, "Failed to ioremap metric table addr\n");
617*9df19308SSuma Hegde 		return -ENOMEM;
618*9df19308SSuma Hegde 	}
619*9df19308SSuma Hegde 	return 0;
620*9df19308SSuma Hegde }
621*9df19308SSuma Hegde 
622*9df19308SSuma Hegde static umode_t hsmp_is_sock_attr_visible(struct kobject *kobj,
623*9df19308SSuma Hegde 					 struct bin_attribute *battr, int id)
624*9df19308SSuma Hegde {
625*9df19308SSuma Hegde 	if (plat_dev.proto_ver == HSMP_PROTO_VER6)
626*9df19308SSuma Hegde 		return battr->attr.mode;
627*9df19308SSuma Hegde 	else
628*9df19308SSuma Hegde 		return 0;
629*9df19308SSuma Hegde }
630*9df19308SSuma Hegde 
631*9df19308SSuma Hegde static int hsmp_init_metric_tbl_bin_attr(struct bin_attribute **hattrs, u16 sock_ind)
632*9df19308SSuma Hegde {
633*9df19308SSuma Hegde 	struct bin_attribute *hattr = &plat_dev.sock[sock_ind].hsmp_attr;
634*9df19308SSuma Hegde 
635*9df19308SSuma Hegde 	sysfs_bin_attr_init(hattr);
636*9df19308SSuma Hegde 	hattr->attr.name	= HSMP_METRICS_TABLE_NAME;
637*9df19308SSuma Hegde 	hattr->attr.mode	= 0444;
638*9df19308SSuma Hegde 	hattr->read		= hsmp_metric_tbl_read;
639*9df19308SSuma Hegde 	hattr->size		= sizeof(struct hsmp_metric_table);
640*9df19308SSuma Hegde 	hattr->private		= &plat_dev.sock[sock_ind];
641*9df19308SSuma Hegde 	hattrs[0]		= hattr;
642*9df19308SSuma Hegde 
643*9df19308SSuma Hegde 	if (plat_dev.proto_ver == HSMP_PROTO_VER6)
644*9df19308SSuma Hegde 		return hsmp_get_tbl_dram_base(sock_ind);
645*9df19308SSuma Hegde 	else
646*9df19308SSuma Hegde 		return 0;
647*9df19308SSuma Hegde }
648*9df19308SSuma Hegde 
649*9df19308SSuma Hegde /* One bin sysfs for metrics table */
650*9df19308SSuma Hegde #define NUM_HSMP_ATTRS		1
651*9df19308SSuma Hegde 
652*9df19308SSuma Hegde static int hsmp_create_attr_list(struct attribute_group *attr_grp,
653*9df19308SSuma Hegde 				 struct device *dev, u16 sock_ind)
654*9df19308SSuma Hegde {
655*9df19308SSuma Hegde 	struct bin_attribute **hsmp_bin_attrs;
656*9df19308SSuma Hegde 
657*9df19308SSuma Hegde 	/* Null terminated list of attributes */
658*9df19308SSuma Hegde 	hsmp_bin_attrs = devm_kcalloc(dev, NUM_HSMP_ATTRS + 1,
659*9df19308SSuma Hegde 				      sizeof(*hsmp_bin_attrs),
660*9df19308SSuma Hegde 				      GFP_KERNEL);
661*9df19308SSuma Hegde 	if (!hsmp_bin_attrs)
662*9df19308SSuma Hegde 		return -ENOMEM;
663*9df19308SSuma Hegde 
664*9df19308SSuma Hegde 	attr_grp->bin_attrs = hsmp_bin_attrs;
665*9df19308SSuma Hegde 
666*9df19308SSuma Hegde 	return hsmp_init_metric_tbl_bin_attr(hsmp_bin_attrs, sock_ind);
667*9df19308SSuma Hegde }
668*9df19308SSuma Hegde 
669*9df19308SSuma Hegde static int hsmp_create_non_acpi_sysfs_if(struct device *dev)
670*9df19308SSuma Hegde {
671*9df19308SSuma Hegde 	const struct attribute_group **hsmp_attr_grps;
672*9df19308SSuma Hegde 	struct attribute_group *attr_grp;
673*9df19308SSuma Hegde 	u16 i;
674*9df19308SSuma Hegde 
675*9df19308SSuma Hegde 	hsmp_attr_grps = devm_kcalloc(dev, plat_dev.num_sockets + 1,
676*9df19308SSuma Hegde 				      sizeof(*hsmp_attr_grps),
677*9df19308SSuma Hegde 				      GFP_KERNEL);
678*9df19308SSuma Hegde 	if (!hsmp_attr_grps)
679*9df19308SSuma Hegde 		return -ENOMEM;
680*9df19308SSuma Hegde 
681*9df19308SSuma Hegde 	/* Create a sysfs directory for each socket */
682*9df19308SSuma Hegde 	for (i = 0; i < plat_dev.num_sockets; i++) {
683*9df19308SSuma Hegde 		attr_grp = devm_kzalloc(dev, sizeof(struct attribute_group),
684*9df19308SSuma Hegde 					GFP_KERNEL);
685*9df19308SSuma Hegde 		if (!attr_grp)
686*9df19308SSuma Hegde 			return -ENOMEM;
687*9df19308SSuma Hegde 
688*9df19308SSuma Hegde 		snprintf(plat_dev.sock[i].name, HSMP_ATTR_GRP_NAME_SIZE, "socket%u", (u8)i);
689*9df19308SSuma Hegde 		attr_grp->name			= plat_dev.sock[i].name;
690*9df19308SSuma Hegde 		attr_grp->is_bin_visible	= hsmp_is_sock_attr_visible;
691*9df19308SSuma Hegde 		hsmp_attr_grps[i]		= attr_grp;
692*9df19308SSuma Hegde 
693*9df19308SSuma Hegde 		hsmp_create_attr_list(attr_grp, dev, i);
694*9df19308SSuma Hegde 	}
695*9df19308SSuma Hegde 
696*9df19308SSuma Hegde 	return device_add_groups(dev, hsmp_attr_grps);
697*9df19308SSuma Hegde }
698*9df19308SSuma Hegde 
699*9df19308SSuma Hegde static int hsmp_create_acpi_sysfs_if(struct device *dev)
700*9df19308SSuma Hegde {
701*9df19308SSuma Hegde 	struct attribute_group *attr_grp;
702*9df19308SSuma Hegde 	u16 sock_ind;
703*9df19308SSuma Hegde 	int ret;
704*9df19308SSuma Hegde 
705*9df19308SSuma Hegde 	attr_grp = devm_kzalloc(dev, sizeof(struct attribute_group), GFP_KERNEL);
706*9df19308SSuma Hegde 	if (!attr_grp)
707*9df19308SSuma Hegde 		return -ENOMEM;
708*9df19308SSuma Hegde 
709*9df19308SSuma Hegde 	attr_grp->is_bin_visible = hsmp_is_sock_attr_visible;
710*9df19308SSuma Hegde 
711*9df19308SSuma Hegde 	ret = hsmp_get_uid(dev, &sock_ind);
712*9df19308SSuma Hegde 	if (ret)
713*9df19308SSuma Hegde 		return ret;
714*9df19308SSuma Hegde 
715*9df19308SSuma Hegde 	ret = hsmp_create_attr_list(attr_grp, dev, sock_ind);
716*9df19308SSuma Hegde 	if (ret)
717*9df19308SSuma Hegde 		return ret;
718*9df19308SSuma Hegde 
719*9df19308SSuma Hegde 	return devm_device_add_group(dev, attr_grp);
720*9df19308SSuma Hegde }
721*9df19308SSuma Hegde 
722*9df19308SSuma Hegde static int hsmp_cache_proto_ver(u16 sock_ind)
723*9df19308SSuma Hegde {
724*9df19308SSuma Hegde 	struct hsmp_message msg = { 0 };
725*9df19308SSuma Hegde 	int ret;
726*9df19308SSuma Hegde 
727*9df19308SSuma Hegde 	msg.msg_id	= HSMP_GET_PROTO_VER;
728*9df19308SSuma Hegde 	msg.sock_ind	= sock_ind;
729*9df19308SSuma Hegde 	msg.response_sz = hsmp_msg_desc_table[HSMP_GET_PROTO_VER].response_sz;
730*9df19308SSuma Hegde 
731*9df19308SSuma Hegde 	ret = hsmp_send_message(&msg);
732*9df19308SSuma Hegde 	if (!ret)
733*9df19308SSuma Hegde 		plat_dev.proto_ver = msg.args[0];
734*9df19308SSuma Hegde 
735*9df19308SSuma Hegde 	return ret;
736*9df19308SSuma Hegde }
737*9df19308SSuma Hegde 
738*9df19308SSuma Hegde static inline bool is_f1a_m0h(void)
739*9df19308SSuma Hegde {
740*9df19308SSuma Hegde 	if (boot_cpu_data.x86 == 0x1A && boot_cpu_data.x86_model <= 0x0F)
741*9df19308SSuma Hegde 		return true;
742*9df19308SSuma Hegde 
743*9df19308SSuma Hegde 	return false;
744*9df19308SSuma Hegde }
745*9df19308SSuma Hegde 
746*9df19308SSuma Hegde static int init_platform_device(struct device *dev)
747*9df19308SSuma Hegde {
748*9df19308SSuma Hegde 	struct hsmp_socket *sock;
749*9df19308SSuma Hegde 	int ret, i;
750*9df19308SSuma Hegde 
751*9df19308SSuma Hegde 	for (i = 0; i < plat_dev.num_sockets; i++) {
752*9df19308SSuma Hegde 		if (!node_to_amd_nb(i))
753*9df19308SSuma Hegde 			return -ENODEV;
754*9df19308SSuma Hegde 		sock = &plat_dev.sock[i];
755*9df19308SSuma Hegde 		sock->root			= node_to_amd_nb(i)->root;
756*9df19308SSuma Hegde 		sock->sock_ind			= i;
757*9df19308SSuma Hegde 		sock->dev			= dev;
758*9df19308SSuma Hegde 		sock->mbinfo.base_addr		= SMN_HSMP_BASE;
759*9df19308SSuma Hegde 
760*9df19308SSuma Hegde 		/*
761*9df19308SSuma Hegde 		 * This is a transitional change from non-ACPI to ACPI, only
762*9df19308SSuma Hegde 		 * family 0x1A, model 0x00 platform is supported for both ACPI and non-ACPI.
763*9df19308SSuma Hegde 		 */
764*9df19308SSuma Hegde 		if (is_f1a_m0h())
765*9df19308SSuma Hegde 			sock->mbinfo.msg_id_off	= SMN_HSMP_MSG_ID_F1A_M0H;
766*9df19308SSuma Hegde 		else
767*9df19308SSuma Hegde 			sock->mbinfo.msg_id_off	= SMN_HSMP_MSG_ID;
768*9df19308SSuma Hegde 
769*9df19308SSuma Hegde 		sock->mbinfo.msg_resp_off	= SMN_HSMP_MSG_RESP;
770*9df19308SSuma Hegde 		sock->mbinfo.msg_arg_off	= SMN_HSMP_MSG_DATA;
771*9df19308SSuma Hegde 		sema_init(&sock->hsmp_sem, 1);
772*9df19308SSuma Hegde 
773*9df19308SSuma Hegde 		/* Test the hsmp interface on each socket */
774*9df19308SSuma Hegde 		ret = hsmp_test(i, 0xDEADBEEF);
775*9df19308SSuma Hegde 		if (ret) {
776*9df19308SSuma Hegde 			dev_err(dev, "HSMP test message failed on Fam:%x model:%x\n",
777*9df19308SSuma Hegde 				boot_cpu_data.x86, boot_cpu_data.x86_model);
778*9df19308SSuma Hegde 			dev_err(dev, "Is HSMP disabled in BIOS ?\n");
779*9df19308SSuma Hegde 			return ret;
780*9df19308SSuma Hegde 		}
781*9df19308SSuma Hegde 	}
782*9df19308SSuma Hegde 
783*9df19308SSuma Hegde 	return 0;
784*9df19308SSuma Hegde }
785*9df19308SSuma Hegde 
786*9df19308SSuma Hegde static const struct acpi_device_id amd_hsmp_acpi_ids[] = {
787*9df19308SSuma Hegde 	{ACPI_HSMP_DEVICE_HID, 0},
788*9df19308SSuma Hegde 	{}
789*9df19308SSuma Hegde };
790*9df19308SSuma Hegde MODULE_DEVICE_TABLE(acpi, amd_hsmp_acpi_ids);
791*9df19308SSuma Hegde 
792*9df19308SSuma Hegde static int hsmp_pltdrv_probe(struct platform_device *pdev)
793*9df19308SSuma Hegde {
794*9df19308SSuma Hegde 	struct acpi_device *adev;
795*9df19308SSuma Hegde 	u16 sock_ind = 0;
796*9df19308SSuma Hegde 	int ret;
797*9df19308SSuma Hegde 
798*9df19308SSuma Hegde 	/*
799*9df19308SSuma Hegde 	 * On ACPI supported BIOS, there is an ACPI HSMP device added for
800*9df19308SSuma Hegde 	 * each socket, so the per socket probing, but the memory allocated for
801*9df19308SSuma Hegde 	 * sockets should be contiguous to access it as an array,
802*9df19308SSuma Hegde 	 * Hence allocate memory for all the sockets at once instead of allocating
803*9df19308SSuma Hegde 	 * on each probe.
804*9df19308SSuma Hegde 	 */
805*9df19308SSuma Hegde 	if (!plat_dev.is_probed) {
806*9df19308SSuma Hegde 		plat_dev.sock = devm_kcalloc(&pdev->dev, plat_dev.num_sockets,
807*9df19308SSuma Hegde 					     sizeof(*plat_dev.sock),
808*9df19308SSuma Hegde 					     GFP_KERNEL);
809*9df19308SSuma Hegde 		if (!plat_dev.sock)
810*9df19308SSuma Hegde 			return -ENOMEM;
811*9df19308SSuma Hegde 	}
812*9df19308SSuma Hegde 	adev = ACPI_COMPANION(&pdev->dev);
813*9df19308SSuma Hegde 	if (adev && !acpi_match_device_ids(adev, amd_hsmp_acpi_ids)) {
814*9df19308SSuma Hegde 		ret = hsmp_get_uid(&pdev->dev, &sock_ind);
815*9df19308SSuma Hegde 		if (ret)
816*9df19308SSuma Hegde 			return ret;
817*9df19308SSuma Hegde 		if (sock_ind >= plat_dev.num_sockets)
818*9df19308SSuma Hegde 			return -EINVAL;
819*9df19308SSuma Hegde 		ret = hsmp_parse_acpi_table(&pdev->dev, sock_ind);
820*9df19308SSuma Hegde 		if (ret) {
821*9df19308SSuma Hegde 			dev_err(&pdev->dev, "Failed to parse ACPI table\n");
822*9df19308SSuma Hegde 			return ret;
823*9df19308SSuma Hegde 		}
824*9df19308SSuma Hegde 		/* Test the hsmp interface */
825*9df19308SSuma Hegde 		ret = hsmp_test(sock_ind, 0xDEADBEEF);
826*9df19308SSuma Hegde 		if (ret) {
827*9df19308SSuma Hegde 			dev_err(&pdev->dev, "HSMP test message failed on Fam:%x model:%x\n",
828*9df19308SSuma Hegde 				boot_cpu_data.x86, boot_cpu_data.x86_model);
829*9df19308SSuma Hegde 			dev_err(&pdev->dev, "Is HSMP disabled in BIOS ?\n");
830*9df19308SSuma Hegde 			return ret;
831*9df19308SSuma Hegde 		}
832*9df19308SSuma Hegde 	} else {
833*9df19308SSuma Hegde 		ret = init_platform_device(&pdev->dev);
834*9df19308SSuma Hegde 		if (ret) {
835*9df19308SSuma Hegde 			dev_err(&pdev->dev, "Failed to init HSMP mailbox\n");
836*9df19308SSuma Hegde 			return ret;
837*9df19308SSuma Hegde 		}
838*9df19308SSuma Hegde 	}
839*9df19308SSuma Hegde 
840*9df19308SSuma Hegde 	ret = hsmp_cache_proto_ver(sock_ind);
841*9df19308SSuma Hegde 	if (ret) {
842*9df19308SSuma Hegde 		dev_err(&pdev->dev, "Failed to read HSMP protocol version\n");
843*9df19308SSuma Hegde 		return ret;
844*9df19308SSuma Hegde 	}
845*9df19308SSuma Hegde 
846*9df19308SSuma Hegde 	if (plat_dev.is_acpi_device)
847*9df19308SSuma Hegde 		ret = hsmp_create_acpi_sysfs_if(&pdev->dev);
848*9df19308SSuma Hegde 	else
849*9df19308SSuma Hegde 		ret = hsmp_create_non_acpi_sysfs_if(&pdev->dev);
850*9df19308SSuma Hegde 	if (ret)
851*9df19308SSuma Hegde 		dev_err(&pdev->dev, "Failed to create HSMP sysfs interface\n");
852*9df19308SSuma Hegde 
853*9df19308SSuma Hegde 	if (!plat_dev.is_probed) {
854*9df19308SSuma Hegde 		plat_dev.hsmp_device.name	= HSMP_CDEV_NAME;
855*9df19308SSuma Hegde 		plat_dev.hsmp_device.minor	= MISC_DYNAMIC_MINOR;
856*9df19308SSuma Hegde 		plat_dev.hsmp_device.fops	= &hsmp_fops;
857*9df19308SSuma Hegde 		plat_dev.hsmp_device.parent	= &pdev->dev;
858*9df19308SSuma Hegde 		plat_dev.hsmp_device.nodename	= HSMP_DEVNODE_NAME;
859*9df19308SSuma Hegde 		plat_dev.hsmp_device.mode	= 0644;
860*9df19308SSuma Hegde 
861*9df19308SSuma Hegde 		ret = misc_register(&plat_dev.hsmp_device);
862*9df19308SSuma Hegde 		if (ret)
863*9df19308SSuma Hegde 			return ret;
864*9df19308SSuma Hegde 
865*9df19308SSuma Hegde 		plat_dev.is_probed = true;
866*9df19308SSuma Hegde 	}
867*9df19308SSuma Hegde 
868*9df19308SSuma Hegde 	return 0;
869*9df19308SSuma Hegde 
870*9df19308SSuma Hegde }
871*9df19308SSuma Hegde 
872*9df19308SSuma Hegde static void hsmp_pltdrv_remove(struct platform_device *pdev)
873*9df19308SSuma Hegde {
874*9df19308SSuma Hegde 	/*
875*9df19308SSuma Hegde 	 * We register only one misc_device even on multi socket system.
876*9df19308SSuma Hegde 	 * So, deregister should happen only once.
877*9df19308SSuma Hegde 	 */
878*9df19308SSuma Hegde 	if (plat_dev.is_probed) {
879*9df19308SSuma Hegde 		misc_deregister(&plat_dev.hsmp_device);
880*9df19308SSuma Hegde 		plat_dev.is_probed = false;
881*9df19308SSuma Hegde 	}
882*9df19308SSuma Hegde }
883*9df19308SSuma Hegde 
884*9df19308SSuma Hegde static struct platform_driver amd_hsmp_driver = {
885*9df19308SSuma Hegde 	.probe		= hsmp_pltdrv_probe,
886*9df19308SSuma Hegde 	.remove		= hsmp_pltdrv_remove,
887*9df19308SSuma Hegde 	.driver		= {
888*9df19308SSuma Hegde 		.name	= DRIVER_NAME,
889*9df19308SSuma Hegde 		.acpi_match_table = amd_hsmp_acpi_ids,
890*9df19308SSuma Hegde 	},
891*9df19308SSuma Hegde };
892*9df19308SSuma Hegde 
893*9df19308SSuma Hegde static struct platform_device *amd_hsmp_platdev;
894*9df19308SSuma Hegde 
895*9df19308SSuma Hegde static int hsmp_plat_dev_register(void)
896*9df19308SSuma Hegde {
897*9df19308SSuma Hegde 	int ret;
898*9df19308SSuma Hegde 
899*9df19308SSuma Hegde 	amd_hsmp_platdev = platform_device_alloc(DRIVER_NAME, PLATFORM_DEVID_NONE);
900*9df19308SSuma Hegde 	if (!amd_hsmp_platdev)
901*9df19308SSuma Hegde 		return -ENOMEM;
902*9df19308SSuma Hegde 
903*9df19308SSuma Hegde 	ret = platform_device_add(amd_hsmp_platdev);
904*9df19308SSuma Hegde 	if (ret)
905*9df19308SSuma Hegde 		platform_device_put(amd_hsmp_platdev);
906*9df19308SSuma Hegde 
907*9df19308SSuma Hegde 	return ret;
908*9df19308SSuma Hegde }
909*9df19308SSuma Hegde 
910*9df19308SSuma Hegde /*
911*9df19308SSuma Hegde  * This check is only needed for backward compatibility of previous platforms.
912*9df19308SSuma Hegde  * All new platforms are expected to support ACPI based probing.
913*9df19308SSuma Hegde  */
914*9df19308SSuma Hegde static bool legacy_hsmp_support(void)
915*9df19308SSuma Hegde {
916*9df19308SSuma Hegde 	if (boot_cpu_data.x86_vendor != X86_VENDOR_AMD)
917*9df19308SSuma Hegde 		return false;
918*9df19308SSuma Hegde 
919*9df19308SSuma Hegde 	switch (boot_cpu_data.x86) {
920*9df19308SSuma Hegde 	case 0x19:
921*9df19308SSuma Hegde 		switch (boot_cpu_data.x86_model) {
922*9df19308SSuma Hegde 		case 0x00 ... 0x1F:
923*9df19308SSuma Hegde 		case 0x30 ... 0x3F:
924*9df19308SSuma Hegde 		case 0x90 ... 0x9F:
925*9df19308SSuma Hegde 		case 0xA0 ... 0xAF:
926*9df19308SSuma Hegde 			return true;
927*9df19308SSuma Hegde 		default:
928*9df19308SSuma Hegde 			return false;
929*9df19308SSuma Hegde 		}
930*9df19308SSuma Hegde 	case 0x1A:
931*9df19308SSuma Hegde 		switch (boot_cpu_data.x86_model) {
932*9df19308SSuma Hegde 		case 0x00 ... 0x1F:
933*9df19308SSuma Hegde 			return true;
934*9df19308SSuma Hegde 		default:
935*9df19308SSuma Hegde 			return false;
936*9df19308SSuma Hegde 		}
937*9df19308SSuma Hegde 	default:
938*9df19308SSuma Hegde 		return false;
939*9df19308SSuma Hegde 	}
940*9df19308SSuma Hegde 
941*9df19308SSuma Hegde 	return false;
942*9df19308SSuma Hegde }
943*9df19308SSuma Hegde 
944*9df19308SSuma Hegde static int __init hsmp_plt_init(void)
945*9df19308SSuma Hegde {
946*9df19308SSuma Hegde 	int ret = -ENODEV;
947*9df19308SSuma Hegde 
948*9df19308SSuma Hegde 	/*
949*9df19308SSuma Hegde 	 * amd_nb_num() returns number of SMN/DF interfaces present in the system
950*9df19308SSuma Hegde 	 * if we have N SMN/DF interfaces that ideally means N sockets
951*9df19308SSuma Hegde 	 */
952*9df19308SSuma Hegde 	plat_dev.num_sockets = amd_nb_num();
953*9df19308SSuma Hegde 	if (plat_dev.num_sockets == 0 || plat_dev.num_sockets > MAX_AMD_SOCKETS)
954*9df19308SSuma Hegde 		return ret;
955*9df19308SSuma Hegde 
956*9df19308SSuma Hegde 	ret = platform_driver_register(&amd_hsmp_driver);
957*9df19308SSuma Hegde 	if (ret)
958*9df19308SSuma Hegde 		return ret;
959*9df19308SSuma Hegde 
960*9df19308SSuma Hegde 	if (!plat_dev.is_acpi_device) {
961*9df19308SSuma Hegde 		if (legacy_hsmp_support()) {
962*9df19308SSuma Hegde 			/* Not ACPI device, but supports HSMP, register a plat_dev */
963*9df19308SSuma Hegde 			ret = hsmp_plat_dev_register();
964*9df19308SSuma Hegde 		} else {
965*9df19308SSuma Hegde 			/* Not ACPI, Does not support HSMP */
966*9df19308SSuma Hegde 			pr_info("HSMP is not supported on Family:%x model:%x\n",
967*9df19308SSuma Hegde 				boot_cpu_data.x86, boot_cpu_data.x86_model);
968*9df19308SSuma Hegde 			ret = -ENODEV;
969*9df19308SSuma Hegde 		}
970*9df19308SSuma Hegde 		if (ret)
971*9df19308SSuma Hegde 			platform_driver_unregister(&amd_hsmp_driver);
972*9df19308SSuma Hegde 	}
973*9df19308SSuma Hegde 
974*9df19308SSuma Hegde 	return ret;
975*9df19308SSuma Hegde }
976*9df19308SSuma Hegde 
977*9df19308SSuma Hegde static void __exit hsmp_plt_exit(void)
978*9df19308SSuma Hegde {
979*9df19308SSuma Hegde 	platform_device_unregister(amd_hsmp_platdev);
980*9df19308SSuma Hegde 	platform_driver_unregister(&amd_hsmp_driver);
981*9df19308SSuma Hegde }
982*9df19308SSuma Hegde 
983*9df19308SSuma Hegde device_initcall(hsmp_plt_init);
984*9df19308SSuma Hegde module_exit(hsmp_plt_exit);
985*9df19308SSuma Hegde 
986*9df19308SSuma Hegde MODULE_DESCRIPTION("AMD HSMP Platform Interface Driver");
987*9df19308SSuma Hegde MODULE_VERSION(DRIVER_VERSION);
988*9df19308SSuma Hegde MODULE_LICENSE("GPL v2");
989