/* $NetBSD: acpi_cppc.c,v 1.3 2025/01/31 12:29:19 jmcneill Exp $ */

/*-
* Copyright (c) 2020 Jared McNeill <[email protected]>
* All rights reserved.
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions
* are met:
* 1. Redistributions of source code must retain the above copyright
*    notice, this list of conditions and the following disclaimer.
* 2. Redistributions in binary form must reproduce the above copyright
*    notice, this list of conditions and the following disclaimer in the
*    documentation and/or other materials provided with the distribution.
*
* THIS SOFTWARE IS PROVIDED BY THE NETBSD FOUNDATION, INC. AND CONTRIBUTORS
* ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED
* TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
* PURPOSE ARE DISCLAIMED.  IN NO EVENT SHALL THE FOUNDATION OR CONTRIBUTORS
* BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
* CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
* SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
* INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
* CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
* ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
* POSSIBILITY OF SUCH DAMAGE.
*/

/*
* ACPI Collaborative Processor Performance Control support.
*/

#include <sys/cdefs.h>
__KERNEL_RCSID(0, "$NetBSD: acpi_cppc.c,v 1.3 2025/01/31 12:29:19 jmcneill Exp $");

#include <sys/param.h>
#include <sys/bus.h>
#include <sys/cpu.h>
#include <sys/device.h>
#include <sys/kmem.h>
#include <sys/sysctl.h>

#include <dev/acpi/acpireg.h>
#include <dev/acpi/acpivar.h>
#include <dev/acpi/acpi_pcc.h>

#include <external/bsd/acpica/dist/include/amlresrc.h>

/* _CPC package elements */
typedef enum CPCPackageElement {
       CPCNumEntries,
       CPCRevision,
       CPCHighestPerformance,
       CPCNominalPerformance,
       CPCLowestNonlinearPerformance,
       CPCLowestPerformance,
       CPCGuaranteedPerformanceReg,
       CPCDesiredPerformanceReg,
       CPCMinimumPerformanceReg,
       CPCMaximumPerformanceReg,
       CPCPerformanceReductionToleranceReg,
       CPCTimeWindowReg,
       CPCCounterWraparoundTime,
       CPCReferencePerformanceCounterReg,
       CPCDeliveredPerformanceCounterReg,
       CPCPerformanceLimitedReg,
       CPCCPPCEnableReg,
       CPCAutonomousSelectionEnable,
       CPCAutonomousActivityWindowReg,
       CPCEnergyPerformancePreferenceReg,
       CPCReferencePerformance,
       CPCLowestFrequency,
       CPCNominalFrequency,
} CPCPackageElement;

/* PCC command numbers */
#define CPPC_PCC_READ   0x00
#define CPPC_PCC_WRITE  0x01

struct cppc_softc {
       device_t                sc_dev;
       struct cpu_info *       sc_cpuinfo;
       ACPI_HANDLE             sc_handle;
       ACPI_OBJECT *           sc_cpc;
       u_int                   sc_ncpc;

       char *                  sc_available;
       int                     sc_node_target;
       int                     sc_node_current;
       ACPI_INTEGER            sc_max_target;
       ACPI_INTEGER            sc_min_target;
       u_int                   sc_freq_range;
       u_int                   sc_perf_range;
};

static int              cppc_match(device_t, cfdata_t, void *);
static void             cppc_attach(device_t, device_t, void *);

static ACPI_STATUS      cppc_parse_cpc(struct cppc_softc *);
static ACPI_STATUS      cppc_cpufreq_init(struct cppc_softc *);
static ACPI_STATUS      cppc_read(struct cppc_softc *, CPCPackageElement,
                                 ACPI_INTEGER *);
static ACPI_STATUS      cppc_write(struct cppc_softc *, CPCPackageElement,
                                  ACPI_INTEGER);

CFATTACH_DECL_NEW(acpicppc, sizeof(struct cppc_softc),
   cppc_match, cppc_attach, NULL, NULL);

static const struct device_compatible_entry compat_data[] = {
       { .compat = "ACPI0007" },       /* ACPI Processor Device */
       DEVICE_COMPAT_EOL
};

static int
cppc_match(device_t parent, cfdata_t cf, void *aux)
{
       struct acpi_attach_args * const aa = aux;
       ACPI_HANDLE handle;
       ACPI_STATUS rv;

       if (acpi_compatible_match(aa, compat_data) == 0)
               return 0;

       rv = AcpiGetHandle(aa->aa_node->ad_handle, "_CPC", &handle);
       if (ACPI_FAILURE(rv)) {
               return 0;
       }

       if (acpi_match_cpu_handle(aa->aa_node->ad_handle) == NULL) {
               return 0;
       }

       /* When CPPC and P-states/T-states are both available, prefer CPPC */
       return ACPI_MATCHSCORE_CID_MAX + 1;
}

static void
cppc_attach(device_t parent, device_t self, void *aux)
{
       struct cppc_softc * const sc = device_private(self);
       struct acpi_attach_args * const aa = aux;
       ACPI_HANDLE handle = aa->aa_node->ad_handle;
       struct cpu_info *ci;
       ACPI_STATUS rv;

       ci = acpi_match_cpu_handle(handle);
       KASSERT(ci != NULL);

       aprint_naive("\n");
       aprint_normal(": Processor Performance Control (%s)\n", cpu_name(ci));

       sc->sc_dev = self;
       sc->sc_cpuinfo = ci;
       sc->sc_handle = handle;

       rv = cppc_parse_cpc(sc);
       if (ACPI_FAILURE(rv)) {
               aprint_error_dev(self, "failed to parse CPC package: %s\n",
                   AcpiFormatException(rv));
               return;
       }

       cppc_cpufreq_init(sc);
}

/*
* cppc_parse_cpc --
*
*      Read and verify the contents of the _CPC package.
*/
static ACPI_STATUS
cppc_parse_cpc(struct cppc_softc *sc)
{
       ACPI_BUFFER buf;
       ACPI_STATUS rv;

       buf.Pointer = NULL;
       buf.Length = ACPI_ALLOCATE_BUFFER;
       rv = AcpiEvaluateObjectTyped(sc->sc_handle, "_CPC", NULL, &buf,
           ACPI_TYPE_PACKAGE);
       if (ACPI_FAILURE(rv)) {
               return rv;
       }

       sc->sc_cpc = (ACPI_OBJECT *)buf.Pointer;
       if (sc->sc_cpc->Package.Count == 0) {
               return AE_NOT_EXIST;
       }
       if (sc->sc_cpc->Package.Elements[CPCNumEntries].Type !=
           ACPI_TYPE_INTEGER) {
               return AE_TYPE;
       }
       sc->sc_ncpc =
           sc->sc_cpc->Package.Elements[CPCNumEntries].Integer.Value;

       return AE_OK;
}

/*
* cppc_perf_to_freq, cppc_freq_to_perf --
*
*      Convert between abstract performance values and CPU frequencies,
*      when possible.
*/
static ACPI_INTEGER
cppc_perf_to_freq(struct cppc_softc *sc, ACPI_INTEGER perf)
{
       return howmany(perf * sc->sc_freq_range, sc->sc_perf_range);
}
static ACPI_INTEGER
cppc_freq_to_perf(struct cppc_softc *sc, ACPI_INTEGER freq)
{
       return howmany(freq * sc->sc_perf_range, sc->sc_freq_range);
}

/*
* cppc_cpufreq_sysctl --
*
*      sysctl helper function for machdep.cpu.cpuN.{target,current}
*      nodes.
*/
static int
cppc_cpufreq_sysctl(SYSCTLFN_ARGS)
{
       struct cppc_softc * const sc = rnode->sysctl_data;
       struct sysctlnode node;
       u_int fq, oldfq = 0;
       ACPI_INTEGER val;
       ACPI_STATUS rv;
       int error;

       node = *rnode;
       node.sysctl_data = &fq;

       if (rnode->sysctl_num == sc->sc_node_target) {
               rv = cppc_read(sc, CPCDesiredPerformanceReg, &val);
       } else {
               /*
                * XXX We should measure the delivered performance and
                *     report it here. For now, just report the desired
                *     performance level.
                */
               rv = cppc_read(sc, CPCDesiredPerformanceReg, &val);
       }
       if (ACPI_FAILURE(rv)) {
               return EIO;
       }
       if (val > UINT32_MAX) {
               return ERANGE;
       }
       fq = (u_int)cppc_perf_to_freq(sc, val);

       if (rnode->sysctl_num == sc->sc_node_target) {
               oldfq = fq;
       }

       error = sysctl_lookup(SYSCTLFN_CALL(&node));
       if (error != 0 || newp == NULL) {
               return error;
       }

       if (fq == oldfq || rnode->sysctl_num != sc->sc_node_target) {
               return 0;
       }

       if (fq < sc->sc_min_target || fq > sc->sc_max_target) {
               return EINVAL;
       }

       rv = cppc_write(sc, CPCDesiredPerformanceReg,
           cppc_freq_to_perf(sc, fq));
       if (ACPI_FAILURE(rv)) {
               return EIO;
       }

       return 0;
}

/*
* cppc_cpufreq_init --
*
*      Create sysctl machdep.cpu.cpuN.* sysctl tree.
*/
static ACPI_STATUS
cppc_cpufreq_init(struct cppc_softc *sc)
{
       static CPCPackageElement perf_regs[4] = {
               CPCHighestPerformance,
               CPCNominalPerformance,
               CPCLowestNonlinearPerformance,
               CPCLowestPerformance
       };
       ACPI_INTEGER perf[4], min_freq = 0, nom_freq = 0, last;
       const struct sysctlnode *node, *cpunode;
       struct sysctllog *log = NULL;
       struct cpu_info *ci = sc->sc_cpuinfo;
       ACPI_STATUS rv;
       int error, i, n;

       /*
        * Read optional nominal and lowest frequencies. These are used,
        * when present, to scale units for display in the sysctl interface.
        */
       cppc_read(sc, CPCLowestFrequency, &min_freq);
       cppc_read(sc, CPCNominalFrequency, &nom_freq);
       /*
        * Read highest, nominal, lowest nonlinear, and lowest performance
        * levels.
        */
       for (i = 0, n = 0; i < __arraycount(perf_regs); i++) {
               rv = cppc_read(sc, perf_regs[i], &perf[i]);
               if (ACPI_FAILURE(rv)) {
                       return rv;
               }
       }
       if (min_freq && nom_freq) {
               sc->sc_freq_range = nom_freq - min_freq;
               sc->sc_perf_range = perf[1] - perf[3];
       } else {
               sc->sc_freq_range = 1;
               sc->sc_perf_range = 1;
       }

       /*
        * Build a list of performance levels for the
        * machdep.cpufreq.cpuN.available sysctl.
        */
       sc->sc_available = kmem_zalloc(
           strlen("########## ") * __arraycount(perf_regs), KM_SLEEP);
       last = 0;
       for (i = 0, n = 0; i < __arraycount(perf_regs); i++) {
               rv = cppc_read(sc, perf_regs[i], &perf[i]);
               if (ACPI_FAILURE(rv)) {
                       return rv;
               }
               if (perf[i] != last) {
                       char buf[12];
                       snprintf(buf, sizeof(buf), n ? " %u" : "%u",
                           (u_int)cppc_perf_to_freq(sc, perf[i]));
                       strcat(sc->sc_available, buf);
                       last = perf[i];
                       n++;
               }
       }
       sc->sc_max_target = cppc_perf_to_freq(sc, perf[0]);
       sc->sc_min_target = cppc_perf_to_freq(sc, perf[3]);

       error = sysctl_createv(&log, 0, NULL, &node,
           CTLFLAG_PERMANENT, CTLTYPE_NODE, "machdep", NULL,
           NULL, 0, NULL, 0, CTL_MACHDEP, CTL_EOL);
       if (error != 0) {
               goto sysctl_failed;
       }

       error = sysctl_createv(&log, 0, &node, &node,
           0, CTLTYPE_NODE, "cpufreq", NULL,
           NULL, 0, NULL, 0, CTL_CREATE, CTL_EOL);
       if (error != 0) {
               goto sysctl_failed;
       }

       error = sysctl_createv(&log, 0, &node, &cpunode,
           0, CTLTYPE_NODE, cpu_name(ci), NULL,
           NULL, 0, NULL, 0, CTL_CREATE, CTL_EOL);
       if (error != 0) {
               goto sysctl_failed;
       }

       error = sysctl_createv(&log, 0, &cpunode, &node,
           CTLFLAG_READWRITE, CTLTYPE_INT, "target", NULL,
           cppc_cpufreq_sysctl, 0, (void *)sc, 0,
           CTL_CREATE, CTL_EOL);
       if (error != 0) {
               goto sysctl_failed;
       }
       sc->sc_node_target = node->sysctl_num;

       error = sysctl_createv(&log, 0, &cpunode, &node,
           CTLFLAG_READONLY, CTLTYPE_INT, "current", NULL,
           cppc_cpufreq_sysctl, 0, (void *)sc, 0,
           CTL_CREATE, CTL_EOL);
       if (error != 0) {
               goto sysctl_failed;
       }
       sc->sc_node_current = node->sysctl_num;

       error = sysctl_createv(&log, 0, &cpunode, &node,
           CTLFLAG_READONLY, CTLTYPE_STRING, "available", NULL,
           NULL, 0, sc->sc_available, 0,
           CTL_CREATE, CTL_EOL);
       if (error != 0) {
               goto sysctl_failed;
       }

       return AE_OK;

sysctl_failed:
       aprint_error_dev(sc->sc_dev, "couldn't create sysctl nodes: %d\n",
           error);
       sysctl_teardown(&log);

       return AE_ERROR;
}

/*
* cppc_read --
*
*      Read a value from the CPC package that contains either an integer
*      or indirect register reference.
*/
static ACPI_STATUS
cppc_read(struct cppc_softc *sc, CPCPackageElement index, ACPI_INTEGER *val)
{
       ACPI_OBJECT *obj;
       ACPI_GENERIC_ADDRESS addr;
       ACPI_STATUS rv;

       if (index >= sc->sc_ncpc) {
               return AE_NOT_EXIST;
       }

       obj = &sc->sc_cpc->Package.Elements[index];
       switch (obj->Type) {
       case ACPI_TYPE_INTEGER:
               *val = obj->Integer.Value;
               return AE_OK;

       case ACPI_TYPE_BUFFER:
               if (obj->Buffer.Length <
                   sizeof(AML_RESOURCE_GENERIC_REGISTER)) {
                       return AE_TYPE;
               }
               memcpy(&addr, obj->Buffer.Pointer +
                   sizeof(AML_RESOURCE_LARGE_HEADER), sizeof(addr));
               if (addr.SpaceId == ACPI_ADR_SPACE_PLATFORM_COMM) {
                       rv = pcc_message(&addr, CPPC_PCC_READ, PCC_READ, val);
               } else {
                       rv = AcpiRead(val, &addr);
               }
               return rv;

       default:
               return AE_SUPPORT;
       }
}

/*
* cppc_write --
*
*      Write a value based on the CPC package to the specified register.
*/
static ACPI_STATUS
cppc_write(struct cppc_softc *sc, CPCPackageElement index, ACPI_INTEGER val)
{
       ACPI_OBJECT *obj;
       ACPI_GENERIC_ADDRESS addr;
       ACPI_STATUS rv;

       if (index >= sc->sc_ncpc) {
               return AE_NOT_EXIST;
       }

       obj = &sc->sc_cpc->Package.Elements[index];
       if (obj->Type != ACPI_TYPE_BUFFER ||
           obj->Buffer.Length < sizeof(AML_RESOURCE_GENERIC_REGISTER)) {
               return AE_TYPE;
       }

       memcpy(&addr, obj->Buffer.Pointer +
           sizeof(AML_RESOURCE_LARGE_HEADER), sizeof(addr));
       if (addr.SpaceId == ACPI_ADR_SPACE_PLATFORM_COMM) {
               rv = pcc_message(&addr, CPPC_PCC_WRITE, PCC_WRITE, &val);
       } else {
               rv = AcpiWrite(val, &addr);
       }

       return rv;
}