Index: sys/arm64/arm64/cmn600.c =================================================================== --- /dev/null +++ sys/arm64/arm64/cmn600.c @@ -0,0 +1,836 @@ +/*- + * SPDX-License-Identifier: BSD-2-Clause-FreeBSD + * + * Copyright (c) 2021 ARM Ltd + * + * 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 AUTHOR ``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 AUTHOR 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. + * + * $FreeBSD$ + */ + +/* Arm CoreLink CMN-600 Coherent Mesh Network Driver */ + +#include +__FBSDID("$FreeBSD$"); + +#include "opt_acpi.h" + +#include +#include +#include +#include +#include +#include +#include +#include +#include + +#include +#include + +#include +#include + +#include + +#define RD4(sc, r) bus_read_4((sc)->sc_res[0], (r)) +#define RD8(sc, r) bus_read_8((sc)->sc_res[0], (r)) +#define WR4(sc, r, v) bus_write_4((sc)->sc_res[0], (r), (v)) +#define WR8(sc, r, v) bus_write_8((sc)->sc_res[0], (r), (v)) +#define MD4(sc, r, c, s) WR4((sc), (r), RD4((sc), (r)) & ~(c) | (s)) +#define FLD(v, n) (((v) & n ## _MASK) >> n ## _SHIFT) + +static char *cmn600_ids[] = { + "ARMHC600", + NULL +}; + +static struct resource_spec cmn600_res_spec[] = { + { SYS_RES_MEMORY, 0, RF_ACTIVE }, + { SYS_RES_MEMORY, 1, RF_ACTIVE | RF_UNMAPPED | RF_OPTIONAL }, + { SYS_RES_IRQ, 0, RF_ACTIVE /*| RF_SHAREABLE*/ }, + { -1, 0 } +}; + +struct cmn600_node; + +typedef uint64_t (*nd_read_8_t)(struct cmn600_node *, uint32_t); +typedef uint32_t (*nd_read_4_t)(struct cmn600_node *, uint32_t); +typedef void (*nd_write_8_t)(struct cmn600_node *, uint32_t, uint64_t); +typedef void (*nd_write_4_t)(struct cmn600_node *, uint32_t, uint32_t); + +struct cmn600_node { + struct cmn600_softc *sc; + off_t nd_offset; + int nd_type; + uint16_t nd_id; + uint16_t nd_logical_id; + uint8_t nd_x, nd_y, nd_port, nd_sub; + uint16_t nd_child_count; + uint32_t nd_paired; + struct cmn600_node *nd_parent; + nd_read_8_t nd_read8; + nd_read_4_t nd_read4; + nd_write_8_t nd_write8; + nd_write_4_t nd_write4; + struct cmn600_node **nd_children; +}; + +struct cmn600_softc { + device_t sc_dev; + int sc_unit; + int sc_domain; + int sc_longid; + int sc_mesh_x; + int sc_mesh_y; + struct resource *sc_res[3]; + void *sc_ih; + int sc_r2; + int sc_rev; + struct cmn600_node *sc_rootnode; + struct cmn600_node *sc_dtcnode; + struct cmn600_node *sc_dvmnode; + struct cmn600_node *sc_xpnodes[64]; + int (*sc_pmu_ih)(struct trapframe *tf, int unit, int i); +}; + +static struct cmn600_pmc cmn600_pmcs[CMN600_UNIT_MAX]; +static int cmn600_npmcs = 0; + +static int cmn600_acpi_detach(device_t dev); +static int cmn600_intr(void *arg); + +static void +cmn600_pmc_register(int unit, void *arg, int domain) +{ + + if (unit >= CMN600_UNIT_MAX) { + /* TODO */ + return; + } + + cmn600_pmcs[unit].arg = arg; + cmn600_pmcs[unit].domain = domain; + cmn600_npmcs++; +} + +static void +cmn600_pmc_unregister(int unit) +{ + + cmn600_pmcs[unit].arg = NULL; + cmn600_npmcs--; +} + +int +cmn600_pmc_nunits(void) +{ + + return (cmn600_npmcs); +} + +int +cmn600_pmc_getunit(int unit, void **arg, int *domain) +{ + + if (unit >= cmn600_npmcs) + return (EINVAL); + if (cmn600_pmcs[unit].arg == NULL) + return (EINVAL); + *arg = cmn600_pmcs[unit].arg; + *domain = cmn600_pmcs[unit].domain; + return (0); +} + +int +pmu_cmn600_rev(void *arg) +{ + struct cmn600_softc *sc; + + sc = (struct cmn600_softc *)arg; + switch (sc->sc_rev) { + case 0x0: + return (0x100); + case 0x1: + return (0x101); + case 0x2: + return (0x102); + case 0x3: + return (0x103); + case 0x4: + return (0x200); + case 0x5: + return (0x300); + case 0x6: + return (0x301); + } + return (0x302); /* Unknown revision. */ +} + +static uint64_t +cmn600_node_read8(struct cmn600_node *nd, uint32_t reg) +{ + + return (RD8(nd->sc, nd->nd_offset + reg)); +} + +static void +cmn600_node_write8(struct cmn600_node *nd, uint32_t reg, uint64_t val) +{ + + WR8(nd->sc, nd->nd_offset + reg, val); +} + +static uint32_t +cmn600_node_read4(struct cmn600_node *nd, uint32_t reg) +{ + + return (RD4(nd->sc, nd->nd_offset + reg)); +} + +static void +cmn600_node_write4(struct cmn600_node *nd, uint32_t reg, uint32_t val) +{ + + WR4(nd->sc, nd->nd_offset + reg, val); +} + +static const char * +cmn600_node_type_str(int type) +{ + +#define NAME_OF(t, n) case NODE_TYPE_ ## t: return n + switch (type) { + NAME_OF(INVALID, ""); + NAME_OF(DVM, "DVM"); + NAME_OF(CFG, "CFG"); + NAME_OF(DTC, "DTC"); + NAME_OF(HN_I, "HN-I"); + NAME_OF(HN_F, "HN-F"); + NAME_OF(XP, "XP"); + NAME_OF(SBSX, "SBSX"); + NAME_OF(RN_I, "RN-I"); + NAME_OF(RN_D, "RN-D"); + NAME_OF(RN_SAM, "RN-SAM"); + NAME_OF(CXRA, "CXRA"); + NAME_OF(CXHA, "CXHA"); + NAME_OF(CXLA, "CXLA"); + default: + return ""; + } +#undef NAME_OF +} + +static const char * +cmn600_xpport_dev_type_str(uint8_t type) +{ + +#define NAME_OF(t, n) case POR_MXP_PX_INFO_DEV_TYPE_ ## t: return n + switch (type) { + NAME_OF(RN_I, "RN-I"); + NAME_OF(RN_D, "RN-D"); + NAME_OF(RN_F_CHIB, "RN-F CHIB"); + NAME_OF(RN_F_CHIB_ESAM, "RN-F CHIB ESAM"); + NAME_OF(RN_F_CHIA, "RN-F CHIA"); + NAME_OF(RN_F_CHIA_ESAM, "RN-F CHIA ESAM"); + NAME_OF(HN_T, "HN-T"); + NAME_OF(HN_I, "HN-I"); + NAME_OF(HN_D, "HN-D"); + NAME_OF(SN_F, "SN-F"); + NAME_OF(SBSX, "SBSX"); + NAME_OF(HN_F, "HN-F"); + NAME_OF(CXHA, "CXHA"); + NAME_OF(CXRA, "CXRA"); + NAME_OF(CXRH, "CXRH"); + default: + return ""; + } +#undef NAME_OF +} + +static void +cmn600_dump_node(struct cmn600_node *node, int lvl) +{ + int i; + + for (i = 0; i < lvl; i++) printf(" "); + printf("%s [%dx%d:%d:%d] id: 0x%x @0x%lx Logical Id: 0x%x", + cmn600_node_type_str(node->nd_type), node->nd_x, node->nd_y, + node->nd_port, node->nd_sub, node->nd_id, node->nd_offset, + node->nd_logical_id); + if (node->nd_child_count > 0) + printf(", Children: %d", node->nd_child_count); + printf("\n"); + if (node->nd_type == NODE_TYPE_XP) + printf("\tPort 0: %s\n\tPort 1: %s\n", + cmn600_xpport_dev_type_str(node->nd_read4(node, + POR_MXP_P0_INFO) & 0x1f), + cmn600_xpport_dev_type_str(node->nd_read4(node, + POR_MXP_P1_INFO) & 0x1f)); +} + +static void +cmn600_dump_node_recursive(struct cmn600_node *node, int lvl) +{ + int i; + + cmn600_dump_node(node, lvl); + for (i = 0; i < node->nd_child_count; i++) { + cmn600_dump_node_recursive(node->nd_children[i], lvl + 1); + } +} + +static void +cmn600_dump_nodes_tree(struct cmn600_softc *sc) +{ + + device_printf(sc->sc_dev, " nodes:\n"); + cmn600_dump_node_recursive(sc->sc_rootnode, 0); +} + +static int +cmn600_sysctl_dump_nodes(SYSCTL_HANDLER_ARGS) +{ + struct cmn600_softc *sc; + uint32_t val; + int err; + + sc = (struct cmn600_softc *)arg1; + val = 0; + err = sysctl_handle_int(oidp, &val, 0, req); + + if (err) + return (err); + + if (val != 0) + cmn600_dump_nodes_tree(sc); + + return (0); +} + +static struct cmn600_node * +cmn600_create_node(struct cmn600_softc *sc, off_t node_offset, + struct cmn600_node *parent, int lvl) +{ + struct cmn600_node *node; + off_t child_offset; + uint64_t val; + int i; + + node = (struct cmn600_node *)malloc(sizeof(struct cmn600_node), + M_DEVBUF, M_WAITOK); + if (node == NULL) + return (NULL); + + node->sc = sc; + node->nd_offset = node_offset; + node->nd_parent = parent; + node->nd_read4 = cmn600_node_read4; + node->nd_read8 = cmn600_node_read8; + node->nd_write4 = cmn600_node_write4; + node->nd_write8 = cmn600_node_write8; + + val = node->nd_read8(node, POR_CFGM_NODE_INFO); + node->nd_type = FLD(val, POR_CFGM_NODE_INFO_NODE_TYPE); + node->nd_id = FLD(val, POR_CFGM_NODE_INFO_NODE_ID); + node->nd_logical_id = FLD(val, POR_CFGM_NODE_INFO_LOGICAL_ID); + + val = node->nd_read8(node, POR_CFGM_CHILD_INFO); + node->nd_child_count = FLD(val, POR_CFGM_CHILD_INFO_CHILD_COUNT); + child_offset = FLD(val, POR_CFGM_CHILD_INFO_CHILD_PTR_OFFSET); + + if (parent == NULL) { + /* Find XP node with Id 8. It have to be last in a row. */ + for (i = 0; i < node->nd_child_count; i++) { + val = node->nd_read8(node, child_offset + (i * 8)); + val &= POR_CFGM_CHILD_POINTER_BASE_MASK; + val = RD8(sc, val + POR_CFGM_NODE_INFO); + + if (FLD(val, POR_CFGM_NODE_INFO_NODE_ID) != 8) + continue; + + sc->sc_mesh_x = FLD(val, POR_CFGM_NODE_INFO_LOGICAL_ID); + sc->sc_mesh_y = node->nd_child_count / sc->sc_mesh_x; + if (bootverbose) + printf("Mesh width X/Y %d/%d\n", sc->sc_mesh_x, + sc->sc_mesh_y); + + if ((sc->sc_mesh_x > 4) || (sc->sc_mesh_y > 4)) + sc->sc_longid = 1; + break; + } + + val = node->nd_read8(node, POR_INFO_GLOBAL); + sc->sc_r2 = (val & POR_INFO_GLOBAL_R2_ENABLE) ? 1 : 0; + val = node->nd_read4(node, POR_CFGM_PERIPH_ID_2_PERIPH_ID_3); + sc->sc_rev = FLD(val, POR_CFGM_PERIPH_ID_2_REV); + if (bootverbose) + printf(" Rev: %d, R2_ENABLE = %s\n", sc->sc_rev, + sc->sc_r2 ? "true" : "false"); + } + node->nd_sub = node->nd_id & 0x3; + node->nd_port = (node->nd_id >> 2) & 0x1; + node->nd_paired = 0; + if (sc->sc_longid == 1) { + node->nd_x = (node->nd_id >> 3) & 0x7; + node->nd_y = (node->nd_id >> 6) & 0x7; + } else { + node->nd_x = (node->nd_id >> 3) & 0x3; + node->nd_y = (node->nd_id >> 5) & 0x3; + } + + if (bootverbose) { + cmn600_dump_node(node, lvl); + } + + node->nd_children = (struct cmn600_node **)mallocarray( + node->nd_child_count, sizeof(struct cmn600_node *), M_DEVBUF, + M_WAITOK); + if (node->nd_children == NULL) + goto FAIL; + for (i = 0; i < node->nd_child_count; i++) { + val = node->nd_read8(node, child_offset + (i * 8)); + node->nd_children[i] = cmn600_create_node(sc, val & + POR_CFGM_CHILD_POINTER_BASE_MASK, node, lvl + 1); + } + switch (node->nd_type) { + case NODE_TYPE_DTC: + sc->sc_dtcnode = node; + break; + case NODE_TYPE_DVM: + sc->sc_dvmnode = node; + break; + case NODE_TYPE_XP: + sc->sc_xpnodes[node->nd_id >> 3] = node; + break; + default: + break; + } + return (node); +FAIL: + free(node, M_DEVBUF); + return (NULL); +} + +static void +cmn600_destroy_node(struct cmn600_node *node) +{ + int i; + + for (i = 0; i < node->nd_child_count; i++) { + if (node->nd_children[i] == NULL) + continue; + cmn600_destroy_node(node->nd_children[i]); + } + free(node->nd_children, M_DEVBUF); + free(node, M_DEVBUF); +} + +static int +cmn600_find_node(struct cmn600_softc *sc, int node_id, int type, + struct cmn600_node **node) +{ + struct cmn600_node *xp, *child; + uint8_t xp_xy; + int i; + + switch (type) { + case NODE_TYPE_INVALID: + return (ENXIO); + case NODE_TYPE_CFG: + *node = sc->sc_rootnode; + return (0); + case NODE_TYPE_DTC: + *node = sc->sc_dtcnode; + return (0); + case NODE_TYPE_DVM: + *node = sc->sc_dvmnode; + return (0); + default: + break; + } + + xp_xy = node_id >> 3; + if (xp_xy >= 64) + return (ENXIO); + if (sc->sc_xpnodes[xp_xy] == NULL) + return (ENOENT); + + switch (type) { + case NODE_TYPE_XP: + *node = sc->sc_xpnodes[xp_xy]; + return (0); + default: + xp = sc->sc_xpnodes[xp_xy]; + for (i = 0; i < xp->nd_child_count; i++) { + child = xp->nd_children[i]; + if (child->nd_id == node_id && child->nd_type == type) { + *node = child; + return (0); + } + } + } + return (ENOENT); +} + +int +pmu_cmn600_alloc_localpmc(void *arg, int nodeid, int node_type, int *counter) +{ + struct cmn600_node *node; + struct cmn600_softc *sc; + uint32_t new, old; + int i, ret; + + sc = (struct cmn600_softc *)arg; + switch (node_type) { + case NODE_TYPE_CXLA: + break; + default: + node_type = NODE_TYPE_XP; + /* Parent XP node has always zero port and device bits. */ + nodeid &= ~0x07; + } + ret = cmn600_find_node(sc, nodeid, node_type, &node); + if (ret != 0) + return (ret); + for (i = 0; i < 4; i++) { + new = old = node->nd_paired; + if (old == 0xf) + return (EBUSY); + if ((old & (1 << i)) != 0) + continue; + new |= 1 << i; + if (atomic_cmpset_32(&node->nd_paired, old, new) != 0) + break; + } + *counter = i; + return (0); +} + +int +pmu_cmn600_free_localpmc(void *arg, int nodeid, int node_type, int counter) +{ + struct cmn600_node *node; + struct cmn600_softc *sc; + uint32_t new, old; + int ret; + + sc = (struct cmn600_softc *)arg; + switch (node_type) { + case NODE_TYPE_CXLA: + break; + default: + node_type = NODE_TYPE_XP; + } + ret = cmn600_find_node(sc, nodeid, node_type, &node); + if (ret != 0) + return (ret); + + do { + new = old = node->nd_paired; + new &= ~(1 << counter); + } while (atomic_cmpset_32(&node->nd_paired, old, new) == 0); + return (0); +} + +uint32_t +pmu_cmn600_rd4(void *arg, int nodeid, int node_type, off_t reg) +{ + struct cmn600_node *node; + struct cmn600_softc *sc; + int ret; + + sc = (struct cmn600_softc *)arg; + ret = cmn600_find_node(sc, nodeid, node_type, &node); + if (ret != 0) + return (0xffffffff); + return (cmn600_node_read4(node, reg)); +} + +int +pmu_cmn600_wr4(void *arg, int nodeid, int node_type, off_t reg, uint32_t val) +{ + struct cmn600_node *node; + struct cmn600_softc *sc; + int ret; + + sc = (struct cmn600_softc *)arg; + ret = cmn600_find_node(sc, nodeid, node_type, &node); + if (ret != 0) + return (ret); + cmn600_node_write4(node, reg, val); + return (0); +} + +uint64_t +pmu_cmn600_rd8(void *arg, int nodeid, int node_type, off_t reg) +{ + struct cmn600_node *node; + struct cmn600_softc *sc; + int ret; + + sc = (struct cmn600_softc *)arg; + ret = cmn600_find_node(sc, nodeid, node_type, &node); + if (ret != 0) + return (0xffffffffffffffffUL); + return (cmn600_node_read8(node, reg)); +} + +int +pmu_cmn600_wr8(void *arg, int nodeid, int node_type, off_t reg, uint64_t val) +{ + struct cmn600_node *node; + struct cmn600_softc *sc; + int ret; + + sc = (struct cmn600_softc *)arg; + ret = cmn600_find_node(sc, nodeid, node_type, &node); + if (ret != 0) + return (ret); + cmn600_node_write8(node, reg, val); + return (0); +} + +int +pmu_cmn600_set8(void *arg, int nodeid, int node_type, off_t reg, uint64_t val) +{ + struct cmn600_node *node; + struct cmn600_softc *sc; + int ret; + + sc = (struct cmn600_softc *)arg; + ret = cmn600_find_node(sc, nodeid, node_type, &node); + if (ret != 0) + return (ret); + cmn600_node_write8(node, reg, cmn600_node_read8(node, reg) | val); + return (0); +} + +int +pmu_cmn600_clr8(void *arg, int nodeid, int node_type, off_t reg, uint64_t val) +{ + struct cmn600_node *node; + struct cmn600_softc *sc; + int ret; + + sc = (struct cmn600_softc *)arg; + ret = cmn600_find_node(sc, nodeid, node_type, &node); + if (ret != 0) + return (ret); + cmn600_node_write8(node, reg, cmn600_node_read8(node, reg) & ~val); + return (0); +} + +int +pmu_cmn600_md8(void *arg, int nodeid, int node_type, off_t reg, uint64_t mask, + uint64_t val) +{ + struct cmn600_node *node; + struct cmn600_softc *sc; + int ret; + + sc = (struct cmn600_softc *)arg; + ret = cmn600_find_node(sc, nodeid, node_type, &node); + if (ret != 0) + return (ret); + cmn600_node_write8(node, reg, (cmn600_node_read8(node, reg) & ~mask) | + val); + return (0); +} + +static int +cmn600_acpi_probe(device_t dev) +{ + int err; + + err = ACPI_ID_PROBE(device_get_parent(dev), dev, cmn600_ids, NULL); + if (err <= 0) + device_set_desc(dev, "Arm CoreLink CMN-600 Coherent Mesh Network"); + + return (err); +} + +static int +cmn600_acpi_attach(device_t dev) +{ + struct sysctl_ctx_list *ctx; + struct sysctl_oid_list *child; + struct cmn600_softc *sc; + int cpu, domain, i, u; + const char *dname; + rman_res_t count, periph_base, rootnode_base; + struct cmn600_node *node; + + dname = device_get_name(dev); + sc = device_get_softc(dev); + sc->sc_dev = dev; + u = device_get_unit(dev); + sc->sc_unit = u; + domain = 0; + + if ((resource_int_value(dname, u, "domain", &domain) == 0 || + bus_get_domain(dev, &domain) == 0) && domain < MAXMEMDOM) { + sc->sc_domain = domain; + } + if (domain == -1) /* NUMA not supported. Use single domain. */ + domain = 0; + sc->sc_domain = domain; + device_printf(dev, "domain=%d\n", sc->sc_domain); + + cpu = CPU_FFS(&cpuset_domain[domain]) - 1; + + i = bus_alloc_resources(dev, cmn600_res_spec, sc->sc_res); + if (i != 0) { + device_printf(dev, "cannot allocate resources for device (%d)\n", + i); + return (i); + } + + bus_get_resource(dev, cmn600_res_spec[0].type, cmn600_res_spec[0].rid, + &periph_base, &count); + bus_get_resource(dev, cmn600_res_spec[1].type, cmn600_res_spec[1].rid, + &rootnode_base, &count); + rootnode_base -= periph_base; + if (bootverbose) + printf("ROOTNODE at %lx x %lx\n", rootnode_base, count); + + sc->sc_rootnode = cmn600_create_node(sc, rootnode_base, NULL, 0); + ctx = device_get_sysctl_ctx(sc->sc_dev); + + child = SYSCTL_CHILDREN(device_get_sysctl_tree(sc->sc_dev)); + SYSCTL_ADD_PROC(ctx, child, OID_AUTO, "dump_nodes", CTLTYPE_INT | + CTLFLAG_RW | CTLFLAG_NEEDGIANT, sc, 0, cmn600_sysctl_dump_nodes, + "U", "Dump CMN-600 nodes tree"); + + node = sc->sc_dtcnode; + if (node == NULL) + return (ENXIO); + + cmn600_pmc_register(sc->sc_unit, (void *)sc, domain); + + node->nd_write8(node, POR_DT_PMCR, 0); + node->nd_write8(node, POR_DT_PMOVSR_CLR, POR_DT_PMOVSR_ALL); + node->nd_write8(node, POR_DT_PMCR, POR_DT_PMCR_OVFL_INTR_EN); + node->nd_write8(node, POR_DT_DTC_CTL, POR_DT_DTC_CTL_DT_EN); + + if (bus_setup_intr(dev, sc->sc_res[2], INTR_TYPE_MISC | INTR_MPSAFE, + cmn600_intr, NULL, sc, &sc->sc_ih)) { + bus_release_resources(dev, cmn600_res_spec, sc->sc_res); + device_printf(dev, "cannot setup interrupt handler\n"); + cmn600_acpi_detach(dev); + return (ENXIO); + } + if (bus_bind_intr(dev, sc->sc_res[2], cpu)) { + bus_teardown_intr(dev, sc->sc_res[2], sc->sc_ih); + bus_release_resources(dev, cmn600_res_spec, sc->sc_res); + device_printf(dev, "cannot setup interrupt handler\n"); + cmn600_acpi_detach(dev); + return (ENXIO); + } + return (0); +} + +static int +cmn600_acpi_detach(device_t dev) +{ + struct cmn600_softc *sc; + struct cmn600_node *node; + + sc = device_get_softc(dev); + if (sc->sc_res[2] != NULL) { + bus_teardown_intr(dev, sc->sc_res[2], sc->sc_ih); + } + + node = sc->sc_dtcnode; + node->nd_write4(node, POR_DT_DTC_CTL, + node->nd_read4(node, POR_DT_DTC_CTL) & ~POR_DT_DTC_CTL_DT_EN); + node->nd_write8(node, POR_DT_PMOVSR_CLR, POR_DT_PMOVSR_ALL); + + cmn600_pmc_unregister(sc->sc_unit); + cmn600_destroy_node(sc->sc_rootnode); + bus_release_resources(dev, cmn600_res_spec, sc->sc_res); + + return (0); +} + +int +cmn600_pmu_intr_cb(void *arg, int (*handler)(struct trapframe *tf, int unit, + int i)) +{ + struct cmn600_softc *sc; + + sc = (struct cmn600_softc *) arg; + sc->sc_pmu_ih = handler; + return (0); +} + +static int +cmn600_intr(void *arg) +{ + struct cmn600_node *node; + struct cmn600_softc *sc; + struct trapframe *tf; + uint64_t mask, ready, val; + int i; + + tf = PCPU_GET(curthread)->td_intr_frame; + sc = (struct cmn600_softc *) arg; + node = sc->sc_dtcnode; + val = node->nd_read8(node, POR_DT_PMOVSR); + if (val & POR_DT_PMOVSR_CYCLE_COUNTER) + node->nd_write8(node, POR_DT_PMOVSR_CLR, + POR_DT_PMOVSR_CYCLE_COUNTER); + if (val & POR_DT_PMOVSR_EVENT_COUNTERS) { + for (ready = 0, i = 0; i < 8; i++) { + mask = 1 << i; + if ((val & mask) == 0) + continue; + if (sc->sc_pmu_ih != NULL) + sc->sc_pmu_ih(tf, sc->sc_unit, i); + ready |= mask; + + } + node->nd_write8(node, POR_DT_PMOVSR_CLR, ready); + } + + return (FILTER_HANDLED); +} + +static device_method_t cmn600_acpi_methods[] = { + /* Device interface */ + DEVMETHOD(device_probe, cmn600_acpi_probe), + DEVMETHOD(device_attach, cmn600_acpi_attach), + DEVMETHOD(device_detach, cmn600_acpi_detach), + + /* End */ + DEVMETHOD_END +}; + +static driver_t cmn600_acpi_driver = { + "cmn600", + cmn600_acpi_methods, + sizeof(struct cmn600_softc), +}; +static devclass_t cmn600_acpi_devclass; + +DRIVER_MODULE(cmn600, acpi, cmn600_acpi_driver, cmn600_acpi_devclass, 0, 0); +MODULE_VERSION(cmn600, 1); Index: sys/dev/hwpmc/hwpmc_cmn600.c =================================================================== --- /dev/null +++ sys/dev/hwpmc/hwpmc_cmn600.c @@ -0,0 +1,835 @@ +/*- + * SPDX-License-Identifier: BSD-2-Clause-FreeBSD + * + * Copyright (c) 2003-2008 Joseph Koshy + * Copyright (c) 2007 The FreeBSD Foundation + * Copyright (c) 2021 ARM Ltd + * + * Portions of this software were developed by A. Joseph Koshy under + * sponsorship from the FreeBSD Foundation and Google, Inc. + * + * 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 AUTHOR 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 AUTHOR 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. + */ + +/* Arm CoreLink CMN-600 Coherent Mesh Network PMU Driver */ + +#include +__FBSDID("$FreeBSD$"); + +#include +#include +#include +#include +#include +#include +#include +#include + +#include + +struct cmn600_descr { + struct pmc_descr pd_descr; /* "base class" */ + void *pd_rw_arg; /* Argument to use with read/write */ + struct pmc *pd_pmc; + struct pmc_hw *pd_phw; + uint32_t pd_config; + uint32_t pd_occupancy; + uint32_t pd_nodeid; + int32_t pd_node_type; + int pd_local_counter; + +}; + +static struct cmn600_descr **cmn600_pmcdesc[2]; + +static struct cmn600_pmc cmn600_pmcs[CMN600_UNIT_MAX]; +static int cmn600_npmcs = 0; + +static inline struct cmn600_descr * +cmn600desc(int cpu, int ri) +{ + + return (cmn600_pmcdesc[0][ri]); +} + +static inline int +class_ri2unit(int ri) +{ + + return (ri / CMN600_COUNTERS_N); +} + +static uint64_t +cmn600_pmu_readcntr(void *arg, u_int nodeid, u_int xpcntr, u_int dtccntr, + u_int width) +{ + uint64_t dtcval, xpval; + + KASSERT(xpcntr < 4, ("[cmn600,%d] XP counter number %d is too big." + " Max: 3", __LINE__, xpcntr)); + KASSERT(dtccntr < 8, ("[cmn600,%d] Global counter number %d is too" + " big. Max: 7", __LINE__, dtccntr)); + dtcval = pmu_cmn600_rd8(arg, nodeid, NODE_TYPE_DTC, + POR_DT_PMEVCNT(dtccntr >> 1)); + if (width == 4) { + dtcval = (dtccntr & 1) ? (dtcval >> 16) : (dtcval << 16); + dtcval &= 0xffffffff0000UL; + } + xpval = pmu_cmn600_rd8(arg, nodeid, NODE_TYPE_XP, POR_DTM_PMEVCNT); + xpval >>= xpcntr * 16; + xpval &= 0xffffUL; + return (dtcval | xpval); +} + +static void +cmn600_pmu_writecntr(void *arg, u_int nodeid, u_int xpcntr, u_int dtccntr, + u_int width, uint64_t val) +{ + int shift; + + KASSERT(xpcntr < 4, ("[cmn600,%d] XP counter number %d is too big." + " Max: 3", __LINE__, xpcntr)); + KASSERT(dtccntr < 8, ("[cmn600,%d] Global counter number %d is too" + " big. Max: 7", __LINE__, dtccntr)); + + if (width == 4) { + shift = (dtccntr & 1) ? 32 : 0; + pmu_cmn600_md8(arg, nodeid, NODE_TYPE_DTC, + POR_DT_PMEVCNT(dtccntr >> 1), 0xffffffffUL << shift, + ((val >> 16) & 0xffffffff) << shift); + } else + pmu_cmn600_wr8(arg, nodeid, NODE_TYPE_DTC, + POR_DT_PMEVCNT(dtccntr & ~0x1), val >> 16); + + shift = xpcntr * 16; + val &= 0xffffUL; + pmu_cmn600_md8(arg, nodeid, NODE_TYPE_XP, POR_DTM_PMEVCNT, + 0xffffUL << shift, val << shift); +} + +/* + * read a pmc register + */ +static int +cmn600_read_pmc(int cpu, int ri, pmc_value_t *v) +{ + int counter, local_counter, nodeid, unit; + struct cmn600_descr *desc; + enum pmc_mode mode; + struct pmc *pm; + void *arg; + + KASSERT(cpu >= 0 && cpu < pmc_cpu_max(), + ("[cmn600,%d] illegal CPU value %d", __LINE__, cpu)); + KASSERT(ri >= 0, ("[cmn600,%d] row-index %d out of range", __LINE__, + ri)); + + counter = ri % CMN600_COUNTERS_N; + unit = class_ri2unit(ri); + desc = cmn600desc(cpu, ri); + pm = desc->pd_phw->phw_pmc; + arg = desc->pd_rw_arg; + nodeid = pm->pm_md.pm_cmn600.pm_cmn600_nodeid; + local_counter = pm->pm_md.pm_cmn600.pm_cmn600_local_counter; + + KASSERT(pm != NULL, + ("[cmn600,%d] No owner for HWPMC [cpu%d,pmc%d]", __LINE__, + cpu, ri)); + + mode = PMC_TO_MODE(pm); + + *v = cmn600_pmu_readcntr(arg, nodeid, local_counter, counter, 4); + PMCDBG3(MDP, REA, 2, "%s id=%d -> %jd", __func__, ri, *v); + + return 0; +} + +/* + * Write a pmc register. + */ +static int +cmn600_write_pmc(int cpu, int ri, pmc_value_t v) +{ + int counter, local_counter, nodeid, unit; + struct cmn600_descr *desc; + enum pmc_mode mode; + struct pmc *pm; + void *arg; + + KASSERT(cpu >= 0 && cpu < pmc_cpu_max(), + ("[cmn600,%d] illegal CPU value %d", __LINE__, cpu)); + KASSERT(ri >= 0, ("[cmn600,%d] row-index %d out of range", __LINE__, + ri)); + + counter = ri % CMN600_COUNTERS_N; + unit = class_ri2unit(ri); + desc = cmn600desc(cpu, ri); + pm = desc->pd_phw->phw_pmc; + arg = desc->pd_rw_arg; + nodeid = pm->pm_md.pm_cmn600.pm_cmn600_nodeid; + local_counter = pm->pm_md.pm_cmn600.pm_cmn600_local_counter; + + KASSERT(pm != NULL, + ("[cmn600,%d] PMC not owned (cpu%d,pmc%d)", __LINE__, + cpu, ri)); + + mode = PMC_TO_MODE(pm); + + PMCDBG4(MDP, WRI, 1, "%s cpu=%d ri=%d v=%jx", __func__, cpu, ri, v); + + cmn600_pmu_writecntr(arg, nodeid, local_counter, counter, 4, v); + return (0); +} + +/* + * configure hardware pmc according to the configuration recorded in + * pmc 'pm'. + */ +static int +cmn600_config_pmc(int cpu, int ri, struct pmc *pm) +{ + struct pmc_hw *phw; + + PMCDBG4(MDP, CFG, 1, "%s cpu=%d ri=%d pm=%p", __func__, cpu, ri, pm); + + KASSERT(cpu >= 0 && cpu < pmc_cpu_max(), + ("[cmn600,%d] illegal CPU value %d", __LINE__, cpu)); + KASSERT(ri >= 0, ("[cmn600,%d] row-index %d out of range", __LINE__, + ri)); + + phw = cmn600desc(cpu, ri)->pd_phw; + + KASSERT(pm == NULL || phw->phw_pmc == NULL, + ("[cmn600,%d] pm=%p phw->pm=%p hwpmc not unconfigured", + __LINE__, pm, phw->phw_pmc)); + + phw->phw_pmc = pm; + return (0); +} + +/* + * Retrieve a configured PMC pointer from hardware state. + */ +static int +cmn600_get_config(int cpu, int ri, struct pmc **ppm) +{ + + *ppm = cmn600desc(cpu, ri)->pd_phw->phw_pmc; + + return (0); +} + +#define CASE_DN_VER_EVT(n, id) case PMC_EV_CMN600_PMU_ ## n: { *event = id; \ + return (0); } +static int +cmn600_map_ev2event(int ev, int rev, int *node_type, uint8_t *event) +{ + if (ev < PMC_EV_CMN600_PMU_dn_rxreq_dvmop || + ev > PMC_EV_CMN600_PMU_rni_rdb_ord) + return (EINVAL); + if (ev <= PMC_EV_CMN600_PMU_dn_rxreq_trk_full) { + *node_type = NODE_TYPE_DVM; + if (rev < 0x200) { + switch (ev) { + CASE_DN_VER_EVT(dn_rxreq_dvmop, 1); + CASE_DN_VER_EVT(dn_rxreq_dvmsync, 2); + CASE_DN_VER_EVT(dn_rxreq_dvmop_vmid_filtered, 3); + CASE_DN_VER_EVT(dn_rxreq_retried, 4); + CASE_DN_VER_EVT(dn_rxreq_trk_occupancy, 5); + } + } else { + switch (ev) { + CASE_DN_VER_EVT(dn_rxreq_tlbi_dvmop, 0x01); + CASE_DN_VER_EVT(dn_rxreq_bpi_dvmop, 0x02); + CASE_DN_VER_EVT(dn_rxreq_pici_dvmop, 0x03); + CASE_DN_VER_EVT(dn_rxreq_vivi_dvmop, 0x04); + CASE_DN_VER_EVT(dn_rxreq_dvmsync, 0x05); + CASE_DN_VER_EVT(dn_rxreq_dvmop_vmid_filtered, 0x06); + CASE_DN_VER_EVT(dn_rxreq_dvmop_other_filtered, 0x07); + CASE_DN_VER_EVT(dn_rxreq_retried, 0x08); + CASE_DN_VER_EVT(dn_rxreq_snp_sent, 0x09); + CASE_DN_VER_EVT(dn_rxreq_snp_stalled, 0x0a); + CASE_DN_VER_EVT(dn_rxreq_trk_full, 0x0b); + CASE_DN_VER_EVT(dn_rxreq_trk_occupancy, 0x0c); + } + } + return (EINVAL); + } else if (ev <= PMC_EV_CMN600_PMU_hnf_snp_fwded) { + *node_type = NODE_TYPE_HN_F; + *event = ev - PMC_EV_CMN600_PMU_hnf_cache_miss; + return (0); + } else if (ev <= PMC_EV_CMN600_PMU_hni_pcie_serialization) { + *node_type = NODE_TYPE_HN_I; + *event = ev - PMC_EV_CMN600_PMU_hni_rrt_rd_occ_cnt_ovfl; + return (0); + } else if (ev <= PMC_EV_CMN600_PMU_xp_partial_dat_flit) { + *node_type = NODE_TYPE_XP; + *event = ev - PMC_EV_CMN600_PMU_xp_txflit_valid; + return (0); + } else if (ev <= PMC_EV_CMN600_PMU_sbsx_txrsp_stall) { + *node_type = NODE_TYPE_SBSX; + *event = ev - PMC_EV_CMN600_PMU_sbsx_rd_req; + return (0); + } else if (ev <= PMC_EV_CMN600_PMU_rnd_rdb_ord) { + *node_type = NODE_TYPE_RN_D; + *event = ev - PMC_EV_CMN600_PMU_rnd_s0_rdata_beats; + return (0); + } else if (ev <= PMC_EV_CMN600_PMU_rni_rdb_ord) { + *node_type = NODE_TYPE_RN_I; + *event = ev - PMC_EV_CMN600_PMU_rni_s0_rdata_beats; + return (0); + } else if (ev <= PMC_EV_CMN600_PMU_cxha_snphaz_occ) { + *node_type = NODE_TYPE_CXHA; + *event = ev - PMC_EV_CMN600_PMU_cxha_rddatbyp; + return (0); + } else if (ev <= PMC_EV_CMN600_PMU_cxra_ext_dat_stall) { + *node_type = NODE_TYPE_CXRA; + *event = ev - PMC_EV_CMN600_PMU_cxra_req_trk_occ; + return (0); + } else if (ev <= PMC_EV_CMN600_PMU_cxla_avg_latency_form_tx_tlp) { + *node_type = NODE_TYPE_CXLA; + *event = ev - PMC_EV_CMN600_PMU_cxla_rx_tlp_link0; + return (0); + } + return (EINVAL); +} + +/* + * Check if a given allocation is feasible. + */ + +static int +cmn600_allocate_pmc(int cpu, int ri, struct pmc *pm, + const struct pmc_op_pmcallocate *a) +{ + struct cmn600_descr *desc; + const struct pmc_descr *pd; + uint64_t caps; + int local_counter, node_type; + enum pmc_event pe; + void *arg; + uint8_t e; + int err; + + (void) cpu; + + KASSERT(cpu >= 0 && cpu < pmc_cpu_max(), + ("[cmn600,%d] illegal CPU value %d", __LINE__, cpu)); + KASSERT(ri >= 0, ("[cmn600,%d] row-index %d out of range", __LINE__, + ri)); + + desc = cmn600desc(cpu, ri); + arg = desc->pd_rw_arg; + pd = &desc->pd_descr; + if (cmn600_pmcs[class_ri2unit(ri)].domain != pcpu_find(cpu)->pc_domain) + return (EINVAL); + + /* check class match */ + if (pd->pd_class != a->pm_class) + return (EINVAL); + + caps = pm->pm_caps; + + PMCDBG3(MDP, ALL, 1, "%s ri=%d caps=0x%x", __func__, ri, caps); + + if ((pd->pd_caps & caps) != caps) + return (EPERM); + + pe = a->pm_ev; + err = cmn600_map_ev2event(pe, pmu_cmn600_rev(arg), &node_type, &e); + if (err != 0) + return (err); + err = pmu_cmn600_alloc_localpmc(arg, + a->pm_md.pm_cmn600.pma_cmn600_nodeid, node_type, &local_counter); + if (err != 0) + return (err); + + pm->pm_md.pm_cmn600.pm_cmn600_config = + a->pm_md.pm_cmn600.pma_cmn600_config; + pm->pm_md.pm_cmn600.pm_cmn600_occupancy = + a->pm_md.pm_cmn600.pma_cmn600_occupancy; + desc->pd_nodeid = pm->pm_md.pm_cmn600.pm_cmn600_nodeid = + a->pm_md.pm_cmn600.pma_cmn600_nodeid; + desc->pd_node_type = pm->pm_md.pm_cmn600.pm_cmn600_node_type = + node_type; + pm->pm_md.pm_cmn600.pm_cmn600_event = e; + desc->pd_local_counter = pm->pm_md.pm_cmn600.pm_cmn600_local_counter = + local_counter; + + PMCDBG3(MDP, ALL, 2, "%s ri=%d -> control=0x%x", __func__, ri, control); + + return (0); +} + +/* + * Release machine dependent state associated with a PMC. This is a + * no-op on this architecture. + * + */ + +/* ARGSUSED0 */ +static int +cmn600_release_pmc(int cpu, int ri, struct pmc *pmc) +{ + struct cmn600_descr *desc; + struct pmc_hw *phw; + struct pmc *pm; + int err; + + (void) pmc; + + KASSERT(cpu >= 0 && cpu < pmc_cpu_max(), + ("[cmn600,%d] illegal CPU value %d", __LINE__, cpu)); + KASSERT(ri >= 0, ("[cmn600,%d] row-index %d out of range", __LINE__, + ri)); + + desc = cmn600desc(cpu, ri); + phw = desc->pd_phw; + pm = phw->phw_pmc; + err = pmu_cmn600_free_localpmc(desc->pd_rw_arg, desc->pd_nodeid, + desc->pd_node_type, desc->pd_local_counter); + if (err != 0) + return (err); + + KASSERT(pm == NULL, ("[cmn600,%d] PHW pmc %p non-NULL", __LINE__, pm)); + + return (0); +} + +static inline uint64_t +cmn600_encode_source(int node_type, int counter, int port, int sub) +{ + if (node_type == NODE_TYPE_XP) + return (0x4 | counter); + + return (((port + 1) << 4) | (sub << 2) | counter); +} + +/* + * start a PMC. + */ + +static int +cmn600_start_pmc(int cpu, int ri) +{ + int counter, local_counter, node_type, shift, unit; + uint64_t config, occupancy, source, xp_pmucfg; + struct cmn600_descr *desc; + struct pmc_hw *phw; + struct pmc *pm; + uint8_t event, port, sub; + uint16_t nodeid; + void *arg; + + KASSERT(cpu >= 0 && cpu < pmc_cpu_max(), + ("[cmn600,%d] illegal CPU value %d", __LINE__, cpu)); + KASSERT(ri >= 0, ("[cmn600,%d] row-index %d out of range", __LINE__, + ri)); + + counter = ri % CMN600_COUNTERS_N; + unit = class_ri2unit(ri); + desc = cmn600desc(cpu, ri); + phw = desc->pd_phw; + pm = phw->phw_pmc; + arg = desc->pd_rw_arg; + + KASSERT(pm != NULL, + ("[cmn600,%d] starting cpu%d,pmc%d with null pmc record", __LINE__, + cpu, ri)); + + PMCDBG3(MDP, STA, 1, "%s cpu=%d ri=%d", __func__, cpu, ri); + + config = pm->pm_md.pm_cmn600.pm_cmn600_config; + occupancy = pm->pm_md.pm_cmn600.pm_cmn600_occupancy; + node_type = pm->pm_md.pm_cmn600.pm_cmn600_node_type; + event = pm->pm_md.pm_cmn600.pm_cmn600_event; + nodeid = pm->pm_md.pm_cmn600.pm_cmn600_nodeid; + local_counter = pm->pm_md.pm_cmn600.pm_cmn600_local_counter; + port = (nodeid >> 2) & 1; + sub = nodeid & 3; + + switch (node_type) { + case NODE_TYPE_DVM: + case NODE_TYPE_HN_F: + case NODE_TYPE_CXHA: + case NODE_TYPE_CXRA: + pmu_cmn600_md8(arg, nodeid, node_type, + CMN600_COMMON_PMU_EVENT_SEL, + CMN600_COMMON_PMU_EVENT_SEL_OCC_MASK, + occupancy << CMN600_COMMON_PMU_EVENT_SEL_OCC_SHIFT); + break; + case NODE_TYPE_XP: + /* Set PC and Interface.*/ + event |= config; + } + + /* 1. HW */ + /* 2. Select event of target node for one of four outputs. */ + pmu_cmn600_md8(arg, nodeid, node_type, CMN600_COMMON_PMU_EVENT_SEL, + 0xff << (local_counter * 8), + event << (local_counter * 8)); + + xp_pmucfg = pmu_cmn600_rd8(arg, nodeid, NODE_TYPE_XP, + POR_DTM_PMU_CONFIG); + /* + * 3. configure XP to connect one of four target node outputs to local + * counter. + */ + source = cmn600_encode_source(node_type, local_counter, port, sub); + shift = (local_counter * POR_DTM_PMU_CONFIG_VCNT_INPUT_SEL_WIDTH) + + POR_DTM_PMU_CONFIG_VCNT_INPUT_SEL_SHIFT; + xp_pmucfg &= ~(0xffUL << shift); + xp_pmucfg |= source << shift; + + /* 4. Pair with global counters A, B, C, ..., H. */ + shift = (local_counter * 4) + 16; + xp_pmucfg &= ~(0xfUL << shift); + xp_pmucfg |= counter << shift; + /* Enable pairing.*/ + xp_pmucfg |= 1 << (local_counter + 4); + + /* 5. Combine local counters 0 with 1, 2 with 3 or all four. */ + xp_pmucfg &= ~0xeUL; + + /* 6. Enable XP's PMU function. */ + xp_pmucfg |= POR_DTM_PMU_CONFIG_PMU_EN; + pmu_cmn600_wr8(arg, nodeid, NODE_TYPE_XP, POR_DTM_PMU_CONFIG, xp_pmucfg); + if (node_type == NODE_TYPE_CXLA) + pmu_cmn600_set8(arg, nodeid, NODE_TYPE_CXLA, + POR_CXG_RA_CFG_CTL, EN_CXLA_PMUCMD_PROP); + + /* 7. Enable DTM. */ + pmu_cmn600_set8(arg, nodeid, NODE_TYPE_XP, POR_DTM_CONTROL, + POR_DTM_CONTROL_DTM_ENABLE); + + /* 8. Reset grouping of global counters. Use 32 bits. */ + pmu_cmn600_clr8(arg, nodeid, NODE_TYPE_DTC, POR_DT_PMCR, + POR_DT_PMCR_CNTCFG_MASK); + + /* 9. Enable DTC. */ + pmu_cmn600_set8(arg, nodeid, NODE_TYPE_DTC, POR_DT_DTC_CTL, + POR_DT_DTC_CTL_DT_EN); + + /* 10. Enable Overflow Interrupt. */ + pmu_cmn600_set8(arg, nodeid, NODE_TYPE_DTC, POR_DT_PMCR, + POR_DT_PMCR_OVFL_INTR_EN); + + /* 11. Run PMC. */ + pmu_cmn600_set8(arg, nodeid, NODE_TYPE_DTC, POR_DT_PMCR, + POR_DT_PMCR_PMU_EN); + + PMCDBG2(MDP, STA, 2, "%s control=0x%x", __func__, control); + + return (0); +} + +/* + * Stop a PMC. + */ + +static int +cmn600_stop_pmc(int cpu, int ri) +{ + struct cmn600_descr *desc; + struct pmc_hw *phw; + struct pmc *pm; + int local_counter; + uint64_t val; + + KASSERT(cpu >= 0 && cpu < pmc_cpu_max(), + ("[cmn600,%d] illegal CPU value %d", __LINE__, cpu)); + KASSERT(ri >= 0, ("[cmn600,%d] row-index %d out of range", __LINE__, + ri)); + + desc = cmn600desc(cpu, ri); + phw = desc->pd_phw; + pm = phw->phw_pmc; + + KASSERT(pm != NULL, + ("[cmn600,%d] cpu%d,pmc%d no PMC to stop", __LINE__, + cpu, ri)); + + PMCDBG2(MDP, STO, 1, "%s ri=%d", __func__, ri); + + /* Disable pairing. */ + local_counter = pm->pm_md.pm_cmn600.pm_cmn600_local_counter; + pmu_cmn600_clr8(desc->pd_rw_arg, pm->pm_md.pm_cmn600.pm_cmn600_nodeid, + NODE_TYPE_XP, POR_DTM_PMU_CONFIG, (1 << (local_counter + 4))); + + /* Shutdown XP's DTM function if no paired counters. */ + val = pmu_cmn600_rd8(desc->pd_rw_arg, + pm->pm_md.pm_cmn600.pm_cmn600_nodeid, NODE_TYPE_XP, + POR_DTM_PMU_CONFIG); + if ((val & 0xf0) == 0) + pmu_cmn600_clr8(desc->pd_rw_arg, + pm->pm_md.pm_cmn600.pm_cmn600_nodeid, NODE_TYPE_XP, + POR_DTM_PMU_CONFIG, POR_DTM_CONTROL_DTM_ENABLE); + + return (0); +} + +/* + * describe a PMC + */ +static int +cmn600_describe(int cpu, int ri, struct pmc_info *pi, struct pmc **ppmc) +{ + struct pmc_hw *phw; + size_t copied; + int error; + + KASSERT(cpu >= 0 && cpu < pmc_cpu_max(), + ("[cmn600,%d] illegal CPU %d", __LINE__, cpu)); + KASSERT(ri >= 0, ("[cmn600,%d] row-index %d out of range", __LINE__, + ri)); + + phw = cmn600desc(cpu, ri)->pd_phw; + + if ((error = copystr(cmn600desc(cpu, ri)->pd_descr.pd_name, + pi->pm_name, PMC_NAME_MAX, &copied)) != 0) + return (error); + + pi->pm_class = cmn600desc(cpu, ri)->pd_descr.pd_class; + + if (phw->phw_state & PMC_PHW_FLAG_IS_ENABLED) { + pi->pm_enabled = TRUE; + *ppmc = phw->phw_pmc; + } else { + pi->pm_enabled = FALSE; + *ppmc = NULL; + } + + return (0); +} + +/* + * processor dependent initialization. + */ + +static int +cmn600_pcpu_init(struct pmc_mdep *md, int cpu) +{ + int first_ri, n, npmc; + struct pmc_hw *phw; + struct pmc_cpu *pc; + int mdep_class; + + mdep_class = PMC_MDEP_CLASS_INDEX_CMN600; + KASSERT(cpu >= 0 && cpu < pmc_cpu_max(), + ("[cmn600,%d] insane cpu number %d", __LINE__, cpu)); + + PMCDBG1(MDP, INI, 1, "cmn600-init cpu=%d", cpu); + + /* + * Set the content of the hardware descriptors to a known + * state and initialize pointers in the MI per-cpu descriptor. + */ + + pc = pmc_pcpu[cpu]; + first_ri = md->pmd_classdep[mdep_class].pcd_ri; + npmc = md->pmd_classdep[mdep_class].pcd_num; + + for (n = 0; n < npmc; n++, phw++) { + phw = cmn600desc(cpu, n)->pd_phw; + phw->phw_state = PMC_PHW_CPU_TO_STATE(cpu) | + PMC_PHW_INDEX_TO_STATE(n); + /* Set enabled only if unit present. */ + if (cmn600_pmcs[class_ri2unit(n)].arg != NULL) + phw->phw_state |= PMC_PHW_FLAG_IS_ENABLED; + phw->phw_pmc = NULL; + pc->pc_hwpmcs[n + first_ri] = phw; + } + return (0); +} + +/* + * processor dependent cleanup prior to the KLD + * being unloaded + */ + +static int +cmn600_pcpu_fini(struct pmc_mdep *md, int cpu) +{ + + return (0); +} + +static int +cmn600_pmu_intr(struct trapframe *tf, int unit, int i) +{ + struct pmc_cpu *pc; + struct pmc_hw *phw; + struct pmc *pm; + int error, cpu, ri; + + ri = i + unit * CMN600_COUNTERS_N; + cpu = curcpu; + KASSERT(cpu >= 0 && cpu < pmc_cpu_max(), + ("[cmn600,%d] CPU %d out of range", __LINE__, cpu)); + pc = pmc_pcpu[cpu]; + KASSERT(pc != NULL, ("pc != NULL")); + + phw = cmn600desc(cpu, ri)->pd_phw; + KASSERT(phw != NULL, ("phw != NULL")); + pm = phw->phw_pmc; + if (pm == NULL) + return (0); + + if (!PMC_IS_SAMPLING_MODE(PMC_TO_MODE(pm))) { + /* Always CPU0. */ + pm->pm_pcpu_state[0].pps_overflowcnt += 1; + return (0); + } + + if (pm->pm_state != PMC_STATE_RUNNING) + return (0); + + error = pmc_process_interrupt(PMC_HR, pm, tf); + if (error) + cmn600_stop_pmc(cpu, ri); + + /* Reload sampling count */ + cmn600_write_pmc(cpu, ri, pm->pm_sc.pm_reloadcount); + + return (0); +} + +/* + * Initialize ourselves. + */ +static int +cmn600_init_pmc_units() +{ + int i; + + if (cmn600_npmcs > 0) { /* Already initialized. */ + return (0); + } + + cmn600_npmcs = cmn600_pmc_nunits(); + if (cmn600_npmcs == 0) + return (ENOENT); + + for (i = 0; i < cmn600_npmcs; i++) { + if (cmn600_pmc_getunit(i, &cmn600_pmcs[i].arg, + &cmn600_pmcs[i].domain) != 0) + cmn600_pmcs[i].arg = NULL; + } + return (0); +} + +int +pmc_cmn600_nclasses() +{ + + if (cmn600_pmc_nunits() > 0) + return (1); + return (0); +} + +int +pmc_cmn600_initialize(struct pmc_mdep *md) +{ + struct pmc_classdep *pcd; + int i, npmc, unit; + + cmn600_init_pmc_units(); + KASSERT(md != NULL, ("[cmn600,%d] md is NULL", __LINE__)); + KASSERT(cmn600_npmcs < CMN600_UNIT_MAX, + ("[cmn600,%d] cmn600_npmcs too big", __LINE__)); + + PMCDBG0(MDP,INI,1, "cmn600-initialize"); + + npmc = CMN600_COUNTERS_N * cmn600_npmcs; + pcd = &md->pmd_classdep[PMC_MDEP_CLASS_INDEX_CMN600]; + + pcd->pcd_caps = PMC_CAP_SYSTEM | PMC_CAP_READ | + PMC_CAP_WRITE | PMC_CAP_QUALIFIER | PMC_CAP_INTERRUPT | + PMC_CAP_DOMWIDE; + pcd->pcd_class = PMC_CLASS_CMN600_PMU; + pcd->pcd_num = npmc; + pcd->pcd_ri = md->pmd_npmc; + pcd->pcd_width = 48; + + pcd->pcd_allocate_pmc = cmn600_allocate_pmc; + pcd->pcd_config_pmc = cmn600_config_pmc; + pcd->pcd_describe = cmn600_describe; + pcd->pcd_get_config = cmn600_get_config; + pcd->pcd_get_msr = NULL; + pcd->pcd_pcpu_fini = cmn600_pcpu_fini; + pcd->pcd_pcpu_init = cmn600_pcpu_init; + pcd->pcd_read_pmc = cmn600_read_pmc; + pcd->pcd_release_pmc = cmn600_release_pmc; + pcd->pcd_start_pmc = cmn600_start_pmc; + pcd->pcd_stop_pmc = cmn600_stop_pmc; + pcd->pcd_write_pmc = cmn600_write_pmc; + + md->pmd_npmc += npmc; + cmn600_pmcdesc[0] = (struct cmn600_descr **)malloc( + sizeof(struct cmn600_descr *) * npmc * CMN600_PMU_DEFAULT_UNITS_N, + M_PMC, M_WAITOK|M_ZERO); + for (i = 0; i < npmc; i++) { + cmn600_pmcdesc[0][i] = (struct cmn600_descr *)malloc( + sizeof(struct cmn600_descr), M_PMC, M_WAITOK|M_ZERO); + + unit = i / CMN600_COUNTERS_N; + KASSERT(unit >= 0, ("unit >= 0")); + KASSERT(cmn600_pmcs[unit].arg != NULL, ("arg != NULL")); + + cmn600_pmcdesc[0][i]->pd_rw_arg = cmn600_pmcs[unit].arg; + cmn600_pmcdesc[0][i]->pd_descr.pd_class = + PMC_CLASS_CMN600_PMU; + cmn600_pmcdesc[0][i]->pd_descr.pd_caps = pcd->pcd_caps; + cmn600_pmcdesc[0][i]->pd_phw = (struct pmc_hw *)malloc( + sizeof(struct pmc_hw), M_PMC, M_WAITOK|M_ZERO); + snprintf(cmn600_pmcdesc[0][i]->pd_descr.pd_name, 63, + "CMN600_%d", i); + cmn600_pmu_intr_cb(cmn600_pmcs[unit].arg, cmn600_pmu_intr); + } + + return (0); +} + +void +pmc_cmn600_finalize(struct pmc_mdep *md) +{ + struct pmc_classdep *pcd; + int i, npmc; + +#if defined(INVARIANTS) + KASSERT(md->pmd_classdep[PMC_MDEP_CLASS_INDEX_CMN600].pcd_class == + PMC_CLASS_CMN600_PMU, ("[cmn600,%d] pmc class mismatch", + __LINE__)); + +#endif + pcd = &md->pmd_classdep[PMC_MDEP_CLASS_INDEX_CMN600]; + + npmc = pcd->pcd_num; + for (i = 0; i < npmc; i++) { + free(cmn600_pmcdesc[0][i]->pd_phw, M_PMC); + free(cmn600_pmcdesc[0][i], M_PMC); + } + free(cmn600_pmcdesc[0], M_PMC); + cmn600_pmcdesc[0] = NULL; +} + +MODULE_DEPEND(pmc, cmn600, 1, 1, 1);