| /* |
| * Copyright (c) 2017-2018, ARM Limited and Contributors. All rights reserved. |
| * |
| * SPDX-License-Identifier: BSD-3-Clause |
| */ |
| |
| #include <arch.h> |
| #include <arch_helpers.h> |
| #include <pubsub.h> |
| #include <spe.h> |
| #include <stdbool.h> |
| |
| static inline void psb_csync(void) |
| { |
| /* |
| * The assembler does not yet understand the psb csync mnemonic |
| * so use the equivalent hint instruction. |
| */ |
| __asm__ volatile("hint #17"); |
| } |
| |
| bool spe_supported(void) |
| { |
| uint64_t features; |
| |
| features = read_id_aa64dfr0_el1() >> ID_AA64DFR0_PMS_SHIFT; |
| return (features & ID_AA64DFR0_PMS_MASK) == 1U; |
| } |
| |
| void spe_enable(bool el2_unused) |
| { |
| uint64_t v; |
| |
| if (!spe_supported()) |
| return; |
| |
| if (el2_unused) { |
| /* |
| * MDCR_EL2.TPMS (ARM v8.2): Do not trap statistical |
| * profiling controls to EL2. |
| * |
| * MDCR_EL2.E2PB (ARM v8.2): SPE enabled in Non-secure |
| * state. Accesses to profiling buffer controls at |
| * Non-secure EL1 are not trapped to EL2. |
| */ |
| v = read_mdcr_el2(); |
| v &= ~MDCR_EL2_TPMS; |
| v |= MDCR_EL2_E2PB(MDCR_EL2_E2PB_EL1); |
| write_mdcr_el2(v); |
| } |
| |
| /* |
| * MDCR_EL2.NSPB (ARM v8.2): SPE enabled in Non-secure state |
| * and disabled in secure state. Accesses to SPE registers at |
| * S-EL1 generate trap exceptions to EL3. |
| */ |
| v = read_mdcr_el3(); |
| v |= MDCR_NSPB(MDCR_NSPB_EL1); |
| write_mdcr_el3(v); |
| } |
| |
| void spe_disable(void) |
| { |
| uint64_t v; |
| |
| if (!spe_supported()) |
| return; |
| |
| /* Drain buffered data */ |
| psb_csync(); |
| dsbnsh(); |
| |
| /* Disable profiling buffer */ |
| v = read_pmblimitr_el1(); |
| v &= ~(1ULL << 0); |
| write_pmblimitr_el1(v); |
| isb(); |
| } |
| |
| static void *spe_drain_buffers_hook(const void *arg) |
| { |
| if (!spe_supported()) |
| return (void *)-1; |
| |
| /* Drain buffered data */ |
| psb_csync(); |
| dsbnsh(); |
| |
| return (void *)0; |
| } |
| |
| SUBSCRIBE_TO_EVENT(cm_entering_secure_world, spe_drain_buffers_hook); |