linux/drivers/mfd/intel_pmc_bxt.c
<<
>>
Prefs
   1// SPDX-License-Identifier: GPL-2.0
   2/*
   3 * Driver for the Intel Broxton PMC
   4 *
   5 * (C) Copyright 2014 - 2020 Intel Corporation
   6 *
   7 * This driver is based on Intel SCU IPC driver (intel_scu_ipc.c) by
   8 * Sreedhara DS <sreedhara.ds@intel.com>
   9 *
  10 * The PMC (Power Management Controller) running on the ARC processor
  11 * communicates with another entity running in the IA (Intel Architecture)
  12 * core through an IPC (Intel Processor Communications) mechanism which in
  13 * turn sends messages between the IA and the PMC.
  14 */
  15
  16#include <linux/acpi.h>
  17#include <linux/delay.h>
  18#include <linux/errno.h>
  19#include <linux/interrupt.h>
  20#include <linux/io-64-nonatomic-lo-hi.h>
  21#include <linux/mfd/core.h>
  22#include <linux/mfd/intel_pmc_bxt.h>
  23#include <linux/module.h>
  24#include <linux/platform_device.h>
  25#include <linux/platform_data/itco_wdt.h>
  26
  27#include <asm/intel_scu_ipc.h>
  28
  29/* Residency with clock rate at 19.2MHz to usecs */
  30#define S0IX_RESIDENCY_IN_USECS(d, s)           \
  31({                                              \
  32        u64 result = 10ull * ((d) + (s));       \
  33        do_div(result, 192);                    \
  34        result;                                 \
  35})
  36
  37/* Resources exported from IFWI */
  38#define PLAT_RESOURCE_IPC_INDEX         0
  39#define PLAT_RESOURCE_IPC_SIZE          0x1000
  40#define PLAT_RESOURCE_GCR_OFFSET        0x1000
  41#define PLAT_RESOURCE_GCR_SIZE          0x1000
  42#define PLAT_RESOURCE_BIOS_DATA_INDEX   1
  43#define PLAT_RESOURCE_BIOS_IFACE_INDEX  2
  44#define PLAT_RESOURCE_TELEM_SSRAM_INDEX 3
  45#define PLAT_RESOURCE_ISP_DATA_INDEX    4
  46#define PLAT_RESOURCE_ISP_IFACE_INDEX   5
  47#define PLAT_RESOURCE_GTD_DATA_INDEX    6
  48#define PLAT_RESOURCE_GTD_IFACE_INDEX   7
  49#define PLAT_RESOURCE_ACPI_IO_INDEX     0
  50
  51/*
  52 * BIOS does not create an ACPI device for each PMC function, but
  53 * exports multiple resources from one ACPI device (IPC) for multiple
  54 * functions. This driver is responsible for creating a child device and
  55 * to export resources for those functions.
  56 */
  57#define SMI_EN_OFFSET                   0x0040
  58#define SMI_EN_SIZE                     4
  59#define TCO_BASE_OFFSET                 0x0060
  60#define TCO_REGS_SIZE                   16
  61#define TELEM_SSRAM_SIZE                240
  62#define TELEM_PMC_SSRAM_OFFSET          0x1b00
  63#define TELEM_PUNIT_SSRAM_OFFSET        0x1a00
  64
  65/* Commands */
  66#define PMC_NORTHPEAK_CTRL              0xed
  67
  68static inline bool is_gcr_valid(u32 offset)
  69{
  70        return offset < PLAT_RESOURCE_GCR_SIZE - 8;
  71}
  72
  73/**
  74 * intel_pmc_gcr_read64() - Read a 64-bit PMC GCR register
  75 * @pmc: PMC device pointer
  76 * @offset: offset of GCR register from GCR address base
  77 * @data: data pointer for storing the register output
  78 *
  79 * Reads the 64-bit PMC GCR register at given offset.
  80 *
  81 * Return: Negative value on error or 0 on success.
  82 */
  83int intel_pmc_gcr_read64(struct intel_pmc_dev *pmc, u32 offset, u64 *data)
  84{
  85        if (!is_gcr_valid(offset))
  86                return -EINVAL;
  87
  88        spin_lock(&pmc->gcr_lock);
  89        *data = readq(pmc->gcr_mem_base + offset);
  90        spin_unlock(&pmc->gcr_lock);
  91
  92        return 0;
  93}
  94EXPORT_SYMBOL_GPL(intel_pmc_gcr_read64);
  95
  96/**
  97 * intel_pmc_gcr_update() - Update PMC GCR register bits
  98 * @pmc: PMC device pointer
  99 * @offset: offset of GCR register from GCR address base
 100 * @mask: bit mask for update operation
 101 * @val: update value
 102 *
 103 * Updates the bits of given GCR register as specified by
 104 * @mask and @val.
 105 *
 106 * Return: Negative value on error or 0 on success.
 107 */
 108int intel_pmc_gcr_update(struct intel_pmc_dev *pmc, u32 offset, u32 mask, u32 val)
 109{
 110        u32 new_val;
 111
 112        if (!is_gcr_valid(offset))
 113                return -EINVAL;
 114
 115        spin_lock(&pmc->gcr_lock);
 116        new_val = readl(pmc->gcr_mem_base + offset);
 117
 118        new_val = (new_val & ~mask) | (val & mask);
 119        writel(new_val, pmc->gcr_mem_base + offset);
 120
 121        new_val = readl(pmc->gcr_mem_base + offset);
 122        spin_unlock(&pmc->gcr_lock);
 123
 124        /* Check whether the bit update is successful */
 125        return (new_val & mask) != (val & mask) ? -EIO : 0;
 126}
 127EXPORT_SYMBOL_GPL(intel_pmc_gcr_update);
 128
 129/**
 130 * intel_pmc_s0ix_counter_read() - Read S0ix residency
 131 * @pmc: PMC device pointer
 132 * @data: Out param that contains current S0ix residency count.
 133 *
 134 * Writes to @data how many usecs the system has been in low-power S0ix
 135 * state.
 136 *
 137 * Return: An error code or 0 on success.
 138 */
 139int intel_pmc_s0ix_counter_read(struct intel_pmc_dev *pmc, u64 *data)
 140{
 141        u64 deep, shlw;
 142
 143        spin_lock(&pmc->gcr_lock);
 144        deep = readq(pmc->gcr_mem_base + PMC_GCR_TELEM_DEEP_S0IX_REG);
 145        shlw = readq(pmc->gcr_mem_base + PMC_GCR_TELEM_SHLW_S0IX_REG);
 146        spin_unlock(&pmc->gcr_lock);
 147
 148        *data = S0IX_RESIDENCY_IN_USECS(deep, shlw);
 149        return 0;
 150}
 151EXPORT_SYMBOL_GPL(intel_pmc_s0ix_counter_read);
 152
 153/**
 154 * simplecmd_store() - Send a simple IPC command
 155 * @dev: Device under the attribute is
 156 * @attr: Attribute in question
 157 * @buf: Buffer holding data to be stored to the attribute
 158 * @count: Number of bytes in @buf
 159 *
 160 * Expects a string with two integers separated with space. These two
 161 * values hold command and subcommand that is send to PMC.
 162 *
 163 * Return: Number number of bytes written (@count) or negative errno in
 164 *         case of error.
 165 */
 166static ssize_t simplecmd_store(struct device *dev, struct device_attribute *attr,
 167                               const char *buf, size_t count)
 168{
 169        struct intel_pmc_dev *pmc = dev_get_drvdata(dev);
 170        struct intel_scu_ipc_dev *scu = pmc->scu;
 171        int subcmd;
 172        int cmd;
 173        int ret;
 174
 175        ret = sscanf(buf, "%d %d", &cmd, &subcmd);
 176        if (ret != 2) {
 177                dev_err(dev, "Invalid values, expected: cmd subcmd\n");
 178                return -EINVAL;
 179        }
 180
 181        ret = intel_scu_ipc_dev_simple_command(scu, cmd, subcmd);
 182        if (ret)
 183                return ret;
 184
 185        return count;
 186}
 187static DEVICE_ATTR_WO(simplecmd);
 188
 189/**
 190 * northpeak_store() - Enable or disable Northpeak
 191 * @dev: Device under the attribute is
 192 * @attr: Attribute in question
 193 * @buf: Buffer holding data to be stored to the attribute
 194 * @count: Number of bytes in @buf
 195 *
 196 * Expects an unsigned integer. Non-zero enables Northpeak and zero
 197 * disables it.
 198 *
 199 * Return: Number number of bytes written (@count) or negative errno in
 200 *         case of error.
 201 */
 202static ssize_t northpeak_store(struct device *dev, struct device_attribute *attr,
 203                               const char *buf, size_t count)
 204{
 205        struct intel_pmc_dev *pmc = dev_get_drvdata(dev);
 206        struct intel_scu_ipc_dev *scu = pmc->scu;
 207        unsigned long val;
 208        int subcmd;
 209        int ret;
 210
 211        ret = kstrtoul(buf, 0, &val);
 212        if (ret)
 213                return ret;
 214
 215        /* Northpeak is enabled if subcmd == 1 and disabled if it is 0 */
 216        if (val)
 217                subcmd = 1;
 218        else
 219                subcmd = 0;
 220
 221        ret = intel_scu_ipc_dev_simple_command(scu, PMC_NORTHPEAK_CTRL, subcmd);
 222        if (ret)
 223                return ret;
 224
 225        return count;
 226}
 227static DEVICE_ATTR_WO(northpeak);
 228
 229static struct attribute *intel_pmc_attrs[] = {
 230        &dev_attr_northpeak.attr,
 231        &dev_attr_simplecmd.attr,
 232        NULL
 233};
 234
 235static const struct attribute_group intel_pmc_group = {
 236        .attrs = intel_pmc_attrs,
 237};
 238
 239static const struct attribute_group *intel_pmc_groups[] = {
 240        &intel_pmc_group,
 241        NULL
 242};
 243
 244static struct resource punit_res[6];
 245
 246static struct mfd_cell punit = {
 247        .name = "intel_punit_ipc",
 248        .resources = punit_res,
 249};
 250
 251static struct itco_wdt_platform_data tco_pdata = {
 252        .name = "Apollo Lake SoC",
 253        .version = 5,
 254        .no_reboot_use_pmc = true,
 255};
 256
 257static struct resource tco_res[2];
 258
 259static const struct mfd_cell tco = {
 260        .name = "iTCO_wdt",
 261        .ignore_resource_conflicts = true,
 262        .resources = tco_res,
 263        .num_resources = ARRAY_SIZE(tco_res),
 264        .platform_data = &tco_pdata,
 265        .pdata_size = sizeof(tco_pdata),
 266};
 267
 268static const struct resource telem_res[] = {
 269        DEFINE_RES_MEM(TELEM_PUNIT_SSRAM_OFFSET, TELEM_SSRAM_SIZE),
 270        DEFINE_RES_MEM(TELEM_PMC_SSRAM_OFFSET, TELEM_SSRAM_SIZE),
 271};
 272
 273static const struct mfd_cell telem = {
 274        .name = "intel_telemetry",
 275        .resources = telem_res,
 276        .num_resources = ARRAY_SIZE(telem_res),
 277};
 278
 279static int intel_pmc_get_tco_resources(struct platform_device *pdev)
 280{
 281        struct resource *res;
 282
 283        if (acpi_has_watchdog())
 284                return 0;
 285
 286        res = platform_get_resource(pdev, IORESOURCE_IO,
 287                                    PLAT_RESOURCE_ACPI_IO_INDEX);
 288        if (!res) {
 289                dev_err(&pdev->dev, "Failed to get IO resource\n");
 290                return -EINVAL;
 291        }
 292
 293        tco_res[0].flags = IORESOURCE_IO;
 294        tco_res[0].start = res->start + TCO_BASE_OFFSET;
 295        tco_res[0].end = tco_res[0].start + TCO_REGS_SIZE - 1;
 296        tco_res[1].flags = IORESOURCE_IO;
 297        tco_res[1].start = res->start + SMI_EN_OFFSET;
 298        tco_res[1].end = tco_res[1].start + SMI_EN_SIZE - 1;
 299
 300        return 0;
 301}
 302
 303static int intel_pmc_get_resources(struct platform_device *pdev,
 304                                   struct intel_pmc_dev *pmc,
 305                                   struct intel_scu_ipc_data *scu_data)
 306{
 307        struct resource gcr_res;
 308        size_t npunit_res = 0;
 309        struct resource *res;
 310        int ret;
 311
 312        scu_data->irq = platform_get_irq_optional(pdev, 0);
 313
 314        res = platform_get_resource(pdev, IORESOURCE_MEM,
 315                                    PLAT_RESOURCE_IPC_INDEX);
 316        if (!res) {
 317                dev_err(&pdev->dev, "Failed to get IPC resource\n");
 318                return -EINVAL;
 319        }
 320
 321        /* IPC registers */
 322        scu_data->mem.flags = res->flags;
 323        scu_data->mem.start = res->start;
 324        scu_data->mem.end = res->start + PLAT_RESOURCE_IPC_SIZE - 1;
 325
 326        /* GCR registers */
 327        gcr_res.flags = res->flags;
 328        gcr_res.start = res->start + PLAT_RESOURCE_GCR_OFFSET;
 329        gcr_res.end = gcr_res.start + PLAT_RESOURCE_GCR_SIZE - 1;
 330
 331        pmc->gcr_mem_base = devm_ioremap_resource(&pdev->dev, &gcr_res);
 332        if (IS_ERR(pmc->gcr_mem_base))
 333                return PTR_ERR(pmc->gcr_mem_base);
 334
 335        /* Only register iTCO watchdog if there is no WDAT ACPI table */
 336        ret = intel_pmc_get_tco_resources(pdev);
 337        if (ret)
 338                return ret;
 339
 340        /* BIOS data register */
 341        res = platform_get_resource(pdev, IORESOURCE_MEM,
 342                                    PLAT_RESOURCE_BIOS_DATA_INDEX);
 343        if (!res) {
 344                dev_err(&pdev->dev, "Failed to get resource of P-unit BIOS data\n");
 345                return -EINVAL;
 346        }
 347        punit_res[npunit_res++] = *res;
 348
 349        /* BIOS interface register */
 350        res = platform_get_resource(pdev, IORESOURCE_MEM,
 351                                    PLAT_RESOURCE_BIOS_IFACE_INDEX);
 352        if (!res) {
 353                dev_err(&pdev->dev, "Failed to get resource of P-unit BIOS interface\n");
 354                return -EINVAL;
 355        }
 356        punit_res[npunit_res++] = *res;
 357
 358        /* ISP data register, optional */
 359        res = platform_get_resource(pdev, IORESOURCE_MEM,
 360                                    PLAT_RESOURCE_ISP_DATA_INDEX);
 361        if (res)
 362                punit_res[npunit_res++] = *res;
 363
 364        /* ISP interface register, optional */
 365        res = platform_get_resource(pdev, IORESOURCE_MEM,
 366                                    PLAT_RESOURCE_ISP_IFACE_INDEX);
 367        if (res)
 368                punit_res[npunit_res++] = *res;
 369
 370        /* GTD data register, optional */
 371        res = platform_get_resource(pdev, IORESOURCE_MEM,
 372                                    PLAT_RESOURCE_GTD_DATA_INDEX);
 373        if (res)
 374                punit_res[npunit_res++] = *res;
 375
 376        /* GTD interface register, optional */
 377        res = platform_get_resource(pdev, IORESOURCE_MEM,
 378                                    PLAT_RESOURCE_GTD_IFACE_INDEX);
 379        if (res)
 380                punit_res[npunit_res++] = *res;
 381
 382        punit.num_resources = npunit_res;
 383
 384        /* Telemetry SSRAM is optional */
 385        res = platform_get_resource(pdev, IORESOURCE_MEM,
 386                                    PLAT_RESOURCE_TELEM_SSRAM_INDEX);
 387        if (res)
 388                pmc->telem_base = res;
 389
 390        return 0;
 391}
 392
 393static int intel_pmc_create_devices(struct intel_pmc_dev *pmc)
 394{
 395        int ret;
 396
 397        if (!acpi_has_watchdog()) {
 398                ret = devm_mfd_add_devices(pmc->dev, PLATFORM_DEVID_AUTO, &tco,
 399                                           1, NULL, 0, NULL);
 400                if (ret)
 401                        return ret;
 402        }
 403
 404        ret = devm_mfd_add_devices(pmc->dev, PLATFORM_DEVID_AUTO, &punit, 1,
 405                                   NULL, 0, NULL);
 406        if (ret)
 407                return ret;
 408
 409        if (pmc->telem_base) {
 410                ret = devm_mfd_add_devices(pmc->dev, PLATFORM_DEVID_AUTO,
 411                                           &telem, 1, pmc->telem_base, 0, NULL);
 412        }
 413
 414        return ret;
 415}
 416
 417static const struct acpi_device_id intel_pmc_acpi_ids[] = {
 418        { "INT34D2" },
 419        { }
 420};
 421MODULE_DEVICE_TABLE(acpi, intel_pmc_acpi_ids);
 422
 423static int intel_pmc_probe(struct platform_device *pdev)
 424{
 425        struct intel_scu_ipc_data scu_data = {};
 426        struct intel_pmc_dev *pmc;
 427        int ret;
 428
 429        pmc = devm_kzalloc(&pdev->dev, sizeof(*pmc), GFP_KERNEL);
 430        if (!pmc)
 431                return -ENOMEM;
 432
 433        pmc->dev = &pdev->dev;
 434        spin_lock_init(&pmc->gcr_lock);
 435
 436        ret = intel_pmc_get_resources(pdev, pmc, &scu_data);
 437        if (ret) {
 438                dev_err(&pdev->dev, "Failed to request resources\n");
 439                return ret;
 440        }
 441
 442        pmc->scu = devm_intel_scu_ipc_register(&pdev->dev, &scu_data);
 443        if (IS_ERR(pmc->scu))
 444                return PTR_ERR(pmc->scu);
 445
 446        platform_set_drvdata(pdev, pmc);
 447
 448        ret = intel_pmc_create_devices(pmc);
 449        if (ret)
 450                dev_err(&pdev->dev, "Failed to create PMC devices\n");
 451
 452        return ret;
 453}
 454
 455static struct platform_driver intel_pmc_driver = {
 456        .probe = intel_pmc_probe,
 457        .driver = {
 458                .name = "intel_pmc_bxt",
 459                .acpi_match_table = intel_pmc_acpi_ids,
 460                .dev_groups = intel_pmc_groups,
 461        },
 462};
 463module_platform_driver(intel_pmc_driver);
 464
 465MODULE_AUTHOR("Mika Westerberg <mika.westerberg@linux.intel.com>");
 466MODULE_AUTHOR("Zha Qipeng <qipeng.zha@intel.com>");
 467MODULE_DESCRIPTION("Intel Broxton PMC driver");
 468MODULE_LICENSE("GPL v2");
 469