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