/* * QEMU Plugin API - System specific implementations * * This provides the APIs that have a specific system implementation * or are only relevant to system-mode. * * Copyright (C) 2017, Emilio G. Cota * Copyright (C) 2019-2025, Linaro * * SPDX-License-Identifier: GPL-2.0-or-later */ #include "qemu/osdep.h" #include "qemu/main-loop.h" #include "qapi/error.h" #include "migration/blocker.h" #include "hw/boards.h" #include "qemu/plugin-memory.h" #include "qemu/plugin.h" /* * In system mode we cannot trace the binary being executed so the * helpers all return NULL/0. */ const char *qemu_plugin_path_to_binary(void) { return NULL; } uint64_t qemu_plugin_start_code(void) { return 0; } uint64_t qemu_plugin_end_code(void) { return 0; } uint64_t qemu_plugin_entry_code(void) { return 0; } /* * Virtual Memory queries */ static __thread struct qemu_plugin_hwaddr hwaddr_info; struct qemu_plugin_hwaddr *qemu_plugin_get_hwaddr(qemu_plugin_meminfo_t info, uint64_t vaddr) { CPUState *cpu = current_cpu; unsigned int mmu_idx = get_mmuidx(info); enum qemu_plugin_mem_rw rw = get_plugin_meminfo_rw(info); hwaddr_info.is_store = (rw & QEMU_PLUGIN_MEM_W) != 0; assert(mmu_idx < NB_MMU_MODES); if (!tlb_plugin_lookup(cpu, vaddr, mmu_idx, hwaddr_info.is_store, &hwaddr_info)) { error_report("invalid use of qemu_plugin_get_hwaddr"); return NULL; } return &hwaddr_info; } bool qemu_plugin_hwaddr_is_io(const struct qemu_plugin_hwaddr *haddr) { return haddr->is_io; } uint64_t qemu_plugin_hwaddr_phys_addr(const struct qemu_plugin_hwaddr *haddr) { if (haddr) { return haddr->phys_addr; } return 0; } const char *qemu_plugin_hwaddr_device_name(const struct qemu_plugin_hwaddr *h) { if (h && h->is_io) { MemoryRegion *mr = h->mr; if (!mr->name) { unsigned maddr = (uintptr_t)mr; g_autofree char *temp = g_strdup_printf("anon%08x", maddr); return g_intern_string(temp); } else { return g_intern_string(mr->name); } } else { return g_intern_static_string("RAM"); } } /* * Time control */ static bool has_control; static Error *migration_blocker; const void *qemu_plugin_request_time_control(void) { if (!has_control) { has_control = true; error_setg(&migration_blocker, "TCG plugin time control does not support migration"); migrate_add_blocker(&migration_blocker, NULL); return &has_control; } return NULL; } static void advance_virtual_time__async(CPUState *cpu, run_on_cpu_data data) { int64_t new_time = data.host_ulong; qemu_clock_advance_virtual_time(new_time); } void qemu_plugin_update_ns(const void *handle, int64_t new_time) { if (handle == &has_control) { /* Need to execute out of cpu_exec, so bql can be locked. */ async_run_on_cpu(current_cpu, advance_virtual_time__async, RUN_ON_CPU_HOST_ULONG(new_time)); } }